Commit 8fde5e18 authored by Pratap Vardhan's avatar Pratap Vardhan

ADD: LinReg: Visualizing linear relationships

Pipeline #114227 failed
# .editorconfig maintains consistent coding styles between different editors.
# Get plugins at
# - Sublime text:
# - Notepad++:
root = true
# Apply common styles for most standard code files.
# Do not apply to * - that covers binary files as well
end_of_line = lf
insert_final_newline = true
trim_trailing_whitespace = true
charset = utf-8
# Stick to 2-space indenting by default, to conserve space
indent_style = space
indent_size = 2
indent_size = 4
indent_style = tab
indent_size = 4
module.exports = {
"parserOptions": {
"ecmaVersion": 6 // Use ES6 parser. Browsers other than IE support it
"plugins": [
"template" // Handle Tornado templates and JS in HTML files
"env": {
"es6": true, // Allow ES6 in JavaScript
"browser": true, // Include browser globals
"jquery": true, // Include jQuery and $
"mocha": true // Include it(), assert(), etc
"globals": {
"_": true, // underscore.js
"d3": true, // d3.js
"vg": true, // vega.js
"L": true, // leaflet.js
"ga": true, // Google analytics
"g1": true, // g1.min.js
"topojson": true, // topojson.js
"moment": true, // moment.js
"numeral": true, // numeral.js
"assert": true // chai.js
"extends": "eslint:recommended",
"rules": {
/* Override default rules */
"indent": ["off", 2], // We eventually want 2 space indentation
"linebreak-style": ["off", "unix"], // We eventually want UNIX style line
"quotes": ["off", "double"], // We may go for a double-quotes style
"semi": ["off", "never"] // We may go for a no-semicolon style
# Ignore files that should not be committed.
# Do not commit data, passwords, secret information or large files.
# Don't commit the assets/ folder except for the file
# Store them in a shared folder and sync here.
# Ignore npm and bower modules. These should be created by yarn/npm and bower.
# Filenames should NOT have spaces
* *
# Ignore byte-compiled / optimised / DLL files
# Do not commit data files.
# Ignore compressed files
# Avoid documents
# Avoid media files
# Unit test
# Ignore Gramex 0.x artefacts and log files
# Ignore backup files
# Sublime-text workspaces, etc
# IPython Notebook checkpoints
# Windows / Mac OS artefacts
# bash.exe.stackdump on Cygwin
# R history files
# For Linux FUSE file system
# See
- validate v1
# Deploy application to a server
stage: deploy
script: deploy
only: [master]
URL: linreg
SETUP: bash
VERSION: py3v1
PORT: 8050
// Gramex .htmllintrc v1.2
"plugins": [],
"attr-bans": [
// "frameborder", // frameborder is used in YouTube embeds
"style", // DO NOT USE style= attribute
"attr-name-style": false,
"attr-no-dup": false, // attr name may be computed, and get replaced by {}
"attr-no-unsafe-char": false, // title contains single quotes '
"attr-quote-style": "double", // attributes contain double quotes
"attr-req-value": false,
"class-no-dup": true, // no duplicate classes in a tag
"doctype-first": false, // snippet templates need not begin with doctype
"doctype-html5": true,
"fig-req-figcaption": false,
"focusable-tabindex-style": false,
"head-req-title": false, // title may be inside a
"href-style": false,
"html-req-lang": false,
"id-class-ignore-regex": "\\{ *\\}", // ignore tornado template id / class
"id-class-no-ad": false,
"id-class-style": false, // no styles enforced for now
"id-no-dup": false, // template replacement IDs { } cause duplication
"img-req-alt": "allownull", // <img alt=""> needed for dynamic image content
"img-req-src": false,
"indent-style": "spaces",
"indent-width": 2,
"label-req-for": false, // cannot use if multiple forms with same key
"line-end-style": false, // raises too many errors
"raw-ignore-regex": "<%.*?%>\\s*|{[%#{].*?[%#}]}\\s*", // ignore templates
"spec-char-escape": false, // using > or < is not that big a deal
"table-req-caption": false,
"tag-bans": [
// "b", // Bootstrap caret example uses <b>
// "i", // Font-awesome icons use <i>
"s", // avoid strike tag, deprecated
"br", // avoid break tag, bad practice, use margin/padding instead
"style", // Put styles into CSS files
"tag-name-lowercase": true,
"tag-name-match": true,
"tag-self-close": false,
"title-max-len": false, // we sometimes have tables inside the title=""
"title-no-dup": true
"use strict"
module.exports = {
rules: {
"at-rule-no-unknown": true,
"block-no-empty": true,
"color-no-invalid-hex": true,
"comment-no-empty": true,
"declaration-block-no-duplicate-properties": [
ignore: ["consecutive-duplicates-with-different-values"]
"declaration-block-no-shorthand-property-overrides": true,
"font-family-no-duplicate-names": true,
"font-family-no-missing-generic-family-keyword": true,
"function-calc-no-unspaced-operator": true,
"function-linear-gradient-no-nonstandard-direction": true,
"keyframe-declaration-no-important": true,
"media-feature-name-no-unknown": true,
"no-descending-specificity": true,
"no-duplicate-at-import-rules": true,
"no-duplicate-selectors": true,
"no-empty-source": true,
"no-extra-semicolons": true,
"no-invalid-double-slash-comments": true,
"property-no-unknown": true,
"selector-pseudo-class-no-unknown": true,
"selector-pseudo-element-no-unknown": true,
"selector-type-no-unknown": true,
"string-no-newline": true,
"unit-no-unknown": true
# LinReg: Visualizing linear relationships
- Expose sales dataset via formahandler
- Summary visual with groupby metric control
- Linear relationships on numeric columns via seaborn
- ~100 lines of code (HTML, JS, Python) and gramex.yaml setup
## Setup
- [Install Gramex 1.x](
- Clone this repository
- Setup assets -- `bash`
- From the repo folder, run `gramex`
## Contributions
- Pratap Vardhan <>

11.8 KB

# Generated by gramex init 1.58.0 (don't delete this line)
# on 2020-02-24 by Pratap Vardhan <>
# Configurations for app: linreg
# ----------------------------------------------------------------------------
pattern: /$YAMLURL/
handler: FileHandler
path: $YAMLPATH/index.html
# See for auth rules
auth: true
template: true
Cache-Control: private, max-age=1
cache: {expiry: {duration: 1}}
pattern: /$YAMLURL/data
handler: FormHandler
url: $YAMLPATH/assets/data/Sample-Superstore.csv
format: sns
chart: jointplot
kind: reg
ext: png
x: '{x}'
y: '{y}'
dpi: 72
width: '{width}'
height: '{height}'
Content-Type: image/png
pattern: /$YAMLURL/datagroup
handler: FormHandler
url: $YAMLPATH/assets/data/Sample-Superstore.csv
modify:, handler)
format: sns
chart: barplot
ext: png
x: '{group}'
y: '{metric}'
dpi: 72
width: '{width}'
height: '{height}'
Content-Type: image/png
# Default login_url is /login/. If you change this, change auth.login_url
pattern: /$YAMLURL/login/
# You MUST change the auth before deploying. DBAuth is commonly used.
# See
handler: SimpleAuth
template: $YAMLPATH/login.html
credentials: { mluser: ml2020pass }
pattern: /$YAMLURL/logout/
handler: LogoutHandler
# Gramex init configurations for app: pdftra
# ----------------------------------------------------------------------------
# /ui/ has Gramex UI components -- use this like the node_modules/ directory
path: $GRAMEXAPPS/ui/gramex.yaml
{% set numeric_columns = ['Sales', 'Quantity', 'Discount', 'Profit'] %}
{% set group_columns = ['Region', 'State', 'Category', 'Ship Mode', 'Segment'] %}
{% set base = '.' %}
<!doctype html>
<html lang="en">
<meta charset="utf-8">
<meta http-equiv="x-ua-compatible" content="ie=edge">
<title>LinReg / Visualizing linear relationships</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="style.css">
{% include template-navbar.html %}
<div class="container-fluid py-4">
<div class="card-deck">
<div class="card">
<div class="card-header bg-white d-flex justify-content-between">
<h5 class="card-title my-0">Visual Summary</h5>
<div class="d-flex align-items-center small">
<span class="text-muted mx-2">Group by</span>
<select name="group" data-draw="groupchart" class="selector">
{% for col in group_columns %}
<option value="{{ col }}">{{ col }}</option>
{% end %}
<span class="text-muted mx-2">Size by</span>
<select name="metric" data-draw="groupchart" class="selector">
{% for col in numeric_columns %}
<option value="{{ col }}">{{ col }}</option>
{% end %}
</div><!-- .card-header -->
<div class="card-body">
<img class="img-fluid viz-groupchart" src="./assets/img/loading.gif" alt="" >
</div><!-- .card-body -->
</div><!-- .card -->
<div class="card">
<div class="card-header bg-white d-flex justify-content-between">
<h5 class="card-title my-0">Linear relationships</h5>
<div class="d-flex align-items-center small">
<span class="text-muted mx-2">X (metric)</span>
<select name="x" data-draw="linreg" class="selector">
{% for col in numeric_columns %}
<option value="{{ col }}">{{ col }}</option>
{% end %}
<span class="text-muted mx-2">Y (metric)</span>
<select name="y" data-draw="linreg" class="selector">
{% for col in numeric_columns[1:] + [numeric_columns[0]] %}
<option value="{{ col }}">{{ col }}</option>
{% end %}
</div><!-- .card-header -->
<div class="card-body">
<img class="img-fluid viz-linreg" src="./assets/img/loading.gif" alt="" >
</div><!-- .card-body -->
</div><!-- .card -->
</div><!-- .card -->
</div><!-- .container-fluid -->
<script src="ui/jquery/dist/jquery.min.js"></script>
<script src="ui/bootstrap/dist/js/bootstrap.bundle.min.js"></script>
// draw
const draw = {
loading: el => el.attr('src', './assets/img/loading.gif'),
linreg: () => {
const $el = $('.viz-linreg')
const select = '.selector[data-draw="linreg"]'
const x = $(`${select}[name="x"]`).val(), y = $(`${select}[name="y"]`).val()
$el.attr('src', `data?_format=linreg&x=${x}&y=${y}&width=500&height=400`)
groupchart: () => {
const $el = $('.viz-groupchart')
const select = '.selector[data-draw="groupchart"]'
const group = $(`${select}[name="group"]`).val(), metric = $(`${select}[name="metric"]`).val()
$el.attr('src', `datagroup?_format=groupchart&group=${group}&metric=${metric}&width=500&height=400`)
// init
// events
$('body').on('change', '.selector', function (e) {
$(window).trigger('draw-view', {draw: this.dataset.draw})
$(window).on('draw-view', function(e, data) { draw[data.draw]() })
def group(data, handler):
args = handler.argparse(group={'default': ''}, metric={'default': ''})
df = data.groupby([args.metric].sum().reset_index()
# Additional Business logic
# New columns
# etc
return df
<!doctype html>
<html lang="en">
<meta charset="utf-8">
<meta http-equiv="x-ua-compatible" content="ie=edge">
<title>LinReg Login</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="../style.css">
<body class="bg-primary gradient-tc bg-no-repeat text-white">
{% set base = '..' %}
{% include template-navbar.html %}
{% set kwargs = handler.kwargs %}
{% try %}{% set user = kwargs.user.arg %}{% except %}{% set user = 'user' %}{% end %}
{% try %}{% set password = kwargs.password.arg %}{% except %}{% set password = 'password' %}{% end %}
<div class="container d-flex flex-column align-items-center">
<div class="card shadow text-dark mx-auto my-4 px-5 py-3 col-md-6">
{% if error %}
<div class="alert alert-danger mx-n3">
<h1 class="h4">Error logging in</h1>
<p>{{ error['error'] }}</p>
<div><small><strong>code</strong>: {{ error['code'] }}</small></div>
{% end %}
<form method="POST">
<div class="form-group">
<label for="{{ user }}">Login</label>
<input type="text" class="form-control" name="{{ user }}" id="{{ user }}" value="{{ handler.get_argument(user, '') }}" placeholder="Login ID" autofocus required>
<div class="form-group">
<label for="{{ password }}">Password</label>
<input type="password" class="form-control" name="{{ password }}" id="{{ password }}" placeholder="Password" required>
<input type="hidden" name="_xsrf" value="{{ handler.xsrf_token }}">
<p><button type="submit" class="btn btn-primary w-100 small">Login</button></p>
{% if kwargs.get('forgot') %}
<p class="small"><a href="?{{ kwargs.forgot.key }}">Forgot password</a></p>
{% end %}
<div>Default login: mluser (password: ml2020pass)</div>
</div><!-- .card -->
<script src="../ui/jquery/dist/jquery.min.js"></script>
<script src="../ui/bootstrap/dist/js/bootstrap.bundle.min.js"></script>
{% if 'hash' in kwargs.get('password', {}) %}
<script src=""></script>
/* globals sha256 */
// hash the password before submitting
$('form').on('submit', function() {
var $password = $('#{{ password }}').get(0)
$password.value = sha256($password.value)
{% end %}
wget -O
unzip; rm;
/* UI component styles. Customize via ?bootstrap-variable=encoded-value. Example:
Colors. Can be a name or a number (e.g. %23aabbcc). Preserve the hues below.
Fonts. Can be a system font or Open+Sans, Roboto, Lato, Anton, Monserrat
@import url("ui/bootstraptheme.css?body-bg=white&navbar-dark-color=rgba(255%2C255%2C255%2C.8)&navbar-dark-hover-color=white");
/* For v4 icons, use url("ui/font-awesome/css/font-awesome.min.css") */
@import url("ui/@fortawesome/fontawesome-free/css/all.min.css");
/* custom styles for app: linreg */
<nav class="navbar navbar-expand-lg navbar-dark bg-dark py-0">
<a class="navbar-brand" href="{{ base }}">
<!-- Replace with your logo. Just specify height, not width. -->
<img height="36" src="" alt="Logo">
<span class="text-uppercase">linreg</span>
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<!-- Navbar middle -->
<div class="navbar-nav mr-auto">
<span class="nav-item nav-link active mb-0">Visualizing linear relationships</span>
<!-- Navbar right -->
<div class="navbar-nav mr-2">
{% if handler.current_user %}
<a class="nav-item nav-link" href="{{ base }}/login/" title="Log in again" data-placement="bottom">
{{ }}
<a class="nav-item nav-link" href="{{ base }}/logout/" title="Log out" data-placement="bottom"><span class="fas fa-arrow-right bg-light round text-dark p-1"></span></a>
{% else %}
<a class="nav-item nav-link" href="{{ base }}/login/" title="Log in again" data-placement="bottom">
Log in
{% end %}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment