rebased
This commit is contained in:
		
							
								
								
									
										152
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										152
									
								
								README.md
									
									
									
									
									
								
							| @ -1,151 +1 @@ | |||||||
| # proxy | # vpn-p2p | ||||||
|  |  | ||||||
| A simple reverse proxy and https termination using openresty/nginx with a managment API and GUI.  |  | ||||||
|  |  | ||||||
| ## API docs |  | ||||||
| [API docs](api.md) |  | ||||||
|  |  | ||||||
| ## Server set up |  | ||||||
|  |  | ||||||
| The server requires: |  | ||||||
| * NodeJS 8.x |  | ||||||
| * inbound Internet access |  | ||||||
| * OpenResty |  | ||||||
| * redis |  | ||||||
| * lua rocks |  | ||||||
|  |  | ||||||
| This has been tested on ubuntu 16.04, but should work on any modern Linux |  | ||||||
| distro. |  | ||||||
| **Optional** Linux users for its user management, so this will |  | ||||||
| **ONLY** work on Linux, no macOS, BSD or Windows and require root. |  | ||||||
|  |  | ||||||
| The steps below are for a new ubuntu server, they should be mostly the same for |  | ||||||
| other distros, but the paths and availability of packages may vary. A dedicated |  | ||||||
| server is highly recommended (since it will make ever user a system user), a VPS |  | ||||||
| like Digital Ocean will do just fine. |  | ||||||
|  |  | ||||||
| * Install openresty |  | ||||||
|  |  | ||||||
|     [OpenResty® Linux Packages](https://openresty.org/en/linux-packages.html) |  | ||||||
|  |  | ||||||
| * These packages are needed for the PAM node package |  | ||||||
|  |  | ||||||
|     ```bash |  | ||||||
|     apt install libpam0g-dev build-essential |  | ||||||
|     ``` |  | ||||||
|  |  | ||||||
| * Install redis |  | ||||||
|  |  | ||||||
|     ```bash |  | ||||||
|     apt install redis-server |  | ||||||
|     ``` |  | ||||||
|  |  | ||||||
| * install lua plugin |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| apt install luarocks |  | ||||||
| sudo luarocks install lua-resty-auto-ssl |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| * openresty config |  | ||||||
|  |  | ||||||
| Set up fail back SSL certs |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| mkdir /etc/ssl/ |  | ||||||
|  |  | ||||||
| openssl req -new -newkey rsa:2048 -days 3650 -nodes -x509   -subj '/CN=sni-support-required-for-valid-ssl'   -keyout /etc/ssl/resty-auto-ssl-fallback.key   -out /etc/ssl/resty-auto-ssl-fallback.crt |  | ||||||
|  |  | ||||||
| openssl req -new -newkey rsa:2048 -days 3650 -nodes -x509   -subj '/CN=sni-support-required-for-valid-ssl'   -keyout /etc/ssl/resty-auto-ssl-fallback.key   -out /etc/ssl/resty-auto-ssl-fallback.crt |  | ||||||
|  |  | ||||||
| # openssl dhparam -out /etc/nginx/dhparam.pem 4096 # This takes a LONG time and is not needed. |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Change the `/etc/openresty/nginx.conf to have this config` |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| #user  nobody; |  | ||||||
| worker_processes 4; |  | ||||||
|  |  | ||||||
| #error_log  logs/error.log; |  | ||||||
| #error_log  logs/error.log  notice; |  | ||||||
| #error_log  logs/error.log  info; |  | ||||||
|  |  | ||||||
| #pid        logs/nginx.pid; |  | ||||||
|  |  | ||||||
|  |  | ||||||
| events { |  | ||||||
|     worker_connections  1024; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| http { |  | ||||||
|     client_max_body_size 4g; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     lua_shared_dict auto_ssl 100m; |  | ||||||
|     lua_shared_dict auto_ssl_settings 64k; |  | ||||||
|  |  | ||||||
|     resolver 8.8.4.4 8.8.8.8; |  | ||||||
|  |  | ||||||
|     init_by_lua_block { |  | ||||||
|         auto_ssl = (require "resty.auto-ssl").new() |  | ||||||
| 	auto_ssl:set("storage_adapter", "resty.auto-ssl.storage_adapters.redis") |  | ||||||
|         auto_ssl:set("allow_domain", function(domain) |  | ||||||
|             return true |  | ||||||
|         end) |  | ||||||
|         auto_ssl:init() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     init_worker_by_lua_block { |  | ||||||
|       auto_ssl:init_worker() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     ssl_session_cache   shared:SSL:10m; |  | ||||||
|     ssl_session_timeout 10m; |  | ||||||
|  |  | ||||||
|     server { |  | ||||||
|       listen 127.0.0.1:8999; |  | ||||||
|  |  | ||||||
|       # Increase the body buffer size, to ensure the internal POSTs can always |  | ||||||
|       # parse the full POST contents into memory. |  | ||||||
|       client_body_buffer_size 128k; |  | ||||||
|       client_max_body_size 128k; |  | ||||||
|  |  | ||||||
|       location / { |  | ||||||
|         content_by_lua_block { |  | ||||||
|           auto_ssl:hook_server() |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     include       mime.types; |  | ||||||
|     default_type  application/octet-stream; |  | ||||||
|  |  | ||||||
|     #log_format  main  '$remote_addr - $remote_user [$time_local] "$request" ' |  | ||||||
|     #                  '$status $body_bytes_sent "$http_referer" ' |  | ||||||
|     #                  '"$http_user_agent" "$http_x_forwarded_for"'; |  | ||||||
|  |  | ||||||
|     access_log /var/log/nginx/access.log; |  | ||||||
|     error_log /var/log/nginx/error.log; |  | ||||||
|  |  | ||||||
|     sendfile        on; |  | ||||||
|     #tcp_nopush     on; |  | ||||||
|  |  | ||||||
|     #keepalive_timeout  0; |  | ||||||
|     keepalive_timeout  65; |  | ||||||
|  |  | ||||||
|     #gzip  on; |  | ||||||
|     include sites-enabled/*; |  | ||||||
|  |  | ||||||
| } |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| add the SSL config file `/etc/openresty/autossl.conf`, contents from here |  | ||||||
| https://github.com/theta42/t42-common/blob/master/templates/openresty/autossl.conf.erb |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Add the proxy config `/etc/openresty/sites-enabled/000-proxy` contents from here |  | ||||||
| https://github.com/theta42/t42-common/blob/master/templates/openresty/010-proxy.conf.erb |  | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								Vagrantfile
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								Vagrantfile
									
									
									
									
										vendored
									
									
								
							| @ -42,7 +42,7 @@ Vagrant.configure("2") do |config| | |||||||
|   # accessing "localhost:8080" will access port 80 on the guest machine. |   # accessing "localhost:8080" will access port 80 on the guest machine. | ||||||
|   # NOTE: This will enable public access to the opened port |   # NOTE: This will enable public access to the opened port | ||||||
|   config.vm.network "forwarded_port", guest: 80, host: 8000 |   config.vm.network "forwarded_port", guest: 80, host: 8000 | ||||||
|   config.vm.network "forwarded_port", guest: 443, host: 8443 |  | ||||||
|   config.vm.network "forwarded_port", guest: 3000, host: 8300 |   config.vm.network "forwarded_port", guest: 3000, host: 8300 | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -100,9 +100,9 @@ Vagrant.configure("2") do |config| | |||||||
|     chef.json = { |     chef.json = { | ||||||
|       'working-dir': '/vagrant', |       'working-dir': '/vagrant', | ||||||
|       'app': { |       'app': { | ||||||
|         'name': 't42-proxy', |         'name': 'vpn-p2p', | ||||||
|         'run_user': 'root', |         'run_user': 'root', | ||||||
|         'domain': 'proxy.local', |         'domain': 'vpn-p2p.local', | ||||||
|       }, |       }, | ||||||
|       'python': { |       'python': { | ||||||
|         # 'working-dir': 'django', |         # 'working-dir': 'django', | ||||||
| @ -120,11 +120,11 @@ Vagrant.configure("2") do |config| | |||||||
|           'perm': '777' |           'perm': '777' | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       'web':{ |       # 'web':{ | ||||||
|         'admin_email': 'admin2342@example.com', |       #   'admin_email': 'admin2342@example.com', | ||||||
|         'do_ssl': true, |       #   'do_ssl': true, | ||||||
|         't42-proxy': true |       #   't42-proxy': true | ||||||
|       }, |       # }, | ||||||
|     }.deep_merge(secrets); |     }.deep_merge(secrets); | ||||||
|   end |   end | ||||||
| end | end | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								nodejs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								nodejs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | sendgrid.env | ||||||
							
								
								
									
										60
									
								
								nodejs/api.md
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										60
									
								
								nodejs/api.md
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,60 @@ | |||||||
|  | ## create invite token | ||||||
|  |  | ||||||
|  | **post** `/users/invite` | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | curl -H "Content-Type: application/json" -H "auth-token: 0b06eb2e-4ca4-4881-9a0f-b8df55431cd1" -X POST https://proxy-host.com/users/invite | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * 200 {"token":"5caf94d2-2c91-4010-8df7-968d10802b9d"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## sing up | ||||||
|  |  | ||||||
|  | **post** `/auth/invite/<INVITE TOKEN>` | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | curl -H "Content-Type: application/json" -X POST -d "{\"username\": \"test9\", \"password\": \"palm7\"}" https://proxy-host.com/auth/invite/b33d8819-ec64-4cf4-a6ec-77562d738fa4 | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * 200 {"user":"test9","token":"af662d8b-3d44-4110-8ad9-047dc752d97f"} | ||||||
|  | * 400 {"message":"Missing fields"} | ||||||
|  | * 401 {"message":"Token not valid"} | ||||||
|  | * 409 {"message":"username taken"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## login | ||||||
|  |  | ||||||
|  | **post** `/auth/login` | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | curl -H "Content-Type: application/json" -X POST -d '{"username": "test8", "password": "mypassword"}' https://proxy-host.com/auth/login | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * 200 {"login":true,"token":"027d3964-7d81-4462-a6f9-2c1f9b40b4be"} | ||||||
|  | * 401 {"login":false} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## verify SSH key | ||||||
|  |  | ||||||
|  | **post** `/auth/verifykey` | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | curl -H "Content-Type: application/json" -X POST -d "{\"key\":\"ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAACAQDM9vboz5YGgESsrR2e4JOeP2qtmQo2S8BjI+Y/VxPQ6WbNFzAkXxDniHcnPCrhkeX36SKINvMjWnt4XOK2S+X+1tCoXJzqtcKKyK0gx8ijBxcWVPxsMWjMYTGSVSKiKnt6CyQzrbVGJMh3iAQ8Yv1JwH+6SAtMgT8it7iLyntNFJCesh4I/znEG58A5VBbdUle1Ztz9afjj1CZns17jk7KPm9ig5DmuvdvnMEfhFjfKv1Rp6S5nxacMoTP4tJNSEUh55IicoWk94ii5GwUVLYgyMmzdlA32TqVLFpU2yAvdA9WSnBaI/ZyktlfI7YAmK2wFBsagr9Pq1TcUAY6rZ/GTMjDxExgdYn/FxlufcuqeNJsJXs2A+0xDS/9mv/yGQzNZrL8DrVhY2OKKLoH4Q7enDbhSgEFmJUJMqPxuPEgLEvKfzcURSvIwRj1iCEw6S4dhdaLJl2RRBb1ZWBQbE5ogIbvAl7GFJUAhj3pqYJnd30VENv1MkK+IoCS7EEP0caqL9RNAId0Plud7q2XElHqzkYUE+z+Q/LvGgclXK1ZmZejNaMnV53wfhAevfwVyNGK9i5gbwc1P2lplIa5laXCcVWezqELEkTpdjp4AeKmMuCr8rY8EnLKIcKWEOsX5UumztCow6e1E55v3VeHvRZLpw4DZP7EE0Q8B/jPFWqbCw== wmantly@gmail.com\"}" https://proxy-host.com/auth/verifykey | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * 200 {"info":"4096 SHA256:dfdCYzt0atMBXVZTJzUxsu99IjXXFXpocSox5q+jOs8 wmantly@gmail.com (RSA)\n"} | ||||||
|  | * 400 {"message":"Key is not a public key file!"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## add ssh key to current user | ||||||
|  |  | ||||||
|  | **post** `/users/key` | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | curl -H "Content-Type: application/json" -H "auth-token: 8eff4f16-086d-40fd-acbd-7634b9a36117" -X POST -d "{\"key\": \"ssh-rsa AAAAB3NzaC1yc2EAAjWnt4XOK2S+X+1tCoXJzqtcKKyK0gx8ijBxcWVPxsMWjMYTGSVSKiKnt6CyQzrbVGJMh3iAQ8Yv1JwH+6SAtMgT8it7iLyntNFJCesh4I/znEG58A5VBbdUle1Ztz9afjj1CZns17jk7KPm9ig5DmuvdvnMEfhFjfKv1Rp6S5nxacMoTP4tJNSEUh55IicoWk94ii5GwUVLYgyMmzdlA32TqVLFpU2yAvdA9WSnBaI/ZyktlfI7YAmK2wFBsagr9Pq1TcUAY6rZ/GTMjDxExgdYn/FxlufcuqeNJsJXs2A+0xDS/9mv/yGQzNZrL8DrVhY2OKKLoH4Q7enDbhSgEFmJUJMqPxuPEgLEvKfzcURSvIwRj1iCEw6S4dhdaLJl2RRBb1ZWBQbE5ogIbvAl7GFJUAhj3pqYJnd30VENv1MkK+IoCS7EEP0caqL9RNAId0Plud7q2XElHqzkYUE+z+Q/LvGgclXK1ZmZejNaMnV53wfhAevfwVyNGK9i5gbwc1P2lplIa5laXCcVWezqELEkTpdjp4AeKmMuCr8rY8EnLKIcKWEOsX5UumztCow6e1E55v3VeHvRZLpw4DZP7EE0Q8B/jPFWqbCw== wmantly@gmail.co\"}" https://proxy-host.com/users/key | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * 200 {"message":true} | ||||||
|  | * 400 {"message":"Bad SSH key"} | ||||||
							
								
								
									
										63
									
								
								nodejs/app.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										63
									
								
								nodejs/app.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const path = require('path'); | ||||||
|  | const ejs = require('ejs') | ||||||
|  | const express = require('express'); | ||||||
|  |  | ||||||
|  | // Set up the express app. | ||||||
|  | const app = express(); | ||||||
|  |  | ||||||
|  | // Allow the express app to be exported into other files.  | ||||||
|  | module.exports = app; | ||||||
|  |  | ||||||
|  | // Build the conf object from the conf files. | ||||||
|  | app.conf = require('./conf/conf'); | ||||||
|  |  | ||||||
|  | // Hold onto the auth middleware  | ||||||
|  | const middleware = require('./middleware/auth'); | ||||||
|  |  | ||||||
|  | // load the JSON parser middleware. Express will parse JSON into native objects | ||||||
|  | // for any request that has JSON in its content type.  | ||||||
|  | app.use(express.json()); | ||||||
|  |  | ||||||
|  | // Set up the templating engine to build HTML for the front end. | ||||||
|  | app.set('views', path.join(__dirname, 'views')); | ||||||
|  | app.set('view engine', 'ejs'); | ||||||
|  |  | ||||||
|  | // Have express server static content( images, CSS, browser JS) from the public | ||||||
|  | // local folder. | ||||||
|  | app.use('/static', express.static(path.join(__dirname, 'public'))) | ||||||
|  |  | ||||||
|  | // Routes for front end content. | ||||||
|  | app.use('/', require('./routes/index')); | ||||||
|  |  | ||||||
|  | // API routes for authentication.  | ||||||
|  | app.use('/api/auth',  require('./routes/auth')); | ||||||
|  |  | ||||||
|  | // API routes for working with users. All endpoints need to be have valid user. | ||||||
|  | app.use('/api/user', middleware.auth, require('./routes/user')); | ||||||
|  |  | ||||||
|  | app.use('/api/token', middleware.auth, require('./routes/token')); | ||||||
|  |  | ||||||
|  | app.use('/api/group', middleware.auth, require('./routes/group')); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // Catch 404 and forward to error handler. If none of the above routes are | ||||||
|  | // used, this is what will be called. | ||||||
|  | app.use(function(req, res, next) { | ||||||
|  |   var err = new Error('Not Found'); | ||||||
|  |   err.message = 'Page not found' | ||||||
|  |   err.status = 404; | ||||||
|  |   next(err); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Error handler. This is where `next()` will go on error | ||||||
|  | app.use(function(err, req, res, next) { | ||||||
|  |   console.error(err.status || res.status, err.name, req.method, req.url); | ||||||
|  |   console.error(err.message); | ||||||
|  |   console.error(err.stack); | ||||||
|  |   console.error('========================================='); | ||||||
|  |  | ||||||
|  |   res.status(err.status || 500); | ||||||
|  |   res.json({name: err.name, message: err.message}); | ||||||
|  | }); | ||||||
							
								
								
									
										90
									
								
								nodejs/bin/www
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										90
									
								
								nodejs/bin/www
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,90 @@ | |||||||
|  | #!/usr/bin/env node | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Module dependencies. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | var app = require('../app'); | ||||||
|  | var debug = require('debug')('proxy-api:server'); | ||||||
|  | var http = require('http'); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Get port from environment and store in Express. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | var port = normalizePort(process.env.NODE_PORT || app.conf.port || '3000'); | ||||||
|  | app.set('port', port); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Create HTTP server. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | var server = http.createServer(app); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Listen on provided port, on all network interfaces. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | server.listen(port); | ||||||
|  | server.on('error', onError); | ||||||
|  | server.on('listening', onListening); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Normalize a port into a number, string, or false. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | function normalizePort(val) { | ||||||
|  |   var port = parseInt(val, 10); | ||||||
|  |  | ||||||
|  |   if (isNaN(port)) { | ||||||
|  |     // named pipe | ||||||
|  |     return val; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (port >= 0) { | ||||||
|  |     // port number | ||||||
|  |     return port; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Event listener for HTTP server "error" event. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | function onError(error) { | ||||||
|  |   if (error.syscall !== 'listen') { | ||||||
|  |     throw error; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var bind = typeof port === 'string' | ||||||
|  |     ? 'Pipe ' + port | ||||||
|  |     : 'Port ' + port; | ||||||
|  |  | ||||||
|  |   // handle specific listen errors with friendly messages | ||||||
|  |   switch (error.code) { | ||||||
|  |     case 'EACCES': | ||||||
|  |       console.error(bind + ' requires elevated privileges'); | ||||||
|  |       process.exit(1); | ||||||
|  |       break; | ||||||
|  |     case 'EADDRINUSE': | ||||||
|  |       console.error(bind + ' is already in use'); | ||||||
|  |       process.exit(1); | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       throw error; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Event listener for HTTP server "listening" event. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | function onListening() { | ||||||
|  |   var addr = server.address(); | ||||||
|  |   var bind = typeof addr === 'string' | ||||||
|  |     ? 'pipe ' + addr | ||||||
|  |     : 'port ' + addr.port; | ||||||
|  |   debug('Listening on ' + bind); | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								nodejs/conf/base.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								nodejs/conf/base.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  | 	userModel: 'ldap', // pam, redis, ldap | ||||||
|  | 	ldap: { | ||||||
|  | 		url: 'ldap://192.168.1.54:389', | ||||||
|  | 		bindDN: 'cn=admin,dc=theta42,dc=com', | ||||||
|  | 		bindPassword: '__IN SRECREST FILE__', | ||||||
|  | 		userBase: 'ou=people,dc=theta42,dc=com', | ||||||
|  | 		groupBase: 'ou=groups,dc=theta42,dc=com',		 | ||||||
|  | 		userFilter: '(objectClass=posixAccount)', | ||||||
|  | 		userNameAttribute: 'uid' | ||||||
|  | 	}, | ||||||
|  | 	SENDGRID_API_KEy: '__IN SRECREST FILE__', | ||||||
|  | }; | ||||||
							
								
								
									
										32
									
								
								nodejs/conf/conf.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								nodejs/conf/conf.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const extend = require('extend'); | ||||||
|  |  | ||||||
|  | const environment = process.env.NODE_ENV || 'development'; | ||||||
|  |  | ||||||
|  | function load(filePath, required){ | ||||||
|  | 	try { | ||||||
|  | 		return require(filePath); | ||||||
|  | 	} catch(error){ | ||||||
|  | 		if(error.name === 'SyntaxError'){ | ||||||
|  | 			console.error(`Loading ${filePath} file failed!\n`, error); | ||||||
|  | 			process.exit(1); | ||||||
|  | 		} else if (error.code === 'MODULE_NOT_FOUND'){ | ||||||
|  | 			console.warn(`No config file ${filePath} FOUND! This may cause issues...`); | ||||||
|  | 			if (required){ | ||||||
|  | 				process.exit(1); | ||||||
|  | 			} | ||||||
|  | 			return {}; | ||||||
|  | 		}else{ | ||||||
|  | 			console.dir(`Unknown error in loading ${filePath} config file.\n`, error); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = extend( | ||||||
|  | 	true, // enable deep copy | ||||||
|  | 	load('./base', true), | ||||||
|  | 	load(`./${environment}`), | ||||||
|  | 	load('./secrets'), | ||||||
|  | 	{environment} | ||||||
|  | ); | ||||||
							
								
								
									
										18
									
								
								nodejs/middleware/auth.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										18
									
								
								nodejs/middleware/auth.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const {Auth} = require('../models/auth');  | ||||||
|  |  | ||||||
|  | async function auth(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await Auth.checkToken({token: req.header('auth-token')}); | ||||||
|  |  | ||||||
|  | 		if(user.uid){ | ||||||
|  | 			req.user = user; | ||||||
|  | 			return next(); | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = {auth}; | ||||||
							
								
								
									
										50
									
								
								nodejs/models/auth.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								nodejs/models/auth.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const {User} = require('./user'); | ||||||
|  | const {Token, AuthToken} = require('./token'); | ||||||
|  |  | ||||||
|  | var Auth = {} | ||||||
|  | Auth.errors = {} | ||||||
|  |  | ||||||
|  | Auth.errors.login = function(){ | ||||||
|  | 	let error = new Error('LDAPLoginFailed'); | ||||||
|  | 	error.name = 'LDAPLoginFailed'; | ||||||
|  | 	error.message = `Invalid Credentials, login failed.`; | ||||||
|  | 	error.status = 401; | ||||||
|  |  | ||||||
|  | 	return error; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Auth.login = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await User.login(data); | ||||||
|  | 		let token = await AuthToken.add(user); | ||||||
|  |  | ||||||
|  | 		return {user, token} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw this.errors.login(); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Auth.checkToken = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await AuthToken.get(data); | ||||||
|  | 		if(token.is_valid){ | ||||||
|  | 			return await User.get(token.created_by); | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw this.errors.login(); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | Auth.logOut = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await AuthToken.get(data); | ||||||
|  | 		await token.remove(); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = {Auth, AuthToken}; | ||||||
							
								
								
									
										32
									
								
								nodejs/models/email.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								nodejs/models/email.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const sgMail = require('@sendgrid/mail'); | ||||||
|  | const mustache = require('mustache'); | ||||||
|  | const conf = require('../app').conf; | ||||||
|  |  | ||||||
|  | sgMail.setApiKey(conf.SENDGRID_API_KEY); | ||||||
|  |  | ||||||
|  | var Mail = {}; | ||||||
|  |  | ||||||
|  | Mail.send = async function(to, subject, message, from){ | ||||||
|  | 	await sgMail.send({ | ||||||
|  | 		to: to, | ||||||
|  | 		from: from || 'Theta 42 Accounts <accounts@no-reply.theta42.com>', | ||||||
|  | 		subject: subject, | ||||||
|  | 		text: message, | ||||||
|  | 		html: message, | ||||||
|  | 	}); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Mail.sendTemplate = async function(to, template, context, from){ | ||||||
|  | 	template = require(`../views/email_templates/${template}`); | ||||||
|  | 	await Mail.send( | ||||||
|  | 		to, | ||||||
|  | 		mustache.render(template.subject, context), | ||||||
|  | 		mustache.render(template.message, context), | ||||||
|  | 		from || (template.from && mustache.render(template.message, context)) | ||||||
|  | 	) | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = {Mail}; | ||||||
							
								
								
									
										213
									
								
								nodejs/models/group_ldap.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								nodejs/models/group_ldap.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,213 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const { Client, Attribute, Change } = require('ldapts'); | ||||||
|  | const conf = require('../app').conf.ldap; | ||||||
|  |  | ||||||
|  | const client = new Client({ | ||||||
|  | 	url: conf.url, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | async function getGroups(client, member){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		let memberFilter = member ? `(member=${member})`: '' | ||||||
|  |  | ||||||
|  | 		let groups = (await client.search(conf.groupBase, { | ||||||
|  | 			scope: 'sub', | ||||||
|  | 			filter: `(&(objectClass=groupOfNames)${memberFilter})`, | ||||||
|  | 			attributes: ['*', 'createTimestamp', 'modifyTimestamp'], | ||||||
|  | 		})).searchEntries; | ||||||
|  |  | ||||||
|  | 		return groups.map(function(group){ | ||||||
|  | 			if(!Array.isArray(group.member)) group.member = [group.member]; | ||||||
|  | 			return group | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function addGroup(client, data){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		await client.add(`cn=${data.name},${conf.groupBase}`, { | ||||||
|  | 			cn: data.name, | ||||||
|  | 			member: data.owner, | ||||||
|  | 			description: data.description, | ||||||
|  | 			owner: data.owner, | ||||||
|  | 			objectclass: [ 'groupOfNames', 'top'  ] | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		return data; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function addMember(client, group, user){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.modify(group.dn, [ | ||||||
|  | 			new Change({ | ||||||
|  | 				operation: 'add', | ||||||
|  | 				modification: new Attribute({ | ||||||
|  | 					type: 'member', | ||||||
|  | 					values: [user.dn]  | ||||||
|  | 				}) | ||||||
|  | 			}), | ||||||
|  | 		]);  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		// if(error = "TypeOrValueExistsError"){ | ||||||
|  | 		// 	console.error('addMember error skipped', error) | ||||||
|  | 		// 	return ; | ||||||
|  | 		// } | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function removeMember(client, group, user){ | ||||||
|  |   try{ | ||||||
|  | 	await client.modify(group.dn, [ | ||||||
|  | 		new Change({ | ||||||
|  | 			operation: 'delete', | ||||||
|  | 			modification: new Attribute({ | ||||||
|  | 				type: 'member', | ||||||
|  | 				values: [user.dn]  | ||||||
|  | 			})}), | ||||||
|  | 		]);  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		if(error = "TypeOrValueExistsError")return ; | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | var Group = {}; | ||||||
|  |  | ||||||
|  | Group.list = async function(member){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		let groups = await getGroups(client, member) | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return groups.map(group => group.cn); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Group.listDetail = async function(member){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		let groups = await getGroups(client, member) | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		return groups; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Group.get = async function(data){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		if(typeof data !== 'object'){ | ||||||
|  | 			let name = data; | ||||||
|  | 			data = {}; | ||||||
|  | 			data.name = name; | ||||||
|  | 		} | ||||||
|  | 		 | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		let group = (await client.search(conf.groupBase, { | ||||||
|  | 			scope: 'sub', | ||||||
|  | 			filter: `(&(objectClass=groupOfNames)(cn=${data.name}))`, | ||||||
|  | 			attributes: ['*', 'createTimestamp', 'modifyTimestamp'], | ||||||
|  | 		})).searchEntries[0]; | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		if(!Array.isArray(group.member)) group.member = [group.member]; | ||||||
|  |  | ||||||
|  | 		if(group){ | ||||||
|  | 			let obj = Object.create(this); | ||||||
|  | 			Object.assign(obj, group); | ||||||
|  | 			 | ||||||
|  | 			return obj; | ||||||
|  | 		}else{ | ||||||
|  | 			let error = new Error('GroupNotFound'); | ||||||
|  | 			error.name = 'GroupNotFound'; | ||||||
|  | 			error.message = `LDAP:${data.cn} does not exists`; | ||||||
|  | 			error.status = 404; | ||||||
|  | 			throw error; | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Group.add = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await addGroup(client, data); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return this.get(data); | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Group.addMember = async function(user){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await addMember(client, this, user); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | Group.removeMember = async function(user){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await removeMember(client, this, user); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | Group.remove = async function(){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await client.del(this.dn); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return true; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = {Group}; | ||||||
							
								
								
									
										71
									
								
								nodejs/models/token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								nodejs/models/token.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const redis_model = require('../utils/redis_model') | ||||||
|  | const UUID = function b(a){return a?(a^Math.random()*16>>a/4).toString(16):([1e7]+-1e3+-4e3+-8e3+-1e11).replace(/[018]/g,b)}; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | const Token = function(data){ | ||||||
|  | 	return redis_model({ | ||||||
|  | 		_name: `token_${data.name}`, | ||||||
|  | 		_key: 'token', | ||||||
|  | 		_keyMap: Object.assign({}, { | ||||||
|  | 			'created_by': {isRequired: true, type: 'string', min: 3, max: 500}, | ||||||
|  | 			'created_on': {default: function(){return (new Date).getTime()}}, | ||||||
|  | 			'updated_on': {default: function(){return (new Date).getTime()}, always: true}, | ||||||
|  | 			'token': {default: UUID, type: 'string', min: 36, max: 36}, | ||||||
|  | 			'is_valid': {default: true, type: 'boolean'} | ||||||
|  | 		}, data.keyMap || {}) | ||||||
|  | 	}); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | Token.check = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		return this.is_valid; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var InviteToken = Object.create(Token({ | ||||||
|  | 	name: 'invite', | ||||||
|  | 	keyMap:{ | ||||||
|  | 		claimed_by: {default:"__NONE__", isRequired: false, type: 'string',}, | ||||||
|  | 		mail: {default:"__NONE__", isRequired: false, type: 'string',}, | ||||||
|  | 		mail_token: {default: UUID, type: 'string', min: 36, max: 36}, | ||||||
|  | 	} | ||||||
|  | })); | ||||||
|  |  | ||||||
|  | InviteToken.consume = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		if(this.is_valid){ | ||||||
|  | 			data['is_valid'] = false; | ||||||
|  |  | ||||||
|  | 			await this.update(data); | ||||||
|  | 			return true; | ||||||
|  | 		} | ||||||
|  | 		return false; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var AuthToken = Object.create(Token({ | ||||||
|  | 	name: 'auth', | ||||||
|  | })); | ||||||
|  |  | ||||||
|  | AuthToken.add = async function(data){ | ||||||
|  | 	data.created_by = data.uid; | ||||||
|  | 	return AuthToken.__proto__.add(data); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | var PasswordResetToken = Object.create(Token({ | ||||||
|  | 	name: 'auth', | ||||||
|  | })); | ||||||
|  |  | ||||||
|  | PasswordResetToken.add = async function(data){ | ||||||
|  | 	data.created_by = data.uid; | ||||||
|  | 	return PasswordResetToken.__proto__.add(data); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = {Token, InviteToken, AuthToken, PasswordResetToken}; | ||||||
							
								
								
									
										7
									
								
								nodejs/models/user.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								nodejs/models/user.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const conf = require('../app').conf; | ||||||
|  |  | ||||||
|  | const User = require(`./user_${conf.userModel}`) | ||||||
|  |  | ||||||
|  | module.exports = User; | ||||||
							
								
								
									
										447
									
								
								nodejs/models/user_ldap.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										447
									
								
								nodejs/models/user_ldap.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,447 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const { Client, Attribute, Change } = require('ldapts'); | ||||||
|  | const crypto = require('crypto'); | ||||||
|  |  | ||||||
|  | const {Mail} = require('./email'); | ||||||
|  | const {Token, InviteToken, PasswordResetToken} = require('./token'); | ||||||
|  | const conf = require('../app').conf.ldap; | ||||||
|  |  | ||||||
|  | const client = new Client({ | ||||||
|  |   url: conf.url, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | async function addPosixGroup(client, data){ | ||||||
|  |  | ||||||
|  |   try{ | ||||||
|  | 	const groups = (await client.search(conf.groupBase, { | ||||||
|  | 	  scope: 'sub', | ||||||
|  | 	  filter: '(&(objectClass=posixGroup))', | ||||||
|  | 	})).searchEntries; | ||||||
|  |  | ||||||
|  | 	data.gidNumber = (Math.max(...groups.map(i => i.gidNumber))+1)+''; | ||||||
|  |  | ||||||
|  | 	await client.add(`cn=${data.cn},${conf.groupBase}`, { | ||||||
|  | 	  cn: data.cn, | ||||||
|  | 	  gidNumber: data.gidNumber, | ||||||
|  | 	  objectclass: [ 'posixGroup', 'top' ] | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | 	return data; | ||||||
|  |  | ||||||
|  |   }catch(error){ | ||||||
|  | 	throw error; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function addPosixAccount(client, data){ | ||||||
|  |   try{ | ||||||
|  | 	const people = (await client.search(conf.userBase, { | ||||||
|  | 		scope: 'sub', | ||||||
|  | 		filter: conf.userFilter, | ||||||
|  | 	})).searchEntries; | ||||||
|  |  | ||||||
|  | 	data.uidNumber = (Math.max(...people.map(i => i.uidNumber))+1)+''; | ||||||
|  |  | ||||||
|  | 	await client.add(`cn=${data.cn},${conf.userBase}`, { | ||||||
|  | 		cn: data.cn, | ||||||
|  | 		sn: data.sn, | ||||||
|  | 		uid: data.uid, | ||||||
|  | 		uidNumber: data.uidNumber, | ||||||
|  | 		gidNumber: data.gidNumber, | ||||||
|  | 		givenName: data.givenName, | ||||||
|  | 		mail: data.mail, | ||||||
|  | 		mobile: data.mobile, | ||||||
|  | 		loginShell: data.loginShell, | ||||||
|  | 		homeDirectory: data.homeDirectory, | ||||||
|  | 		userPassword: data.userPassword, | ||||||
|  | 		description: data.description || ' ',  | ||||||
|  | 		sudoHost: 'ALL', | ||||||
|  | 		sudoCommand: 'ALL', | ||||||
|  | 		sudoUser: data.uid, | ||||||
|  | 		sshPublicKey: data.sshPublicKey, | ||||||
|  | 		objectclass: ['inetOrgPerson', 'sudoRole', 'ldapPublicKey', 'posixAccount', 'top' ] | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | 	return data | ||||||
|  |  | ||||||
|  |   }catch(error){ | ||||||
|  | 	throw error; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function addLdapUser(client, data){ | ||||||
|  |  | ||||||
|  | 	var group; | ||||||
|  |  | ||||||
|  |   try{ | ||||||
|  | 	data.uid = `${data.givenName[0]}${data.sn}`; | ||||||
|  | 	data.cn = data.uid; | ||||||
|  | 	data.loginShell = '/bin/bash'; | ||||||
|  | 	data.homeDirectory= `/home/${data.uid}`; | ||||||
|  | 	data.userPassword = '{MD5}'+crypto.createHash('md5').update(data.userPassword, "binary").digest('base64'); | ||||||
|  |  | ||||||
|  | 	group = await addPosixGroup(client, data); | ||||||
|  | 	data = await addPosixAccount(client, group); | ||||||
|  |  | ||||||
|  | 	return data; | ||||||
|  |  | ||||||
|  |   }catch(error){ | ||||||
|  |   	await deleteLdapDN(client, `cn=${data.uid},${conf.groupBase}`, true); | ||||||
|  | 	throw error; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function deleteLdapUser(client, data){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.del(`cn=${data.cn},${conf.groupBase}`); | ||||||
|  | 		await client.del(data.dn); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function deleteLdapDN(client, dn, ignoreError){ | ||||||
|  | 	try{ | ||||||
|  | 		client.del(dn) | ||||||
|  | 	}catch(error){ | ||||||
|  | 		if(!ignoreError) throw error; | ||||||
|  | 		console.error('ERROR: deleteLdapDN', error) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const user_parse = function(data){ | ||||||
|  | 	if(data[conf.userNameAttribute]){ | ||||||
|  | 		data.username = data[conf.userNameAttribute] | ||||||
|  | 		data.userPassword = undefined; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return data; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var User = {} | ||||||
|  |  | ||||||
|  | User.backing = "LDAP"; | ||||||
|  |  | ||||||
|  | User.list = async function(){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		const res = await client.search(conf.userBase, { | ||||||
|  | 		  scope: 'sub', | ||||||
|  | 		  filter: conf.userFilter, | ||||||
|  | 		  attributes: ['*', 'createTimestamp', 'modifyTimestamp'], | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return res.searchEntries.map(function(user){return user.uid}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.listDetail = async function(){ | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		const res = await client.search(conf.userBase, { | ||||||
|  | 		  scope: 'sub', | ||||||
|  | 		  filter: conf.userFilter, | ||||||
|  | 		  attributes: ['*', 'createTimestamp', 'modifyTimestamp'], | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		let users = [] | ||||||
|  |  | ||||||
|  | 		for(let user of res.searchEntries){ | ||||||
|  | 			let obj = Object.create(this); | ||||||
|  | 			Object.assign(obj, user_parse(user)); | ||||||
|  | 			 | ||||||
|  | 			users.push(obj) | ||||||
|  |  | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return users; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.get = async function(data, key){ | ||||||
|  | 	try{ | ||||||
|  | 		if(typeof data !== 'object'){ | ||||||
|  | 			let uid = data; | ||||||
|  | 			data = {}; | ||||||
|  | 			data.uid = uid; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		data.searchKey = data.searchKey || key || conf.userNameAttribute; | ||||||
|  | 		data.searchValue = data.searchValue || data.uid; | ||||||
|  |  | ||||||
|  | 		let filter = `(&${conf.userFilter}(${data.searchKey}=${data.searchValue}))`; | ||||||
|  |  | ||||||
|  | 		const res = await client.search(conf.userBase, { | ||||||
|  | 			scope: 'sub', | ||||||
|  | 			filter: filter, | ||||||
|  | 			attributes: ['*', 'createTimestamp', 'modifyTimestamp'], | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		let user = res.searchEntries[0] | ||||||
|  |  | ||||||
|  | 		if(user){ | ||||||
|  | 			let obj = Object.create(this); | ||||||
|  | 			Object.assign(obj, user_parse(user)); | ||||||
|  | 			 | ||||||
|  | 			return obj; | ||||||
|  | 		}else{ | ||||||
|  | 			let error = new Error('UserNotFound'); | ||||||
|  | 			error.name = 'UserNotFound'; | ||||||
|  | 			error.message = `LDAP:${data.searchValue} does not exists`; | ||||||
|  | 			error.status = 404; | ||||||
|  | 			throw error; | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.exists = async function(data, key){ | ||||||
|  | 	// Return true or false if the requested entry exists ignoring error's. | ||||||
|  | 	try{ | ||||||
|  | 		await this.get(data, key); | ||||||
|  |  | ||||||
|  | 		return true | ||||||
|  | 	}catch(error){ | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.add = async function(data) { | ||||||
|  | 	try{ | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await addLdapUser(client, data); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		let user = await this.get(data.uid); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		await Mail.sendTemplate( | ||||||
|  | 			user.mail, | ||||||
|  | 			'welcome', | ||||||
|  | 			{ | ||||||
|  | 				user: user | ||||||
|  | 			} | ||||||
|  | 		) | ||||||
|  |  | ||||||
|  | 		return user; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		if(error.message.includes('exists')){ | ||||||
|  | 			let error = new Error('UserNameUsed'); | ||||||
|  | 			error.name = 'UserNameUsed'; | ||||||
|  | 			error.message = `LDAP:${data.uid} already exists`; | ||||||
|  | 			error.status = 409; | ||||||
|  |  | ||||||
|  | 			throw error; | ||||||
|  | 		} | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.update = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let editableFeilds = ['mobile', 'sshPublicKey', 'description']; | ||||||
|  |  | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		for(let field of editableFeilds){ | ||||||
|  | 			if(data[field]){ | ||||||
|  | 				await client.modify(this.dn, [ | ||||||
|  | 					new Change({ | ||||||
|  | 						operation: 'replace', | ||||||
|  | 						modification: new Attribute({ | ||||||
|  | 							type: field, | ||||||
|  | 							values: [data[field]]  | ||||||
|  | 						}) | ||||||
|  | 					}), | ||||||
|  | 				]); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await client.unbind() | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.addByInvite = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await InviteToken.get(data.token); | ||||||
|  |  | ||||||
|  | 		if(!token.is_valid && data.mailToken !== token.mail_token){ | ||||||
|  | 			let error = new Error('Token Invalid'); | ||||||
|  | 			error.name = 'Token Invalid'; | ||||||
|  | 			error.message = `Token is not valid or as allready been used. ${data.token}`; | ||||||
|  | 			error.status = 401; | ||||||
|  | 			throw error; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		data.mail = token.mail; | ||||||
|  |  | ||||||
|  | 		let user = await this.add(data); | ||||||
|  |  | ||||||
|  | 		if(user){ | ||||||
|  | 			await token.consume({claimed_by: user.uid}); | ||||||
|  | 			return user; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.verifyEmail = async function(data){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		let exists = await this.exists(data.mail, 'mail'); | ||||||
|  |  | ||||||
|  | 		if(exists) throw new Error('EmailInUse'); | ||||||
|  |  | ||||||
|  | 		let token = await InviteToken.get(data.token); | ||||||
|  | 		await token.update({mail: data.mail}) | ||||||
|  | 		await Mail.sendTemplate( | ||||||
|  | 			data.mail, | ||||||
|  | 			'validate_link', | ||||||
|  | 			{ | ||||||
|  | 				link:`${data.url}/login/invite/${token.token}/${token.mail_token}` | ||||||
|  | 			} | ||||||
|  | 		) | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.passwordReset = async function(url, mail){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		let user = await User.get({ | ||||||
|  | 			searchKey: 'mail', | ||||||
|  | 			searchValue: mail | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		let token = await PasswordResetToken.add(user); | ||||||
|  |  | ||||||
|  | 		await Mail.sendTemplate( | ||||||
|  | 			user.mail, | ||||||
|  | 			'reset_link', | ||||||
|  | 			{ | ||||||
|  | 				user: user, | ||||||
|  | 				link:`${url}/login/resetpassword/${token.token}` | ||||||
|  | 			} | ||||||
|  | 		) | ||||||
|  |  | ||||||
|  | 		return true; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		// if(error.name === 'UserNotFound') return false; | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | User.remove = async function(data){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await deleteLdapUser(client, this); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return true; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.setPassword = async function(data){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		await client.bind(conf.bindDN, conf.bindPassword); | ||||||
|  |  | ||||||
|  | 		await client.modify(this.dn, [ | ||||||
|  | 		  new Change({ | ||||||
|  | 			operation: 'replace', | ||||||
|  | 			modification: new Attribute({ | ||||||
|  | 			  type: 'userPassword', | ||||||
|  | 			  values: ['{MD5}'+crypto.createHash('md5').update(data.userPassword, "binary").digest('base64')]  | ||||||
|  | 			})}), | ||||||
|  | 		]);  | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.invite = async function(){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await InviteToken.add({created_by: this.uid}); | ||||||
|  | 		 | ||||||
|  | 		return token; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | User.login = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await this.get(data.uid); | ||||||
|  |  | ||||||
|  | 		await client.bind(user.dn, data.password); | ||||||
|  |  | ||||||
|  | 		await client.unbind(); | ||||||
|  |  | ||||||
|  | 		return user; | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | module.exports = {User}; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | // (async function(){ | ||||||
|  | // try{ | ||||||
|  | // 	console.log(await User.list()); | ||||||
|  |  | ||||||
|  | // 	console.log(await User.listDetail()); | ||||||
|  |  | ||||||
|  | // 	console.log(await User.get('wmantly')) | ||||||
|  |  | ||||||
|  | // }catch(error){ | ||||||
|  | // 	console.error(error) | ||||||
|  | // } | ||||||
|  | // })() | ||||||
							
								
								
									
										1501
									
								
								nodejs/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										1501
									
								
								nodejs/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										31
									
								
								nodejs/package.json
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										31
									
								
								nodejs/package.json
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | { | ||||||
|  |   "name": "t42-ldap-manager", | ||||||
|  |   "version": "1.0.0", | ||||||
|  |   "private": true, | ||||||
|  |   "author": [ | ||||||
|  |     { | ||||||
|  |       "name": "William Mantly", | ||||||
|  |       "email": "wmantly@gmail.com" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "scripts": { | ||||||
|  |     "start": "node ./bin/www" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "@sendgrid/mail": "^7.1.0", | ||||||
|  |     "ejs": "^3.0.1", | ||||||
|  |     "express": "~4.16.1", | ||||||
|  |     "extend": "^3.0.2", | ||||||
|  |     "ldapts": "^2.2.1", | ||||||
|  |     "moment": "^2.25.3", | ||||||
|  |     "mustache": "^4.0.1", | ||||||
|  |     "nodemon": "^2.0.4", | ||||||
|  |     "redis": "^2.8.0", | ||||||
|  |     "smtpc": "^0.1.2" | ||||||
|  |   }, | ||||||
|  |   "license": "MIT", | ||||||
|  |   "repository": { | ||||||
|  |     "type": "git", | ||||||
|  |     "url": "https://git.theta42.com/wmantly/proxy.git" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								nodejs/public/css/Thumbs.db
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								nodejs/public/css/Thumbs.db
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										384
									
								
								nodejs/public/css/bootstrap-theme.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										384
									
								
								nodejs/public/css/bootstrap-theme.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @ -0,0 +1,384 @@ | |||||||
|  | .btn-default, | ||||||
|  | .btn-primary, | ||||||
|  | .btn-success, | ||||||
|  | .btn-info, | ||||||
|  | .btn-warning, | ||||||
|  | .btn-danger { | ||||||
|  |   text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.2); | ||||||
|  |   -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15), 0 1px 1px rgba(0, 0, 0, 0.075); | ||||||
|  |           box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15), 0 1px 1px rgba(0, 0, 0, 0.075); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-default:active, | ||||||
|  | .btn-primary:active, | ||||||
|  | .btn-success:active, | ||||||
|  | .btn-info:active, | ||||||
|  | .btn-warning:active, | ||||||
|  | .btn-danger:active, | ||||||
|  | .btn-default.active, | ||||||
|  | .btn-primary.active, | ||||||
|  | .btn-success.active, | ||||||
|  | .btn-info.active, | ||||||
|  | .btn-warning.active, | ||||||
|  | .btn-danger.active { | ||||||
|  |   -webkit-box-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125); | ||||||
|  |           box-shadow: inset 0 3px 5px rgba(0, 0, 0, 0.125); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn:active, | ||||||
|  | .btn.active { | ||||||
|  |   background-image: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-default { | ||||||
|  |   text-shadow: 0 1px 0 #fff; | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#ffffff), to(#e6e6e6)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #ffffff, 0%, #e6e6e6, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #ffffff 0%, #e6e6e6 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #ffffff 0%, #e6e6e6 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #e0e0e0; | ||||||
|  |   border-color: #ccc; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#ffe6e6e6', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-default:active, | ||||||
|  | .btn-default.active { | ||||||
|  |   background-color: #e6e6e6; | ||||||
|  |   border-color: #e0e0e0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-primary { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#428bca), to(#3071a9)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #428bca, 0%, #3071a9, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #428bca 0%, #3071a9 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #428bca 0%, #3071a9 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #2d6ca2; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff428bca', endColorstr='#ff3071a9', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-primary:active, | ||||||
|  | .btn-primary.active { | ||||||
|  |   background-color: #3071a9; | ||||||
|  |   border-color: #2d6ca2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-success { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#5cb85c), to(#449d44)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #5cb85c, 0%, #449d44, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #5cb85c 0%, #449d44 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #5cb85c 0%, #449d44 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #419641; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff449d44', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-success:active, | ||||||
|  | .btn-success.active { | ||||||
|  |   background-color: #449d44; | ||||||
|  |   border-color: #419641; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-warning { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#f0ad4e), to(#ec971f)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #f0ad4e, 0%, #ec971f, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #f0ad4e 0%, #ec971f 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #f0ad4e 0%, #ec971f 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #eb9316; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffec971f', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-warning:active, | ||||||
|  | .btn-warning.active { | ||||||
|  |   background-color: #ec971f; | ||||||
|  |   border-color: #eb9316; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-danger { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#d9534f), to(#c9302c)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #d9534f, 0%, #c9302c, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #d9534f 0%, #c9302c 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #d9534f 0%, #c9302c 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #c12e2a; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc9302c', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-danger:active, | ||||||
|  | .btn-danger.active { | ||||||
|  |   background-color: #c9302c; | ||||||
|  |   border-color: #c12e2a; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-info { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#5bc0de), to(#31b0d5)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #5bc0de, 0%, #31b0d5, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #5bc0de 0%, #31b0d5 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #5bc0de 0%, #31b0d5 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #2aabd2; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff31b0d5', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .btn-info:active, | ||||||
|  | .btn-info.active { | ||||||
|  |   background-color: #31b0d5; | ||||||
|  |   border-color: #2aabd2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .thumbnail, | ||||||
|  | .img-thumbnail { | ||||||
|  |   -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, 0.075); | ||||||
|  |           box-shadow: 0 1px 2px rgba(0, 0, 0, 0.075); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dropdown-menu > li > a:hover, | ||||||
|  | .dropdown-menu > li > a:focus, | ||||||
|  | .dropdown-menu > .active > a, | ||||||
|  | .dropdown-menu > .active > a:hover, | ||||||
|  | .dropdown-menu > .active > a:focus { | ||||||
|  |   background-color: #357ebd; | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#428bca), to(#357ebd)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #428bca, 0%, #357ebd, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #428bca 0%, #357ebd 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #428bca 0%, #357ebd 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff428bca', endColorstr='#ff357ebd', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#ffffff), to(#f8f8f8)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #ffffff, 0%, #f8f8f8, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #ffffff 0%, #f8f8f8 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #ffffff 0%, #f8f8f8 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#fff8f8f8', GradientType=0); | ||||||
|  |   -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15), 0 1px 5px rgba(0, 0, 0, 0.075); | ||||||
|  |           box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.15), 0 1px 5px rgba(0, 0, 0, 0.075); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar .navbar-nav > .active > a { | ||||||
|  |   background-color: #f8f8f8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-brand, | ||||||
|  | .navbar-nav > li > a { | ||||||
|  |   text-shadow: 0 1px 0 rgba(255, 255, 255, 0.25); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-inverse { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#3c3c3c), to(#222222)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #3c3c3c, 0%, #222222, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #3c3c3c 0%, #222222 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #3c3c3c 0%, #222222 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff3c3c3c', endColorstr='#ff222222', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-inverse .navbar-nav > .active > a { | ||||||
|  |   background-color: #222222; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-inverse .navbar-brand, | ||||||
|  | .navbar-inverse .navbar-nav > li > a { | ||||||
|  |   text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-static-top, | ||||||
|  | .navbar-fixed-top, | ||||||
|  | .navbar-fixed-bottom { | ||||||
|  |   border-radius: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .alert { | ||||||
|  |   text-shadow: 0 1px 0 rgba(255, 255, 255, 0.2); | ||||||
|  |   -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.25), 0 1px 2px rgba(0, 0, 0, 0.05); | ||||||
|  |           box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.25), 0 1px 2px rgba(0, 0, 0, 0.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .alert-success { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#dff0d8), to(#c8e5bc)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #dff0d8, 0%, #c8e5bc, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #dff0d8 0%, #c8e5bc 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #dff0d8 0%, #c8e5bc 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #b2dba1; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffc8e5bc', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .alert-info { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#d9edf7), to(#b9def0)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #d9edf7, 0%, #b9def0, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #d9edf7 0%, #b9def0 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #d9edf7 0%, #b9def0 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #9acfea; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffb9def0', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .alert-warning { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#fcf8e3), to(#f8efc0)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #fcf8e3, 0%, #f8efc0, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #fcf8e3 0%, #f8efc0 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #fcf8e3 0%, #f8efc0 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #f5e79e; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fff8efc0', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .alert-danger { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#f2dede), to(#e7c3c3)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #f2dede, 0%, #e7c3c3, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #f2dede 0%, #e7c3c3 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #f2dede 0%, #e7c3c3 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #dca7a7; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffe7c3c3', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#ebebeb), to(#f5f5f5)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #ebebeb, 0%, #f5f5f5, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #ebebeb 0%, #f5f5f5 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #ebebeb 0%, #f5f5f5 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffebebeb', endColorstr='#fff5f5f5', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress-bar { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#428bca), to(#3071a9)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #428bca, 0%, #3071a9, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #428bca 0%, #3071a9 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #428bca 0%, #3071a9 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff428bca', endColorstr='#ff3071a9', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress-bar-success { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#5cb85c), to(#449d44)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #5cb85c, 0%, #449d44, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #5cb85c 0%, #449d44 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #5cb85c 0%, #449d44 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff449d44', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress-bar-info { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#5bc0de), to(#31b0d5)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #5bc0de, 0%, #31b0d5, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #5bc0de 0%, #31b0d5 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #5bc0de 0%, #31b0d5 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff31b0d5', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress-bar-warning { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#f0ad4e), to(#ec971f)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #f0ad4e, 0%, #ec971f, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #f0ad4e 0%, #ec971f 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #f0ad4e 0%, #ec971f 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffec971f', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .progress-bar-danger { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#d9534f), to(#c9302c)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #d9534f, 0%, #c9302c, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #d9534f 0%, #c9302c 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #d9534f 0%, #c9302c 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc9302c', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .list-group { | ||||||
|  |   border-radius: 4px; | ||||||
|  |   -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, 0.075); | ||||||
|  |           box-shadow: 0 1px 2px rgba(0, 0, 0, 0.075); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .list-group-item.active, | ||||||
|  | .list-group-item.active:hover, | ||||||
|  | .list-group-item.active:focus { | ||||||
|  |   text-shadow: 0 -1px 0 #3071a9; | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#428bca), to(#3278b3)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #428bca, 0%, #3278b3, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #428bca 0%, #3278b3 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #428bca 0%, #3278b3 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #3278b3; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff428bca', endColorstr='#ff3278b3', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel { | ||||||
|  |   -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05); | ||||||
|  |           box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-default > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#f5f5f5), to(#e8e8e8)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #f5f5f5, 0%, #e8e8e8, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #f5f5f5 0%, #e8e8e8 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #f5f5f5 0%, #e8e8e8 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#ffe8e8e8', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-primary > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#428bca), to(#357ebd)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #428bca, 0%, #357ebd, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #428bca 0%, #357ebd 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #428bca 0%, #357ebd 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff428bca', endColorstr='#ff357ebd', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-success > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#dff0d8), to(#d0e9c6)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #dff0d8, 0%, #d0e9c6, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #dff0d8 0%, #d0e9c6 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #dff0d8 0%, #d0e9c6 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffd0e9c6', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-info > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#d9edf7), to(#c4e3f3)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #d9edf7, 0%, #c4e3f3, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #d9edf7 0%, #c4e3f3 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #d9edf7 0%, #c4e3f3 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffc4e3f3', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-warning > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#fcf8e3), to(#faf2cc)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #fcf8e3, 0%, #faf2cc, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #fcf8e3 0%, #faf2cc 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #fcf8e3 0%, #faf2cc 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fffaf2cc', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .panel-danger > .panel-heading { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#f2dede), to(#ebcccc)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #f2dede, 0%, #ebcccc, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #f2dede 0%, #ebcccc 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #f2dede 0%, #ebcccc 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffebcccc', GradientType=0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .well { | ||||||
|  |   background-image: -webkit-gradient(linear, left 0%, left 100%, from(#e8e8e8), to(#f5f5f5)); | ||||||
|  |   background-image: -webkit-linear-gradient(top, #e8e8e8, 0%, #f5f5f5, 100%); | ||||||
|  |   background-image: -moz-linear-gradient(top, #e8e8e8 0%, #f5f5f5 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #e8e8e8 0%, #f5f5f5 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   border-color: #dcdcdc; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffe8e8e8', endColorstr='#fff5f5f5', GradientType=0); | ||||||
|  |   -webkit-box-shadow: inset 0 1px 3px rgba(0, 0, 0, 0.05), 0 1px 0 rgba(255, 255, 255, 0.1); | ||||||
|  |           box-shadow: inset 0 1px 3px rgba(0, 0, 0, 0.05), 0 1px 0 rgba(255, 255, 255, 0.1); | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								nodejs/public/css/bootstrap-theme.min.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								nodejs/public/css/bootstrap-theme.min.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										6805
									
								
								nodejs/public/css/bootstrap.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										6805
									
								
								nodejs/public/css/bootstrap.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										9
									
								
								nodejs/public/css/bootstrap.min.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										9
									
								
								nodejs/public/css/bootstrap.min.css
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										24
									
								
								nodejs/public/css/styles.css
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										24
									
								
								nodejs/public/css/styles.css
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,24 @@ | |||||||
|  |  | ||||||
|  | *{ | ||||||
|  | 	transition: all .4s ease; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a:hover, button:hover, div.form-group:hover{ | ||||||
|  | 	color: inherit; | ||||||
|  | 	transition: all .4s ease; | ||||||
|  | 	background-color: inherit; | ||||||
|  | 	-ms-transform: scale(1.05); | ||||||
|  | 	-webkit-transform: scale(1.05);  | ||||||
|  | 	transform: scale(1.05); /* Standard syntax */ | ||||||
|  | 	z-index: 999999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.card-body{ | ||||||
|  | 	padding-left: 1.5em; | ||||||
|  | 	padding-right: 1.5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .my-2 > a { | ||||||
|  | 	padding-right: .5em; | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.eot
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.eot
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										228
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.svg
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										228
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.svg
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,228 @@ | |||||||
|  | <?xml version="1.0" standalone="no"?> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" > | ||||||
|  | <svg xmlns="http://www.w3.org/2000/svg"> | ||||||
|  | <metadata></metadata> | ||||||
|  | <defs> | ||||||
|  | <font id="glyphicons_halflingsregular" horiz-adv-x="1200" > | ||||||
|  | <font-face units-per-em="1200" ascent="960" descent="-240" /> | ||||||
|  | <missing-glyph horiz-adv-x="500" /> | ||||||
|  | <glyph /> | ||||||
|  | <glyph /> | ||||||
|  | <glyph unicode=" " /> | ||||||
|  | <glyph unicode="*" d="M1100 500h-259l183 -183l-141 -141l-183 183v-259h-200v259l-183 -183l-141 141l183 183h-259v200h259l-183 183l141 141l183 -183v259h200v-259l183 183l141 -141l-183 -183h259v-200z" /> | ||||||
|  | <glyph unicode="+" d="M1100 400h-400v-400h-300v400h-400v300h400v400h300v-400h400v-300z" /> | ||||||
|  | <glyph unicode=" " /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="652" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="1304" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="652" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="1304" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="434" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="326" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="217" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="217" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="163" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="260" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="72" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="260" /> | ||||||
|  | <glyph unicode=" " horiz-adv-x="326" /> | ||||||
|  | <glyph unicode="€" d="M800 500h-300q9 -74 33 -132t52.5 -91t62 -54.5t59 -29t46.5 -7.5q29 0 66 13t75 37t63.5 67.5t25.5 96.5h174q-31 -172 -128 -278q-107 -117 -274 -117q-205 0 -324 158q-36 46 -69 131.5t-45 205.5h-217l100 100h113q0 47 5 100h-218l100 100h135q37 167 112 257 q117 141 297 141q242 0 354 -189q60 -103 66 -209h-181q0 55 -25.5 99t-63.5 68t-75 36.5t-67 12.5q-24 0 -52.5 -10t-62.5 -32t-65.5 -67t-50.5 -107h379l-100 -100h-300q-6 -46 -6 -100h406z" /> | ||||||
|  | <glyph unicode="−" d="M1100 700h-900v-300h900v300z" /> | ||||||
|  | <glyph unicode="☁" d="M178 300h750q120 0 205 86t85 208q0 120 -85 206.5t-205 86.5q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5q0 -80 56.5 -137t135.5 -57z" /> | ||||||
|  | <glyph unicode="✉" d="M1200 1100h-1200l600 -603zM300 600l-300 -300v600zM1200 900v-600l-300 300zM800 500l400 -400h-1200l400 400l200 -200z" /> | ||||||
|  | <glyph unicode="✏" d="M1101 889l99 92q13 13 13 32.5t-13 33.5l-153 153q-15 13 -33 13t-33 -13l-94 -97zM401 189l614 614l-214 214l-614 -614zM-13 -13l333 112l-223 223z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="500" d="M0 0z" /> | ||||||
|  | <glyph unicode="" d="M700 100h300v-100h-800v100h300v550l-500 550h1200l-500 -550v-550z" /> | ||||||
|  | <glyph unicode="" d="M1000 934v-521q-64 16 -138 -7q-79 -26 -122.5 -83t-25.5 -111q17 -55 85.5 -75.5t147.5 4.5q70 23 111.5 63.5t41.5 95.5v881q0 10 -7 15.5t-17 2.5l-752 -193q-10 -3 -17 -12.5t-7 -19.5v-689q-64 17 -138 -7q-79 -25 -122.5 -82t-25.5 -112t86 -75.5t147 5.5 q65 21 109 69t44 90v606z" /> | ||||||
|  | <glyph unicode="" d="M913 432l300 -300q7 -8 7 -18t-7 -18l-109 -109q-8 -7 -18 -7t-18 7l-300 300q-119 -78 -261 -78q-200 0 -342 142t-142 342t142 342t342 142t342 -142t142 -342q0 -142 -78 -261zM176 693q0 -136 97 -233t234 -97t233.5 96.5t96.5 233.5t-96.5 233.5t-233.5 96.5 t-234 -97t-97 -233z" /> | ||||||
|  | <glyph unicode="" d="M649 949q48 69 109.5 105t121.5 38t118.5 -20.5t102.5 -64t71 -100.5t27 -123q0 -57 -33.5 -117.5t-94 -124.5t-126.5 -127.5t-150 -152.5t-146 -174q-62 85 -145.5 174t-149.5 152.5t-126.5 127.5t-94 124.5t-33.5 117.5q0 64 28 123t73 100.5t104.5 64t119 20.5 t120 -38.5t104.5 -104.5z" /> | ||||||
|  | <glyph unicode="" d="M791 522l145 -449l-384 275l-382 -275l146 447l-388 280h479l146 400h2l146 -400h472zM168 71l2 1z" /> | ||||||
|  | <glyph unicode="" d="M791 522l145 -449l-384 275l-382 -275l146 447l-388 280h479l146 400h2l146 -400h472zM747 331l-74 229l193 140h-235l-77 211l-78 -211h-239l196 -142l-73 -226l192 140zM168 71l2 1z" /> | ||||||
|  | <glyph unicode="" d="M1200 143v-143h-1200v143l400 257v100q-37 0 -68.5 74.5t-31.5 125.5v200q0 124 88 212t212 88t212 -88t88 -212v-200q0 -51 -31.5 -125.5t-68.5 -74.5v-100z" /> | ||||||
|  | <glyph unicode="" d="M1200 1100v-1100h-1200v1100h1200zM200 1000h-100v-100h100v100zM900 1000h-600v-400h600v400zM1100 1000h-100v-100h100v100zM200 800h-100v-100h100v100zM1100 800h-100v-100h100v100zM200 600h-100v-100h100v100zM1100 600h-100v-100h100v100zM900 500h-600v-400h600 v400zM200 400h-100v-100h100v100zM1100 400h-100v-100h100v100zM200 200h-100v-100h100v100zM1100 200h-100v-100h100v100z" /> | ||||||
|  | <glyph unicode="" d="M500 1050v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5zM1100 1050v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5h400 q21 0 35.5 -14.5t14.5 -35.5zM500 450v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5zM1100 450v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5v400 q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5z" /> | ||||||
|  | <glyph unicode="" d="M300 1050v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM700 1050v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200 q21 0 35.5 -14.5t14.5 -35.5zM1100 1050v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM300 650v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200 q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM700 650v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM1100 650v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200 q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM300 250v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM700 250v-200 q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM1100 250v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5 t14.5 -35.5z" /> | ||||||
|  | <glyph unicode="" d="M300 1050v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM1200 1050v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h700 q21 0 35.5 -14.5t14.5 -35.5zM300 450v200q0 21 -14.5 35.5t-35.5 14.5h-200q-21 0 -35.5 -14.5t-14.5 -35.5v-200q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5zM1200 650v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5v200 q0 21 14.5 35.5t35.5 14.5h700q21 0 35.5 -14.5t14.5 -35.5zM300 250v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5zM1200 250v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700 q-21 0 -35.5 14.5t-14.5 35.5v200q0 21 14.5 35.5t35.5 14.5h700q21 0 35.5 -14.5t14.5 -35.5z" /> | ||||||
|  | <glyph unicode="" d="M448 34l818 820l-212 212l-607 -607l-206 207l-212 -212z" /> | ||||||
|  | <glyph unicode="" d="M882 106l-282 282l-282 -282l-212 212l282 282l-282 282l212 212l282 -282l282 282l212 -212l-282 -282l282 -282z" /> | ||||||
|  | <glyph unicode="" d="M913 432l300 -300q7 -8 7 -18t-7 -18l-109 -109q-8 -7 -18 -7t-18 7l-300 300q-119 -78 -261 -78q-200 0 -342 142t-142 342t142 342t342 142t342 -142t142 -342q0 -142 -78 -261zM507 363q137 0 233.5 96.5t96.5 233.5t-96.5 233.5t-233.5 96.5t-234 -97t-97 -233 t97 -233t234 -97zM600 800h100v-200h-100v-100h-200v100h-100v200h100v100h200v-100z" /> | ||||||
|  | <glyph unicode="" d="M913 432l300 -299q7 -7 7 -18t-7 -18l-109 -109q-8 -8 -18 -8t-18 8l-300 299q-120 -77 -261 -77q-200 0 -342 142t-142 342t142 342t342 142t342 -142t142 -342q0 -141 -78 -262zM176 694q0 -136 97 -233t234 -97t233.5 97t96.5 233t-96.5 233t-233.5 97t-234 -97 t-97 -233zM300 801v-200h400v200h-400z" /> | ||||||
|  | <glyph unicode="" d="M700 750v400q0 21 -14.5 35.5t-35.5 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-400q0 -21 14.5 -35.5t35.5 -14.5h100q21 0 35.5 14.5t14.5 35.5zM800 975v166q167 -62 272 -210t105 -331q0 -118 -45.5 -224.5t-123 -184t-184 -123t-224.5 -45.5t-224.5 45.5t-184 123 t-123 184t-45.5 224.5q0 183 105 331t272 210v-166q-103 -55 -165 -155t-62 -220q0 -177 125 -302t302 -125t302 125t125 302q0 120 -62 220t-165 155z" /> | ||||||
|  | <glyph unicode="" d="M1200 1h-200v1200h200v-1200zM900 1h-200v800h200v-800zM600 1h-200v500h200v-500zM300 301h-200v-300h200v300z" /> | ||||||
|  | <glyph unicode="" d="M488 183l38 -151q40 -5 74 -5q27 0 74 5l38 151l6 2q46 13 93 39l5 3l134 -81q56 44 104 105l-80 134l3 5q24 44 39 93l1 6l152 38q5 40 5 74q0 28 -5 73l-152 38l-1 6q-16 51 -39 93l-3 5l80 134q-44 58 -104 105l-134 -81l-5 3q-45 25 -93 39l-6 1l-38 152q-40 5 -74 5 q-27 0 -74 -5l-38 -152l-5 -1q-50 -14 -94 -39l-5 -3l-133 81q-59 -47 -105 -105l80 -134l-3 -5q-25 -47 -38 -93l-2 -6l-151 -38q-6 -48 -6 -73q0 -33 6 -74l151 -38l2 -6q14 -49 38 -93l3 -5l-80 -134q45 -59 105 -105l133 81l5 -3q45 -26 94 -39zM600 815q89 0 152 -63 t63 -151q0 -89 -63 -152t-152 -63t-152 63t-63 152q0 88 63 151t152 63z" /> | ||||||
|  | <glyph unicode="" d="M900 1100h275q10 0 17.5 -7.5t7.5 -17.5v-50q0 -11 -7 -18t-18 -7h-1050q-11 0 -18 7t-7 18v50q0 10 7.5 17.5t17.5 7.5h275v100q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5v-100zM800 1100v100h-300v-100h300zM200 900h900v-800q0 -41 -29.5 -71 t-70.5 -30h-700q-41 0 -70.5 30t-29.5 71v800zM300 100h100v700h-100v-700zM500 100h100v700h-100v-700zM700 100h100v700h-100v-700zM900 100h100v700h-100v-700z" /> | ||||||
|  | <glyph unicode="" d="M1301 601h-200v-600h-300v400h-300v-400h-300v600h-200l656 644z" /> | ||||||
|  | <glyph unicode="" d="M600 700h400v-675q0 -11 -7 -18t-18 -7h-850q-11 0 -18 7t-7 18v1150q0 11 7 18t18 7h475v-500zM1000 800h-300v300z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM600 600h200 v-100h-300v400h100v-300z" /> | ||||||
|  | <glyph unicode="" d="M721 400h-242l-40 -400h-539l431 1200h209l-21 -300h162l-20 300h208l431 -1200h-538zM712 500l-27 300h-170l-27 -300h224z" /> | ||||||
|  | <glyph unicode="" d="M1100 400v-400h-1100v400h490l-290 300h200v500h300v-500h200l-290 -300h490zM988 300h-175v-100h175v100z" /> | ||||||
|  | <glyph unicode="" d="M600 1199q122 0 233 -47.5t191 -127.5t127.5 -191t47.5 -233t-47.5 -233t-127.5 -191t-191 -127.5t-233 -47.5t-233 47.5t-191 127.5t-127.5 191t-47.5 233t47.5 233t127.5 191t191 127.5t233 47.5zM600 1012q-170 0 -291 -121t-121 -291t121 -291t291 -121t291 121 t121 291t-121 291t-291 121zM700 600h150l-250 -300l-250 300h150v300h200v-300z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM850 600h-150 v-300h-200v300h-150l250 300z" /> | ||||||
|  | <glyph unicode="" d="M0 500l200 700h800q199 -700 200 -700v-475q0 -11 -7 -18t-18 -7h-1150q-11 0 -18 7t-7 18v475zM903 1000h-606l-97 -500h200l50 -200h300l50 200h200z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5q0 -172 121.5 -293t292.5 -121t292.5 121t121.5 293q0 171 -121.5 292.5t-292.5 121.5zM797 598 l-297 -201v401z" /> | ||||||
|  | <glyph unicode="" d="M1177 600h-150q0 -177 -125 -302t-302 -125t-302 125t-125 302t125 302t302 125q136 0 246 -81l-146 -146h400v400l-145 -145q-157 122 -355 122q-118 0 -224.5 -45.5t-184 -123t-123 -184t-45.5 -224.5t45.5 -224.5t123 -184t184 -123t224.5 -45.5t224.5 45.5t184 123 t123 184t45.5 224.5z" /> | ||||||
|  | <glyph unicode="" d="M700 800l147 147q-112 80 -247 80q-177 0 -302 -125t-125 -302h-150q0 118 45.5 224.5t123 184t184 123t224.5 45.5q198 0 355 -122l145 145v-400h-400zM500 400l-147 -147q112 -80 247 -80q177 0 302 125t125 302h150q0 -118 -45.5 -224.5t-123 -184t-184 -123 t-224.5 -45.5q-198 0 -355 122l-145 -145v400h400z" /> | ||||||
|  | <glyph unicode="" d="M100 1200v-1200h1100v1200h-1100zM1100 100h-900v900h900v-900zM400 800h-100v100h100v-100zM1000 800h-500v100h500v-100zM400 600h-100v100h100v-100zM1000 600h-500v100h500v-100zM400 400h-100v100h100v-100zM1000 400h-500v100h500v-100zM400 200h-100v100h100v-100 zM1000 300h-500v-100h500v100z" /> | ||||||
|  | <glyph unicode="" d="M200 0h-100v1100h100v-1100zM1100 600v500q-40 -81 -101.5 -115.5t-127.5 -29.5t-138 25t-139.5 40t-125.5 25t-103 -29.5t-65 -115.5v-500q60 60 127.5 84t127.5 17.5t122 -23t119 -30t110 -11t103 42t91 120.5z" /> | ||||||
|  | <glyph unicode="" d="M1200 275v300q0 116 -49.5 227t-131 192.5t-192.5 131t-227 49.5t-227 -49.5t-192.5 -131t-131 -192.5t-49.5 -227v-300q0 -11 7 -18t18 -7h50q11 0 18 7t7 18v300q0 127 70.5 231.5t184.5 161.5t245 57t245 -57t184.5 -161.5t70.5 -231.5v-300q0 -11 7 -18t18 -7h50 q11 0 18 7t7 18zM400 480v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14v460q0 8 6 14t14 6h160q8 0 14 -6t6 -14zM1000 480v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14v460q0 8 6 14t14 6h160q8 0 14 -6t6 -14z" /> | ||||||
|  | <glyph unicode="" d="M0 800v-400h300l300 -200v800l-300 -200h-300zM971 600l141 -141l-71 -71l-141 141l-141 -141l-71 71l141 141l-141 141l71 71l141 -141l141 141l71 -71z" /> | ||||||
|  | <glyph unicode="" d="M0 800v-400h300l300 -200v800l-300 -200h-300zM700 857l69 53q111 -135 111 -310q0 -169 -106 -302l-67 54q86 110 86 248q0 146 -93 257z" /> | ||||||
|  | <glyph unicode="" d="M974 186l6 8q142 178 142 405q0 230 -144 408l-6 8l-83 -64l7 -8q123 -151 123 -344q0 -189 -119 -339l-7 -8zM300 801l300 200v-800l-300 200h-300v400h300zM702 858l69 53q111 -135 111 -310q0 -170 -106 -303l-67 55q86 110 86 248q0 145 -93 257z" /> | ||||||
|  | <glyph unicode="" d="M100 700h400v100h100v100h-100v300h-500v-600h100v100zM1200 700v500h-600v-200h100v-300h200v-300h300v200h-200v100h200zM100 1100h300v-300h-300v300zM800 800v300h300v-300h-300zM200 900h100v100h-100v-100zM900 1000h100v-100h-100v100zM300 600h-100v-100h-200 v-500h500v500h-200v100zM900 200v-100h-200v100h-100v100h100v200h-200v100h300v-300h200v-100h-100zM400 400v-300h-300v300h300zM300 200h-100v100h100v-100zM1100 300h100v-100h-100v100zM600 100h100v-100h-100v100zM1200 100v-100h-300v100h300z" /> | ||||||
|  | <glyph unicode="" d="M100 1200h-100v-1000h100v1000zM300 200h-100v1000h100v-1000zM700 200h-200v1000h200v-1000zM900 200h-100v1000h100v-1000zM1200 1200v-1000h-200v1000h200zM400 100v-100h-300v100h300zM500 91h100v-91h-100v91zM700 91h100v-91h-100v91zM1100 91v-91h-200v91h200z " /> | ||||||
|  | <glyph unicode="" d="M1200 500l-500 -500l-699 700v475q0 10 7.5 17.5t17.5 7.5h474zM320 882q29 29 29 71t-29 71q-30 30 -71.5 30t-71.5 -30q-29 -29 -29 -71t29 -71q30 -30 71.5 -30t71.5 30z" /> | ||||||
|  | <glyph unicode="" d="M1201 500l-500 -500l-699 700v475q0 11 7 18t18 7h474zM1501 500l-500 -500l-50 50l450 450l-700 700h100zM320 882q30 29 30 71t-30 71q-29 30 -71 30t-71 -30q-30 -29 -30 -71t30 -71q29 -30 71 -30t71 30z" /> | ||||||
|  | <glyph unicode="" d="M1200 1200v-1000l-100 -100v1000h-750l-100 -100h750v-1000h-900v1025l175 175h925z" /> | ||||||
|  | <glyph unicode="" d="M947 829l-94 346q-2 11 -10 18t-18 7h-450q-10 0 -18 -7t-10 -18l-94 -346l40 -124h592zM1200 800v-700h-200v200h-800v-200h-200v700h200l100 -200h600l100 200h200zM881 176l38 -152q2 -10 -3.5 -17t-15.5 -7h-600q-10 0 -15.5 7t-3.5 17l38 152q2 10 11.5 17t19.5 7 h500q10 0 19.5 -7t11.5 -17z" /> | ||||||
|  | <glyph unicode="" d="M1200 0v66q-34 1 -74 43q-18 19 -33 42t-21 37l-6 13l-385 998h-93l-399 -1006q-24 -48 -52 -75q-12 -12 -33 -25t-36 -20l-15 -7v-66h365v66q-41 0 -72 11t-49 38t1 71l92 234h391l82 -222q16 -45 -5.5 -88.5t-74.5 -43.5v-66h417zM416 521l178 457l46 -140l116 -317 h-340z" /> | ||||||
|  | <glyph unicode="" d="M100 1199h471q120 0 213 -88t93 -228q0 -55 -11.5 -101.5t-28 -74t-33.5 -47.5t-28 -28l-12 -7q8 -3 21.5 -9t48 -31.5t60.5 -58t47.5 -91.5t21.5 -129q0 -84 -59 -156.5t-142 -111t-162 -38.5h-500v89q41 7 70.5 32.5t29.5 65.5v827q0 28 -1 39.5t-5.5 26t-15.5 21 t-29 14t-49 14.5v70zM400 1079v-379h139q76 0 130 61.5t54 138.5q0 82 -84 130.5t-239 48.5zM400 200h161q89 0 153 48.5t64 132.5q0 90 -62.5 154.5t-156.5 64.5h-159v-400z" /> | ||||||
|  | <glyph unicode="" d="M877 1200l2 -57q-33 -8 -62 -25.5t-46 -37t-29.5 -38t-17.5 -30.5l-5 -12l-128 -825q-10 -52 14 -82t95 -36v-57h-500v57q77 7 134.5 40.5t65.5 80.5l173 849q10 56 -10 74t-91 37q-6 1 -10.5 2.5t-9.5 2.5v57h425z" /> | ||||||
|  | <glyph unicode="" d="M1150 1200h150v-300h-50q0 29 -8 48.5t-18.5 30t-33.5 15t-39.5 5.5t-50.5 1h-200v-850l100 -50v-100h-400v100l100 50v850h-200q-34 0 -50.5 -1t-40 -5.5t-33.5 -15t-18.5 -30t-8.5 -48.5h-49v300h150h700zM100 1000v-800h75l-125 -167l-125 167h75v800h-75l125 167 l125 -167h-75z" /> | ||||||
|  | <glyph unicode="" d="M950 1201h150v-300h-50q0 29 -8 48.5t-18 30t-33.5 15t-40 5.5t-50.5 1h-200v-650l100 -50v-100h-400v100l100 50v650h-200q-34 0 -50.5 -1t-39.5 -5.5t-33.5 -15t-18.5 -30t-8 -48.5h-50v300h150h700zM200 101h800v75l167 -125l-167 -125v75h-800v-75l-167 125l167 125 v-75z" /> | ||||||
|  | <glyph unicode="" d="M700 950v100q0 21 -14.5 35.5t-35.5 14.5h-600q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h600q21 0 35.5 15t14.5 35zM1100 650v100q0 21 -14.5 35.5t-35.5 14.5h-1000q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h1000 q21 0 35.5 15t14.5 35zM900 350v100q0 21 -14.5 35.5t-35.5 14.5h-800q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h800q21 0 35.5 15t14.5 35zM1200 50v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35 t35.5 -15h1100q21 0 35.5 15t14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M1000 950v100q0 21 -14.5 35.5t-35.5 14.5h-700q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h700q21 0 35.5 15t14.5 35zM1200 650v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h1100 q21 0 35.5 15t14.5 35zM1000 350v100q0 21 -14.5 35.5t-35.5 14.5h-700q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h700q21 0 35.5 15t14.5 35zM1200 50v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35 t35.5 -15h1100q21 0 35.5 15t14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M500 950v100q0 21 14.5 35.5t35.5 14.5h600q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-600q-21 0 -35.5 15t-14.5 35zM100 650v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1000q-21 0 -35.5 15 t-14.5 35zM300 350v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800q-21 0 -35.5 15t-14.5 35zM0 50v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100 q-21 0 -35.5 15t-14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M0 950v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15t-14.5 35zM0 650v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15 t-14.5 35zM0 350v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15t-14.5 35zM0 50v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100 q-21 0 -35.5 15t-14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M0 950v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM300 950v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800q-21 0 -35.5 15 t-14.5 35zM0 650v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM300 650v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800 q-21 0 -35.5 15t-14.5 35zM0 350v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM300 350v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15 h-800q-21 0 -35.5 15t-14.5 35zM0 50v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM300 50v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15 h-800q-21 0 -35.5 15t-14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M400 1100h-100v-1100h100v1100zM700 950v100q0 21 -15 35.5t-35 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h100q20 0 35 15t15 35zM1100 650v100q0 21 -15 35.5t-35 14.5h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15 h500q20 0 35 15t15 35zM100 425v75h-201v100h201v75l166 -125zM900 350v100q0 21 -15 35.5t-35 14.5h-300q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h300q20 0 35 15t15 35zM1200 50v100q0 21 -15 35.5t-35 14.5h-600q-21 0 -35.5 -14.5t-14.5 -35.5 v-100q0 -20 14.5 -35t35.5 -15h600q20 0 35 15t15 35z" /> | ||||||
|  | <glyph unicode="" d="M201 950v100q0 21 -15 35.5t-35 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h100q20 0 35 15t15 35zM801 1100h100v-1100h-100v1100zM601 650v100q0 21 -15 35.5t-35 14.5h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15 h500q20 0 35 15t15 35zM1101 425v75h200v100h-200v75l-167 -125zM401 350v100q0 21 -15 35.5t-35 14.5h-300q-21 0 -35.5 -14.5t-14.5 -35.5v-100q0 -20 14.5 -35t35.5 -15h300q20 0 35 15t15 35zM701 50v100q0 21 -15 35.5t-35 14.5h-600q-21 0 -35.5 -14.5t-14.5 -35.5 v-100q0 -20 14.5 -35t35.5 -15h600q20 0 35 15t15 35z" /> | ||||||
|  | <glyph unicode="" d="M900 925v-650q0 -31 -22 -53t-53 -22h-750q-31 0 -53 22t-22 53v650q0 31 22 53t53 22h750q31 0 53 -22t22 -53zM1200 300l-300 300l300 300v-600z" /> | ||||||
|  | <glyph unicode="" d="M1200 1056v-1012q0 -18 -12.5 -31t-31.5 -13h-1112q-18 0 -31 13t-13 31v1012q0 18 13 31t31 13h1112q19 0 31.5 -13t12.5 -31zM1100 1000h-1000v-737l247 182l298 -131l-74 156l293 318l236 -288v500zM476 750q0 -56 -39 -95t-95 -39t-95 39t-39 95t39 95t95 39t95 -39 t39 -95z" /> | ||||||
|  | <glyph unicode="" d="M600 1213q123 0 227 -63t164.5 -169.5t60.5 -229.5t-73 -272q-73 -114 -166.5 -237t-150.5 -189l-57 -66q-10 9 -27 26t-66.5 70.5t-96 109t-104 135.5t-100.5 155q-63 139 -63 262q0 124 60.5 231.5t165 172t226.5 64.5zM599 514q107 0 182.5 75.5t75.5 182.5t-75.5 182 t-182.5 75t-182 -75.5t-75 -181.5q0 -107 75.5 -182.5t181.5 -75.5z" /> | ||||||
|  | <glyph unicode="" d="M600 1199q122 0 233 -47.5t191 -127.5t127.5 -191t47.5 -233t-47.5 -233t-127.5 -191t-191 -127.5t-233 -47.5t-233 47.5t-191 127.5t-127.5 191t-47.5 233t47.5 233t127.5 191t191 127.5t233 47.5zM600 173v854q-176 0 -301.5 -125t-125.5 -302t125.5 -302t301.5 -125z " /> | ||||||
|  | <glyph unicode="" d="M554 1295q21 -71 57.5 -142.5t76 -130.5t83 -118.5t82 -117t70 -116t50 -125.5t18.5 -136q0 -89 -39 -165.5t-102 -126.5t-140 -79.5t-156 -33.5q-114 6 -211.5 53t-161.5 138.5t-64 210.5q0 94 34 186t88.5 172.5t112 159t115 177t87.5 194.5zM455 296q-7 6 -18 17 t-34 48t-33 77q-15 73 -14 143.5t10 122.5l9 51q-92 -110 -119.5 -185t-12.5 -156q14 -82 59.5 -136t136.5 -80z" /> | ||||||
|  | <glyph unicode="" d="M1108 902l113 113l-21 85l-92 28l-113 -113zM1100 625v-225q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5v300q0 165 117.5 282.5t282.5 117.5q366 -6 397 -14l-186 -186h-311q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5 t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v125zM436 341l161 50l412 412l-114 113l-405 -405z" /> | ||||||
|  | <glyph unicode="" d="M1100 453v-53q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5v300q0 165 117.5 282.5t282.5 117.5h261l2 -80q-133 -32 -218 -120h-145q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5z M813 431l360 324l-359 318v-216q-7 0 -19 -1t-48 -8t-69.5 -18.5t-76.5 -37t-76.5 -59t-62 -88t-39.5 -121.5q30 38 81.5 64t103 35.5t99 14t77.5 3.5l29 -1v-209z" /> | ||||||
|  | <glyph unicode="" d="M1100 569v-169q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5v300q0 165 117.5 282.5t282.5 117.5h300q60 0 127 -23l-178 -177h-349q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v69z M625 348l566 567l-136 137l-430 -431l-147 147l-136 -136z" /> | ||||||
|  | <glyph unicode="" d="M900 303v198h-200v-200h195l-295 -300l-300 300h200v200h-200v-198l-300 300l300 296v-198h200v200h-200l300 300l295 -300h-195v-200h200v198l300 -296z" /> | ||||||
|  | <glyph unicode="" d="M900 0l-500 488v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v1000q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-437l500 487v-1100z" /> | ||||||
|  | <glyph unicode="" d="M1200 0l-500 488v-488l-500 488v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v1000q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-437l500 487v-487l500 487v-1100z" /> | ||||||
|  | <glyph unicode="" d="M1200 0l-500 488v-488l-564 550l564 550v-487l500 487v-1100z" /> | ||||||
|  | <glyph unicode="" d="M1100 550l-900 550v-1100z" /> | ||||||
|  | <glyph unicode="" d="M500 150v800q0 21 -14.5 35.5t-35.5 14.5h-200q-21 0 -35.5 -14.5t-14.5 -35.5v-800q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5zM900 150v800q0 21 -14.5 35.5t-35.5 14.5h-200q-21 0 -35.5 -14.5t-14.5 -35.5v-800q0 -21 14.5 -35.5t35.5 -14.5h200 q21 0 35.5 14.5t14.5 35.5z" /> | ||||||
|  | <glyph unicode="" d="M1100 150v800q0 21 -14.5 35.5t-35.5 14.5h-800q-21 0 -35.5 -14.5t-14.5 -35.5v-800q0 -20 14.5 -35t35.5 -15h800q21 0 35.5 15t14.5 35z" /> | ||||||
|  | <glyph unicode="" d="M500 0v488l-500 -488v1100l500 -487v487l564 -550z" /> | ||||||
|  | <glyph unicode="" d="M1050 1100h100q21 0 35.5 -14.5t14.5 -35.5v-1000q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v438l-500 -488v488l-500 -488v1100l500 -487v487l500 -487v437q0 21 14.5 35.5t35.5 14.5z" /> | ||||||
|  | <glyph unicode="" d="M850 1100h100q21 0 35.5 -14.5t14.5 -35.5v-1000q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v438l-500 -488v1100l500 -487v437q0 21 14.5 35.5t35.5 14.5z" /> | ||||||
|  | <glyph unicode="" d="M650 1064l-550 -564h1100zM1200 350v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5z" /> | ||||||
|  | <glyph unicode="" d="M777 7l240 240l-353 353l353 353l-240 240l-592 -594z" /> | ||||||
|  | <glyph unicode="" d="M513 -46l-241 240l353 353l-353 353l241 240l572 -571l21 -22l-1 -1v-1z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM500 900v-200h-200v-200h200v-200h200v200h200v200h-200v200h-200z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM300 700v-200h600v200h-600z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM247 741l141 -141l-142 -141l213 -213l141 142l141 -142l213 213l-142 141l142 141l-213 212l-141 -141 l-141 142z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM546 623l-102 102l-174 -174l276 -277l411 411l-175 174z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM500 500h200q5 3 14 8t31.5 25.5t39.5 45.5t31 69t14 94q0 51 -17.5 89t-42 58t-58.5 32t-58.5 15t-51.5 3 q-105 0 -172 -56t-67 -183h144q4 0 11.5 -1t11 -1t6.5 3t3 9t1 11t3.5 8.5t3.5 6t5.5 4t6.5 2.5t9 1.5t9 0.5h11.5h12.5q19 0 30 -10t11 -26q0 -22 -4 -28t-27 -22q-5 -1 -12.5 -3t-27 -13.5t-34 -27t-26.5 -46t-11 -68.5zM500 400v-100h200v100h-200z" /> | ||||||
|  | <glyph unicode="" d="M600 1197q162 0 299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5t80 299.5t217.5 217.5t299.5 80zM500 900v-100h200v100h-200zM400 700v-100h100v-200h-100v-100h400v100h-100v300h-300z" /> | ||||||
|  | <glyph unicode="" d="M1200 700v-200h-203q-25 -102 -116.5 -186t-180.5 -117v-197h-200v197q-140 27 -208 102.5t-98 200.5h-194v200h194q15 60 36 104.5t55.5 86t88 69t126.5 40.5v200h200v-200q54 -20 113 -60t112.5 -105.5t71.5 -134.5h203zM700 500v-206q149 48 201 206h-201v200h200 q-25 74 -76 127.5t-124 76.5v-204h-200v203q-75 -24 -130 -77.5t-79 -125.5h209v-200h-210q24 -73 79.5 -127.5t130.5 -78.5v206h200z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM844 735 l-135 -135l135 -135l-109 -109l-135 135l-135 -135l-109 109l135 135l-135 135l109 109l135 -135l135 135z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM896 654 l-346 -345l-228 228l141 141l87 -87l204 205z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM248 385l568 567q-100 62 -216 62q-171 0 -292.5 -121.5t-121.5 -292.5q0 -115 62 -215zM955 809l-564 -564q97 -59 209 -59q171 0 292.5 121.5 t121.5 292.5q0 112 -59 209z" /> | ||||||
|  | <glyph unicode="" d="M1200 400h-600v-301l-600 448l600 453v-300h600v-300z" /> | ||||||
|  | <glyph unicode="" d="M600 400h-600v300h600v300l600 -453l-600 -448v301z" /> | ||||||
|  | <glyph unicode="" d="M1098 600h-298v-600h-300v600h-296l450 600z" /> | ||||||
|  | <glyph unicode="" d="M998 600l-449 -600l-445 600h296v600h300v-600h298z" /> | ||||||
|  | <glyph unicode="" d="M600 199v301q-95 -2 -183 -20t-170 -52t-147 -92.5t-100 -135.5q6 132 41 238.5t103.5 193t184 138t271.5 59.5v271l600 -453z" /> | ||||||
|  | <glyph unicode="" d="M1200 1200h-400l129 -129l-294 -294l142 -142l294 294l129 -129v400zM565 423l-294 -294l129 -129h-400v400l129 -129l294 294z" /> | ||||||
|  | <glyph unicode="" d="M871 730l129 -130h-400v400l129 -129l295 295l142 -141zM200 600h400v-400l-129 130l-295 -295l-142 141l295 295z" /> | ||||||
|  | <glyph unicode="" d="M600 1177q118 0 224.5 -45.5t184 -123t123 -184t45.5 -224.5t-45.5 -224.5t-123 -184t-184 -123t-224.5 -45.5t-224.5 45.5t-184 123t-123 184t-45.5 224.5t45.5 224.5t123 184t184 123t224.5 45.5zM686 549l58 302q4 20 -8 34.5t-33 14.5h-207q-20 0 -32 -14.5t-8 -34.5 l58 -302q4 -20 21.5 -34.5t37.5 -14.5h54q20 0 37.5 14.5t21.5 34.5zM700 400h-200v-100h200v100z" /> | ||||||
|  | <glyph unicode="" d="M1200 900h-111v6t-1 15t-3 18l-34 172q-11 39 -41.5 63t-69.5 24q-32 0 -61 -17l-239 -144q-22 -13 -40 -35q-19 24 -40 36l-238 144q-33 18 -62 18q-39 0 -69.5 -23t-40.5 -61l-35 -177q-2 -8 -3 -18t-1 -15v-6h-111v-100h100v-200h400v300h200v-300h400v200h100v100z M731 900l202 197q5 -12 12 -32.5t23 -64t25 -72t7 -28.5h-269zM481 900h-281q-3 0 14 48t35 96l18 47zM100 0h400v400h-400v-400zM700 400h400v-400h-400v400z" /> | ||||||
|  | <glyph unicode="" d="M0 121l216 193q-9 53 -13 83t-5.5 94t9 113t38.5 114t74 124q47 60 99.5 102.5t103 68t127.5 48t145.5 37.5t184.5 43.5t220 58.5q0 -189 -22 -343t-59 -258t-89 -181.5t-108.5 -120t-122 -68t-125.5 -30t-121.5 -1.5t-107.5 12.5t-87.5 17t-56.5 7.5l-99 -55l-201 -202 v143zM692 611q70 38 118.5 69.5t102 79t99 111.5t86.5 148q22 50 24 60t-6 19q-7 5 -17 5t-26.5 -14.5t-33.5 -39.5q-35 -51 -113.5 -108.5t-139.5 -89.5l-61 -32q-369 -197 -458 -401q-48 -111 -28.5 -117.5t86.5 76.5q55 66 367 234z" /> | ||||||
|  | <glyph unicode="" d="M1261 600l-26 -40q-6 -10 -20 -30t-49 -63.5t-74.5 -85.5t-97 -90t-116.5 -83.5t-132.5 -59t-145.5 -23.5t-145.5 23.5t-132.5 59t-116.5 83.5t-97 90t-74.5 85.5t-49 63.5t-20 30l-26 40l26 40q6 10 20 30t49 63.5t74.5 85.5t97 90t116.5 83.5t132.5 59t145.5 23.5 t145.5 -23.5t132.5 -59t116.5 -83.5t97 -90t74.5 -85.5t49 -63.5t20 -30zM600 240q64 0 123.5 20t100.5 45.5t85.5 71.5t66.5 75.5t58 81.5t47 66q-1 1 -28.5 37.5t-42 55t-43.5 53t-57.5 63.5t-58.5 54q49 -74 49 -163q0 -124 -88 -212t-212 -88t-212 88t-88 212 q0 85 46 158q-102 -87 -226 -258q7 -10 40.5 -58t56 -78.5t68 -77.5t87.5 -75t103 -49.5t125 -21.5zM484 762l-107 -106q49 -124 154 -191l105 105q-37 24 -75 72t-57 84z" /> | ||||||
|  | <glyph unicode="" d="M906 1200l-314 -1200h-148l37 143q-82 21 -165 71.5t-140 102t-109.5 112t-72 88.5t-29.5 43l-26 40l26 40q6 10 20 30t49 63.5t74.5 85.5t97 90t116.5 83.5t132.5 59t145.5 23.5q61 0 121 -17l37 142h148zM1261 600l-26 -40q-7 -12 -25.5 -38t-63.5 -79.5t-95.5 -102.5 t-124 -100t-146.5 -79l38 145q22 15 44.5 34t46 44t40.5 44t41 50.5t33.5 43.5t33 44t24.5 34q-97 127 -140 175l39 146q67 -54 131.5 -125.5t87.5 -103.5t36 -52zM513 264l37 141q-107 18 -178.5 101.5t-71.5 193.5q0 85 46 158q-102 -87 -226 -258q210 -282 393 -336z M484 762l-107 -106q49 -124 154 -191l47 47l23 87q-30 28 -59 69t-44 68z" /> | ||||||
|  | <glyph unicode="" d="M-47 0h1294q37 0 50.5 35.5t-7.5 67.5l-642 1056q-20 33 -48 36t-48 -29l-642 -1066q-21 -32 -7.5 -66t50.5 -34zM700 200v100h-200v-100h-345l445 723l445 -723h-345zM700 700h-200v-100l100 -300l100 300v100z" /> | ||||||
|  | <glyph unicode="" d="M800 711l363 -325q15 -14 26 -38.5t11 -44.5v-41q0 -20 -12 -26.5t-29 5.5l-359 249v-263q100 -91 100 -113v-64q0 -21 -13 -29t-32 1l-94 78h-222l-94 -78q-19 -9 -32 -1t-13 29v64q0 22 100 113v263l-359 -249q-17 -12 -29 -5.5t-12 26.5v41q0 20 11 44.5t26 38.5 l363 325v339q0 62 44 106t106 44t106 -44t44 -106v-339z" /> | ||||||
|  | <glyph unicode="" d="M941 800l-600 -600h-341v200h259l600 600h241v198l300 -295l-300 -300v197h-159zM381 678l141 142l-181 180h-341v-200h259zM1100 598l300 -295l-300 -300v197h-241l-181 181l141 142l122 -123h159v198z" /> | ||||||
|  | <glyph unicode="" d="M100 1100h1000q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-596l-304 -300v300h-100q-41 0 -70.5 29.5t-29.5 70.5v600q0 41 29.5 70.5t70.5 29.5z" /> | ||||||
|  | <glyph unicode="" d="M400 900h-300v300h300v-300zM1100 900h-300v300h300v-300zM1100 800v-200q0 -42 -3 -83t-15 -104t-31.5 -116t-58 -109.5t-89 -96.5t-129 -65.5t-174.5 -25.5t-174.5 25.5t-129 65.5t-89 96.5t-58 109.5t-31.5 116t-15 104t-3 83v200h300v-250q0 -113 6 -145 q17 -92 102 -117q39 -11 92 -11q37 0 66.5 5.5t50 15.5t36 24t24 31.5t14 37.5t7 42t2.5 45t0 47v25v250h300z" /> | ||||||
|  | <glyph unicode="" d="M902 184l226 227l-578 579l-580 -579l227 -227l352 353z" /> | ||||||
|  | <glyph unicode="" d="M650 218l578 579l-226 227l-353 -353l-352 353l-227 -227z" /> | ||||||
|  | <glyph unicode="" d="M1198 400v600h-796l215 -200h381v-400h-198l299 -283l299 283h-200zM-198 700l299 283l300 -283h-203v-400h385l215 -200h-800v600h-196z" /> | ||||||
|  | <glyph unicode="" d="M1050 1200h94q20 0 35 -14.5t15 -35.5t-15 -35.5t-35 -14.5h-54l-201 -961q-2 -4 -6 -10.5t-19 -17.5t-33 -11h-31v-50q0 -20 -14.5 -35t-35.5 -15t-35.5 15t-14.5 35v50h-300v-50q0 -20 -14.5 -35t-35.5 -15t-35.5 15t-14.5 35v50h-50q-21 0 -35.5 15t-14.5 35 q0 21 14.5 35.5t35.5 14.5h535l48 200h-633q-32 0 -54.5 21t-27.5 43l-100 475q-5 24 10 42q14 19 39 19h896l38 162q5 17 18.5 27.5t30.5 10.5z" /> | ||||||
|  | <glyph unicode="" d="M1200 1000v-100h-1200v100h200q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5h500zM0 800h1200v-800h-1200v800z" /> | ||||||
|  | <glyph unicode="" d="M201 800l-200 -400v600h200q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5h500v-200h-1000zM1501 700l-300 -700h-1200l300 700h1200z" /> | ||||||
|  | <glyph unicode="" d="M302 300h198v600h-198l298 300l298 -300h-198v-600h198l-298 -300z" /> | ||||||
|  | <glyph unicode="" d="M900 303v197h-600v-197l-300 297l300 298v-198h600v198l300 -298z" /> | ||||||
|  | <glyph unicode="" d="M31 400l172 739q5 22 23 41.5t38 19.5h672q19 0 37.5 -22.5t23.5 -45.5l172 -732h-1138zM100 300h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM900 200h-100v-100h100v100z M1100 200h-100v-100h100v100z" /> | ||||||
|  | <glyph unicode="" d="M1100 200v850q0 21 14.5 35.5t35.5 14.5q20 0 35 -14.5t15 -35.5v-850q0 -20 -15 -35t-35 -15q-21 0 -35.5 15t-14.5 35zM325 800l675 250v-850l-675 200h-38l47 -276q2 -12 -3 -17.5t-11 -6t-21 -0.5h-8h-83q-20 0 -34.5 14t-18.5 35q-56 337 -56 351v250v5 q0 13 0.5 18.5t2.5 13t8 10.5t15 3h200zM-101 600v50q0 24 25 49t50 38l25 13v-250l-11 5.5t-24 14t-30 21.5t-24 27.5t-11 31.5z" /> | ||||||
|  | <glyph unicode="" d="M445 1180l-45 -233l-224 78l78 -225l-233 -44l179 -156l-179 -155l233 -45l-78 -224l224 78l45 -233l155 179l155 -179l45 233l224 -78l-78 224l234 45l-180 155l180 156l-234 44l78 225l-224 -78l-45 233l-155 -180z" /> | ||||||
|  | <glyph unicode="" d="M700 1200h-50q-27 0 -51 -20t-38 -48l-96 -198l-145 -196q-20 -26 -20 -63v-400q0 -75 100 -75h61q123 -100 139 -100h250q46 0 83 57l238 344q29 31 29 74v100q0 44 -30.5 84.5t-69.5 40.5h-328q28 118 28 125v150q0 44 -30.5 84.5t-69.5 40.5zM700 925l-50 -225h450 v-125l-250 -375h-214l-136 100h-100v375l150 212l100 213h50v-175zM0 800v-600h200v600h-200z" /> | ||||||
|  | <glyph unicode="" d="M700 0h-50q-27 0 -51 20t-38 48l-96 198l-145 196q-20 26 -20 63v400q0 75 100 75h61q123 100 139 100h250q46 0 83 -57l238 -344q29 -31 29 -74v-100q0 -44 -30.5 -84.5t-69.5 -40.5h-328q28 -118 28 -125v-150q0 -44 -30.5 -84.5t-69.5 -40.5zM200 400h-200v600h200 v-600zM700 275l-50 225h450v125l-250 375h-214l-136 -100h-100v-375l150 -212l100 -213h50v175z" /> | ||||||
|  | <glyph unicode="" d="M364 873l362 230q14 6 25 6q17 0 29 -12l109 -112q14 -14 14 -34q0 -18 -11 -32l-85 -121h302q85 0 138.5 -38t53.5 -110t-54.5 -111t-138.5 -39h-107l-130 -339q-7 -22 -20.5 -41.5t-28.5 -19.5h-341q-7 0 -90 81t-83 94v525q0 17 14 35.5t28 28.5zM408 792v-503 l100 -89h293l131 339q6 21 19.5 41t28.5 20h203q16 0 25 15t9 36q0 20 -9 34.5t-25 14.5h-457h-6.5h-7.5t-6.5 0.5t-6 1t-5 1.5t-5.5 2.5t-4 4t-4 5.5q-5 12 -5 20q0 14 10 27l147 183l-86 83zM208 200h-200v600h200v-600z" /> | ||||||
|  | <glyph unicode="" d="M475 1104l365 -230q7 -4 16.5 -10.5t26 -26t16.5 -36.5v-526q0 -13 -85.5 -93.5t-93.5 -80.5h-342q-15 0 -28.5 20t-19.5 41l-131 339h-106q-84 0 -139 39t-55 111t54 110t139 37h302l-85 121q-11 16 -11 32q0 21 14 34l109 113q13 12 29 12q11 0 25 -6zM370 946 l145 -184q10 -11 10 -26q0 -11 -5 -20q-1 -3 -3.5 -5.5l-4 -4t-5 -2.5t-5.5 -1.5t-6.5 -1t-6.5 -0.5h-7.5h-6.5h-476v-100h222q15 0 28.5 -20.5t19.5 -40.5l131 -339h293l106 89v502l-342 237zM1199 201h-200v600h200v-600z" /> | ||||||
|  | <glyph unicode="" d="M1100 473v342q0 15 -20 28.5t-41 19.5l-339 131v106q0 84 -39 139t-111 55t-110 -53.5t-38 -138.5v-302l-121 84q-15 12 -33.5 11.5t-32.5 -13.5l-112 -110q-22 -22 -6 -53l230 -363q4 -6 10.5 -15.5t26 -25t36.5 -15.5h525q13 0 94 83t81 90zM911 400h-503l-236 339 l83 86l183 -146q22 -18 47 -5q3 1 5.5 3.5l4 4t2.5 5t1.5 5.5t1 6.5t0.5 6v7.5v7v456q0 22 25 31t50 -0.5t25 -30.5v-202q0 -16 20 -29.5t41 -19.5l339 -130v-294zM1000 200v-200h-600v200h600z" /> | ||||||
|  | <glyph unicode="" d="M305 1104v200h600v-200h-600zM605 310l339 131q20 6 40.5 19.5t20.5 28.5v342q0 7 -81 90t-94 83h-525q-17 0 -35.5 -14t-28.5 -28l-10 -15l-230 -362q-15 -31 7 -53l112 -110q13 -13 32 -13.5t34 10.5l121 85l-1 -302q0 -84 38.5 -138t110.5 -54t111 55t39 139v106z M905 804v-294l-340 -130q-20 -6 -40 -20t-20 -29v-202q0 -22 -25 -31t-50 0t-25 31v456v14.5t-1.5 11.5t-5 12t-9.5 7q-24 13 -46 -5l-184 -146l-83 86l237 339h503z" /> | ||||||
|  | <glyph unicode="" d="M603 1195q162 0 299.5 -80t217.5 -218t80 -300t-80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5zM598 701h-298v-201h300l-2 -194l402 294l-402 298v-197z" /> | ||||||
|  | <glyph unicode="" d="M597 1195q122 0 232.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-218 -217.5t-300 -80t-299.5 80t-217.5 217.5t-80 299.5q0 122 47.5 232.5t127.5 190.5t190.5 127.5t231.5 47.5zM200 600l400 -294v194h302v201h-300v197z" /> | ||||||
|  | <glyph unicode="" d="M603 1195q121 0 231.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5zM300 600h200v-300h200v300h200l-300 400z" /> | ||||||
|  | <glyph unicode="" d="M603 1195q121 0 231.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5zM500 900v-300h-200l300 -400l300 400h-200v300h-200z" /> | ||||||
|  | <glyph unicode="" d="M603 1195q121 0 231.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5zM627 1101q-15 -12 -36.5 -21t-34.5 -12t-44 -8t-39 -6 q-15 -3 -45.5 0.5t-45.5 -2.5q-21 -7 -52 -26.5t-34 -34.5q-3 -11 6.5 -22.5t8.5 -18.5q-3 -34 -27.5 -90.5t-29.5 -79.5q-8 -33 5.5 -92.5t7.5 -87.5q0 -9 17 -44t16 -60q12 0 23 -5.5t23 -15t20 -13.5q24 -12 108 -42q22 -8 53 -31.5t59.5 -38.5t57.5 -11q8 -18 -15 -55 t-20 -57q42 -71 87 -80q0 -6 -3 -15.5t-3.5 -14.5t4.5 -17q102 -2 221 112q30 29 47 47t34.5 49t20.5 62q-14 9 -37 9.5t-36 7.5q-14 7 -49 15t-52 19q-9 0 -39.5 -0.5t-46.5 -1.5t-39 -6.5t-39 -16.5q-50 -35 -66 -12q-4 2 -3.5 25.5t0.5 25.5q-6 13 -26.5 17t-24.5 7 q2 22 -2 41t-16.5 28t-38.5 -20q-23 -25 -42 4q-19 28 -8 58q6 16 22 22q6 -1 26 -1.5t33.5 -4t19.5 -13.5q12 -19 32 -37.5t34 -27.5l14 -8q0 3 9.5 39.5t5.5 57.5q-4 23 14.5 44.5t22.5 31.5q5 14 10 35t8.5 31t15.5 22.5t34 21.5q-6 18 10 37q8 0 23.5 -1.5t24.5 -1.5 t20.5 4.5t20.5 15.5q-10 23 -30.5 42.5t-38 30t-49 26.5t-43.5 23q11 41 1 44q31 -13 58.5 -14.5t39.5 3.5l11 4q6 36 -17 53.5t-64 28.5t-56 23q-19 -3 -37 0zM613 994q0 -18 8 -42.5t16.5 -44t9.5 -23.5q-9 2 -31 5t-36 5t-32 8t-30 14q3 12 16 30t16 25q10 -10 18.5 -10 t14 6t14.5 14.5t16 12.5z" /> | ||||||
|  | <glyph unicode="" horiz-adv-x="1220" d="M100 1196h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 1096h-200v-100h200v100zM100 796h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000 q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 696h-500v-100h500v100zM100 396h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5v100q0 41 29.5 70.5t70.5 29.5zM1100 296h-300v-100h300v100z " /> | ||||||
|  | <glyph unicode="" d="M1100 1200v-100h-1000v100h1000zM150 1000h900l-350 -500v-300l-200 -200v500z" /> | ||||||
|  | <glyph unicode="" d="M329 729l142 142l-200 200l129 129h-400v-400l129 129zM1200 1200v-400l-129 129l-200 -200l-142 142l200 200l-129 129h400zM271 129l129 -129h-400v400l129 -129l200 200l142 -142zM1071 271l129 129v-400h-400l129 129l-200 200l142 142z" /> | ||||||
|  | <glyph unicode="" d="M596 1192q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM596 1010q-171 0 -292.5 -121.5t-121.5 -292.5q0 -172 121.5 -293t292.5 -121t292.5 121t121.5 293q0 171 -121.5 292.5t-292.5 121.5zM455 905 q22 0 38 -16t16 -39t-16 -39t-38 -16q-23 0 -39 16.5t-16 38.5t16 38.5t39 16.5zM708 821l1 1q-9 14 -9 28q0 22 16 38.5t39 16.5q22 0 38 -16t16 -39t-16 -39t-38 -16q-14 0 -29 10l-55 -145q17 -22 17 -51q0 -36 -25.5 -61.5t-61.5 -25.5t-61.5 25.5t-25.5 61.5 q0 32 20.5 56.5t51.5 29.5zM855 709q23 0 38.5 -15.5t15.5 -38.5t-16 -39t-38 -16q-23 0 -39 16t-16 39q0 22 16 38t39 16zM345 709q23 0 39 -16t16 -38q0 -23 -16 -39t-39 -16q-22 0 -38 16t-16 39t15.5 38.5t38.5 15.5z" /> | ||||||
|  | <glyph unicode="" d="M649 54l-16 22q-90 125 -293 323q-71 70 -104.5 105.5t-77 89.5t-61 99t-17.5 91q0 131 98.5 229.5t230.5 98.5q143 0 241 -129q103 129 246 129q129 0 226 -98.5t97 -229.5q0 -46 -17.5 -91t-61 -99t-77 -89.5t-104.5 -105.5q-203 -198 -293 -323zM844 524l12 12 q64 62 97.5 97t64.5 79t31 72q0 71 -48 119t-105 48q-74 0 -132 -82l-118 -171l-114 174q-51 79 -123 79q-60 0 -109.5 -49t-49.5 -118q0 -27 30.5 -70t61.5 -75.5t95 -94.5l22 -22q93 -90 190 -201q82 92 195 203z" /> | ||||||
|  | <glyph unicode="" d="M476 406l19 -17l105 105l-212 212l389 389l247 -247l-95 -96l18 -18q46 -46 77 -99l29 29q35 35 62.5 88t27.5 96q0 93 -66 159l-141 141q-66 66 -159 66q-95 0 -159 -66l-283 -283q-66 -64 -66 -159q0 -93 66 -159zM123 193l141 -141q66 -66 159 -66q95 0 159 66 l283 283q66 66 66 159t-66 159l-141 141q-12 12 -19 17l-105 -105l212 -212l-389 -389l-247 248l95 95l-18 18q-46 45 -75 101l-55 -55q-66 -66 -66 -159q0 -94 66 -160z" /> | ||||||
|  | <glyph unicode="" d="M200 100v953q0 21 30 46t81 48t129 38t163 15t162 -15t127 -38t79 -48t29 -46v-953q0 -41 -29.5 -70.5t-70.5 -29.5h-600q-41 0 -70.5 29.5t-29.5 70.5zM900 1000h-600v-700h600v700zM600 46q43 0 73.5 30.5t30.5 73.5t-30.5 73.5t-73.5 30.5t-73.5 -30.5t-30.5 -73.5 t30.5 -73.5t73.5 -30.5z" /> | ||||||
|  | <glyph unicode="" d="M700 1029v-307l64 -14q34 -7 64 -16.5t70 -31.5t67.5 -52t47.5 -80.5t20 -112.5q0 -139 -89 -224t-244 -96v-77h-100v78q-152 17 -237 104q-40 40 -52.5 93.5t-15.5 139.5h139q5 -77 48.5 -126.5t117.5 -64.5v335l-27 7q-46 14 -79 26.5t-72 36t-62.5 52t-40 72.5 t-16.5 99q0 92 44 159.5t109 101t144 40.5v78h100v-79q38 -4 72.5 -13.5t75.5 -31.5t71 -53.5t51.5 -84t24.5 -118.5h-159q-8 72 -35 109.5t-101 50.5zM600 755v274q-61 -8 -97.5 -37.5t-36.5 -102.5q0 -29 8 -51t16.5 -34t29.5 -22.5t31 -13.5t38 -10q7 -2 11 -3zM700 548 v-311q170 18 170 151q0 64 -44 99.5t-126 60.5z" /> | ||||||
|  | <glyph unicode="" d="M866 300l50 -147q-41 -25 -80.5 -36.5t-59 -13t-61.5 -1.5q-23 0 -128 33t-155 29q-39 -4 -82 -17t-66 -25l-24 -11l-55 145l16.5 11t15.5 10t13.5 9.5t14.5 12t14.5 14t17.5 18.5q48 55 54 126.5t-30 142.5h-221v100h166q-24 49 -44 104q-10 26 -14.5 55.5t-3 72.5 t25 90t68.5 87q97 88 263 88q129 0 230 -89t101 -208h-153q0 52 -34 89.5t-74 51.5t-76 14q-37 0 -79 -14.5t-62 -35.5q-41 -44 -41 -101q0 -11 2.5 -24.5t5.5 -24t9.5 -26.5t10.5 -25t14 -27.5t14 -25.5t15.5 -27t13.5 -24h242v-100h-197q8 -50 -2.5 -115t-31.5 -94 q-41 -59 -99 -113q35 11 84 18t70 7q32 1 102 -16t104 -17q76 0 136 30z" /> | ||||||
|  | <glyph unicode="" d="M300 0l298 300h-198v900h-200v-900h-198zM900 1200l298 -300h-198v-900h-200v900h-198z" /> | ||||||
|  | <glyph unicode="" d="M400 300h198l-298 -300l-298 300h198v900h200v-900zM1000 1200v-500h-100v100h-100v-100h-100v500h300zM901 1100h-100v-200h100v200zM700 500h300v-200h-99v-100h-100v100h99v100h-200v100zM800 100h200v-100h-300v200h100v-100z" /> | ||||||
|  | <glyph unicode="" d="M400 300h198l-298 -300l-298 300h198v900h200v-900zM1000 1200v-200h-99v-100h-100v100h99v100h-200v100h300zM800 800h200v-100h-300v200h100v-100zM700 500h300v-500h-100v100h-100v-100h-100v500zM801 200h100v200h-100v-200z" /> | ||||||
|  | <glyph unicode="" d="M300 0l298 300h-198v900h-200v-900h-198zM900 1100h-100v100h200v-500h-100v400zM1100 500v-500h-100v100h-200v400h300zM1001 400h-100v-200h100v200z" /> | ||||||
|  | <glyph unicode="" d="M300 0l298 300h-198v900h-200v-900h-198zM1100 1200v-500h-100v100h-200v400h300zM1001 1100h-100v-200h100v200zM900 400h-100v100h200v-500h-100v400z" /> | ||||||
|  | <glyph unicode="" d="M300 0l298 300h-198v900h-200v-900h-198zM900 1000h-200v200h200v-200zM1000 700h-300v200h300v-200zM1100 400h-400v200h400v-200zM1200 100h-500v200h500v-200z" /> | ||||||
|  | <glyph unicode="" d="M300 0l298 300h-198v900h-200v-900h-198zM1200 1000h-500v200h500v-200zM1100 700h-400v200h400v-200zM1000 400h-300v200h300v-200zM900 100h-200v200h200v-200z" /> | ||||||
|  | <glyph unicode="" d="M400 1100h300q162 0 281 -118.5t119 -281.5v-300q0 -165 -118.5 -282.5t-281.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5v300q0 165 117.5 282.5t282.5 117.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5 t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5z" /> | ||||||
|  | <glyph unicode="" d="M700 0h-300q-163 0 -281.5 117.5t-118.5 282.5v300q0 163 119 281.5t281 118.5h300q165 0 282.5 -117.5t117.5 -282.5v-300q0 -165 -117.5 -282.5t-282.5 -117.5zM800 900h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5 t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5zM400 800v-500l333 250z" /> | ||||||
|  | <glyph unicode="" d="M0 400v300q0 163 117.5 281.5t282.5 118.5h300q163 0 281.5 -119t118.5 -281v-300q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5zM900 300v500q0 41 -29.5 70.5t-70.5 29.5h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5 t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5zM800 700h-500l250 -333z" /> | ||||||
|  | <glyph unicode="" d="M1100 700v-300q0 -162 -118.5 -281t-281.5 -119h-300q-165 0 -282.5 118.5t-117.5 281.5v300q0 165 117.5 282.5t282.5 117.5h300q165 0 282.5 -117.5t117.5 -282.5zM900 300v500q0 41 -29.5 70.5t-70.5 29.5h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5 t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5zM550 733l-250 -333h500z" /> | ||||||
|  | <glyph unicode="" d="M500 1100h400q165 0 282.5 -117.5t117.5 -282.5v-300q0 -165 -117.5 -282.5t-282.5 -117.5h-400v200h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5h-500v200zM700 550l-400 -350v200h-300v300h300v200z" /> | ||||||
|  | <glyph unicode="" d="M403 2l9 -1q13 0 26 16l538 630q15 19 6 36q-8 18 -32 16h-300q1 4 78 219.5t79 227.5q2 17 -6 27l-8 8h-9q-16 0 -25 -15q-4 -5 -98.5 -111.5t-228 -257t-209.5 -238.5q-17 -19 -7 -40q10 -19 32 -19h302q-155 -438 -160 -458q-5 -21 4 -32z" /> | ||||||
|  | <glyph unicode="" d="M800 200h-500q-41 0 -70.5 29.5t-29.5 70.5v500q0 41 29.5 70.5t70.5 29.5h500v185q-14 4 -114 7.5t-193 5.5l-93 2q-165 0 -282.5 -117.5t-117.5 -282.5v-300q0 -165 117.5 -282.5t282.5 -117.5h300q47 0 100 15v185zM900 200v200h-300v300h300v200l400 -350z" /> | ||||||
|  | <glyph unicode="" d="M1200 700l-149 149l-342 -353l-213 213l353 342l-149 149h500v-500zM1022 571l-122 -123v-148q0 -41 -29.5 -70.5t-70.5 -29.5h-500q-41 0 -70.5 29.5t-29.5 70.5v500q0 41 29.5 70.5t70.5 29.5h156l118 122l-74 78h-100q-165 0 -282.5 -117.5t-117.5 -282.5v-300 q0 -165 117.5 -282.5t282.5 -117.5h300q163 0 281.5 117.5t118.5 282.5v98z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM600 794 q80 0 137 -57t57 -137t-57 -137t-137 -57t-137 57t-57 137t57 137t137 57z" /> | ||||||
|  | <glyph unicode="" d="M700 800v400h-300v-400h-300l445 -500l450 500h-295zM25 300h1048q11 0 19 -7.5t8 -17.5v-275h-1100v275q0 11 7 18t18 7zM1000 200h-100v-50h100v50z" /> | ||||||
|  | <glyph unicode="" d="M400 700v-300h300v300h295l-445 500l-450 -500h300zM25 300h1048q11 0 19 -7.5t8 -17.5v-275h-1100v275q0 11 7 18t18 7zM1000 200h-100v-50h100v50z" /> | ||||||
|  | <glyph unicode="" d="M405 400l596 596l-154 155l-442 -442l-150 151l-155 -155zM25 300h1048q11 0 19 -7.5t8 -17.5v-275h-1100v275q0 11 7 18t18 7zM1000 200h-100v-50h100v50z" /> | ||||||
|  | <glyph unicode="" d="M409 1103l-97 97l-212 -212l97 -98zM650 861l-149 149l-212 -212l149 -149l-238 -248h700v699zM25 300h1048q11 0 19 -7.5t8 -17.5v-275h-1100v275q0 11 7 18t18 7zM1000 200h-100v-50h100v50z" /> | ||||||
|  | <glyph unicode="" d="M539 950l-149 -149l212 -212l149 148l248 -237v700h-699zM297 709l-97 -97l212 -212l98 97zM25 300h1048q11 0 19 -7.5t8 -17.5v-275h-1100v275q0 11 7 18t18 7zM1000 200h-100v-50h100v50z" /> | ||||||
|  | <glyph unicode="" d="M1200 1199v-1079l-475 272l-310 -393v416h-392zM1166 1148l-672 -712v-226z" /> | ||||||
|  | <glyph unicode="" d="M1100 1000v-850q0 -21 -15 -35.5t-35 -14.5h-150v400h-700v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100zM700 1200h-100v-200h100v200z" /> | ||||||
|  | <glyph unicode="" d="M578 500h-378v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-218l-276 -275l-120 120zM700 1200h-100v-200h100v200zM1300 538l-475 -476l-244 244l123 123l120 -120l353 352z" /> | ||||||
|  | <glyph unicode="" d="M529 500h-329v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-269l-103 -103l-170 170zM700 1200h-100v-200h100v200zM1167 6l-170 170l-170 -170l-127 127l170 170l-170 170l127 127l170 -170l170 170l127 -128 l-170 -169l170 -170z" /> | ||||||
|  | <glyph unicode="" d="M700 500h-500v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-300h-400v-200zM700 1000h-100v200h100v-200zM1000 600h-200v-300h-200l300 -300l300 300h-200v300z" /> | ||||||
|  | <glyph unicode="" d="M602 500h-402v-400h-150q-21 0 -35.5 14.5t-14.5 35.5v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-402l-200 200zM700 1000h-100v200h100v-200zM1000 300h200l-300 300l-300 -300h200v-300h200v300z" /> | ||||||
|  | <glyph unicode="" d="M1200 900v150q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-150h1200zM0 800v-550q0 -21 14.5 -35.5t35.5 -14.5h1100q21 0 35.5 14.5t14.5 35.5v550h-1200zM100 500h400v-200h-400v200z" /> | ||||||
|  | <glyph unicode="" d="M500 1000h400v198l300 -298l-300 -298v198h-400v200zM100 800v200h100v-200h-100zM400 800h-100v200h100v-200zM700 300h-400v-198l-300 298l300 298v-198h400v-200zM800 500h100v-200h-100v200zM1000 500v-200h100v200h-100z" /> | ||||||
|  | <glyph unicode="" d="M1200 50v1106q0 31 -18 40.5t-44 -7.5l-276 -117q-25 -16 -43.5 -50.5t-18.5 -65.5v-359q0 -29 10.5 -55.5t25 -43t29 -28.5t25.5 -18l10 -5v-397q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5zM550 1200l50 -100v-400l-100 -203v-447q0 -21 -14.5 -35.5 t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v447l-100 203v400l50 100l50 -100v-300h100v300l50 100l50 -100v-300h100v300z" /> | ||||||
|  | <glyph unicode="" d="M1100 106v888q0 22 25 34.5t50 13.5l25 2v56h-400v-56q75 0 87.5 -6t12.5 -44v-394h-500v394q0 38 12.5 44t87.5 6v56h-400v-56q4 0 11 -0.5t24 -3t30 -7t24 -15t11 -24.5v-888q0 -22 -25 -34.5t-50 -13.5l-25 -2v-56h400v56q-75 0 -87.5 6t-12.5 44v394h500v-394 q0 -38 -12.5 -44t-87.5 -6v-56h400v56q-4 0 -11 0.5t-24 3t-30 7t-24 15t-11 24.5z" /> | ||||||
|  | <glyph unicode="" d="M675 1000l-100 100h-375l-100 -100h400l200 -200v-98l295 98h105v200h-425zM500 300v500q0 41 -29.5 70.5t-70.5 29.5h-300q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h300q41 0 70.5 29.5t29.5 70.5zM100 800h300v-200h-300v200zM700 565l400 133 v-163l-400 -133v163zM100 500h300v-200h-300v200zM805 300l295 98v-298h-425l-100 -100h-375l-100 100h400l200 200h105z" /> | ||||||
|  | <glyph unicode="" d="M179 1169l-162 -162q-1 -11 -0.5 -32.5t16 -90t46.5 -140t104 -177.5t175 -208q103 -103 207.5 -176t180 -103.5t137 -47t92.5 -16.5l31 1l163 162q16 17 13 40.5t-22 37.5l-192 136q-19 14 -45 12t-42 -19l-119 -118q-143 103 -267 227q-126 126 -227 268l118 118 q17 17 20 41.5t-11 44.5l-139 194q-14 19 -36.5 22t-40.5 -14z" /> | ||||||
|  | <glyph unicode="" d="M1200 712v200q-6 8 -19 20.5t-63 45t-112 57t-171 45t-235 20.5q-92 0 -175 -10.5t-141.5 -27t-108.5 -36.5t-81.5 -40t-53.5 -36.5t-31 -27.5l-9 -10v-200q0 -21 14.5 -33.5t34.5 -8.5l202 33q20 4 34.5 21t14.5 38v146q141 24 300 24t300 -24v-146q0 -21 14.5 -38 t34.5 -21l202 -33q20 -4 34.5 8.5t14.5 33.5zM800 650l365 -303q14 -14 24.5 -39.5t10.5 -45.5v-212q0 -21 -15 -35.5t-35 -14.5h-1100q-21 0 -35.5 14.5t-14.5 35.5v212q0 20 10.5 45.5t24.5 39.5l365 303v50q0 4 1 10.5t12 22.5t30 28.5t60 23t97 10.5t97 -10t60 -23.5 t30 -27.5t12 -24l1 -10v-50z" /> | ||||||
|  | <glyph unicode="" d="M175 200h950l-125 150v250l100 100v400h-100v-200h-100v200h-200v-200h-100v200h-200v-200h-100v200h-100v-400l100 -100v-250zM1200 100v-100h-1100v100h1100z" /> | ||||||
|  | <glyph unicode="" d="M600 1100h100q41 0 70.5 -29.5t29.5 -70.5v-1000h-300v1000q0 41 29.5 70.5t70.5 29.5zM1000 800h100q41 0 70.5 -29.5t29.5 -70.5v-700h-300v700q0 41 29.5 70.5t70.5 29.5zM400 0v400q0 41 -29.5 70.5t-70.5 29.5h-100q-41 0 -70.5 -29.5t-29.5 -70.5v-400h300z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM200 800v-300h200v-100h-200v-100h300v300h-200v100h200v100h-300zM800 800h-200v-500h200v100h100v300h-100 v100zM800 700v-300h-100v300h100z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM400 600h-100v200h-100v-500h100v200h100v-200h100v500h-100v-200zM800 800h-200v-500h200v100h100v300h-100 v100zM800 700v-300h-100v300h100z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM200 800v-500h300v100h-200v300h200v100h-300zM600 800v-500h300v100h-200v300h200v100h-300z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM500 700l-300 -150l300 -150v300zM600 400l300 150l-300 150v-300z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM900 800v-500h-700v500h700zM300 400h130q41 0 68 42t27 107t-28.5 108t-66.5 43h-130v-300zM800 700h-130 q-38 0 -66.5 -43t-28.5 -108t27 -107t68 -42h130v300z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM200 800v-300h200v-100h-200v-100h300v300h-200v100h200v100h-300zM800 300h100v500h-200v-100h100v-400z M601 300h100v100h-100v-100z" /> | ||||||
|  | <glyph unicode="" d="M1200 800v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212zM1000 900h-900v-700h900v700zM300 700v100h-100v-500h300v400h-200zM800 300h100v500h-200v-100h100v-400zM401 400h-100v200h100v-200z M601 300h100v100h-100v-100z" /> | ||||||
|  | <glyph unicode="" d="M200 1100h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212v500q0 124 88 212t212 88zM1000 900h-900v-700h900v700zM400 700h-200v100h300v-300h-99v-100h-100v100h99v200zM800 700h-100v100h200v-500h-100v400zM201 400h100v-100 h-100v100zM701 300h-100v100h100v-100z" /> | ||||||
|  | <glyph unicode="" d="M600 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM600 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM800 700h-300 v-200h300v-100h-300l-100 100v200l100 100h300v-100z" /> | ||||||
|  | <glyph unicode="" d="M596 1196q162 0 299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299t80 299t217 217t299 80zM596 1014q-171 0 -292.5 -121.5t-121.5 -292.5t121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5zM800 700v-100 h-100v100h-200v-100h200v-100h-200v-100h-100v400h300zM800 400h-100v100h100v-100z" /> | ||||||
|  | <glyph unicode="" d="M800 300h128q120 0 205 86t85 208q0 120 -85 206.5t-205 86.5q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5q0 -80 56.5 -137t135.5 -57h222v300h400v-300zM700 200h200l-300 -300 l-300 300h200v300h200v-300z" /> | ||||||
|  | <glyph unicode="" d="M600 714l403 -403q94 26 154.5 104t60.5 178q0 121 -85 207.5t-205 86.5q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5q0 -80 56.5 -137t135.5 -57h8zM700 -100h-200v300h-200l300 300 l300 -300h-200v-300z" /> | ||||||
|  | <glyph unicode="" d="M700 200h400l-270 300h170l-270 300h170l-300 333l-300 -333h170l-270 -300h170l-270 -300h400v-155l-75 -45h350l-75 45v155z" /> | ||||||
|  | <glyph unicode="" d="M700 45v306q46 -30 100 -30q74 0 126.5 52.5t52.5 126.5q0 24 -9 55q50 32 79.5 83t29.5 112q0 90 -61.5 155.5t-150.5 71.5q-26 89 -99.5 145.5t-167.5 56.5q-116 0 -197.5 -81.5t-81.5 -197.5q0 -4 1 -12t1 -11q-14 2 -23 2q-74 0 -126.5 -52.5t-52.5 -126.5 q0 -53 28.5 -97t75.5 -65q-4 -16 -4 -38q0 -74 52.5 -126.5t126.5 -52.5q56 0 100 30v-306l-75 -45h350z" /> | ||||||
|  | <glyph unicode="💼" d="M800 1000h300q41 0 70.5 -29.5t29.5 -70.5v-400h-500v100h-200v-100h-500v400q0 41 29.5 70.5t70.5 29.5h300v100q0 41 29.5 70.5t70.5 29.5h200q41 0 70.5 -29.5t29.5 -70.5v-100zM500 1000h200v100h-200v-100zM1200 400v-200q0 -41 -29.5 -70.5t-70.5 -29.5h-1000 q-41 0 -70.5 29.5t-29.5 70.5v200h1200z" /> | ||||||
|  | <glyph unicode="📅" d="M1100 900v150q0 21 -14.5 35.5t-35.5 14.5h-150v100h-100v-100h-500v100h-100v-100h-150q-21 0 -35.5 -14.5t-14.5 -35.5v-150h1100zM0 800v-750q0 -20 14.5 -35t35.5 -15h1000q21 0 35.5 15t14.5 35v750h-1100zM100 600h100v-100h-100v100zM300 600h100v-100h-100v100z M500 600h100v-100h-100v100zM700 600h100v-100h-100v100zM900 600h100v-100h-100v100zM100 400h100v-100h-100v100zM300 400h100v-100h-100v100zM500 400h100v-100h-100v100zM700 400h100v-100h-100v100zM900 400h100v-100h-100v100zM100 200h100v-100h-100v100zM300 200 h100v-100h-100v100zM500 200h100v-100h-100v100zM700 200h100v-100h-100v100zM900 200h100v-100h-100v100z" /> | ||||||
|  | <glyph unicode="📌" d="M902 1185l283 -282q15 -15 15 -36t-15 -35q-14 -15 -35 -15t-35 15l-36 35l-279 -267v-300l-212 210l-208 -207l-380 -303l303 380l207 208l-210 212h300l267 279l-35 36q-15 14 -15 35t15 35q14 15 35 15t35 -15z" /> | ||||||
|  | <glyph unicode="📎" d="M518 119l69 -60l517 511q67 67 95 157t11 183q-16 87 -67 154t-130 103q-69 33 -152 33q-107 0 -197 -55q-40 -24 -111 -95l-512 -512q-68 -68 -81 -163t35 -173q35 -57 94 -89t129 -32q63 0 119 28q33 16 65 40.5t52.5 45.5t59.5 64q40 44 57 61l394 394q35 35 47 84 t-3 96q-27 87 -117 104q-20 2 -29 2q-46 0 -79.5 -17t-67.5 -51l-388 -396l-7 -7l69 -67l377 373q20 22 39 38q23 23 50 23q38 0 53 -36q16 -39 -20 -75l-547 -547q-52 -52 -125 -52q-55 0 -100 33t-54 96q-5 35 2.5 66t31.5 63t42 50t56 54q24 21 44 41l348 348 q52 52 82.5 79.5t84 54t107.5 26.5q25 0 48 -4q95 -17 154 -94.5t51 -175.5q-7 -101 -98 -192l-252 -249l-253 -256z" /> | ||||||
|  | <glyph unicode="📷" d="M1200 200v600q0 41 -29.5 70.5t-70.5 29.5h-150q-4 8 -11.5 21.5t-33 48t-53 61t-69 48t-83.5 21.5h-200q-41 0 -82 -20.5t-70 -50t-52 -59t-34 -50.5l-12 -20h-150q-41 0 -70.5 -29.5t-29.5 -70.5v-600q0 -41 29.5 -70.5t70.5 -29.5h1000q41 0 70.5 29.5t29.5 70.5z M1000 700h-100v100h100v-100zM844 500q0 -100 -72 -172t-172 -72t-172 72t-72 172t72 172t172 72t172 -72t72 -172zM706 500q0 44 -31 75t-75 31t-75 -31t-31 -75t31 -75t75 -31t75 31t31 75z" /> | ||||||
|  | <glyph unicode="🔒" d="M900 800h100q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-900q-41 0 -70.5 29.5t-29.5 70.5v600q0 41 29.5 70.5t70.5 29.5h100v200q0 82 59 141t141 59h300q82 0 141 -59t59 -141v-200zM400 800h300v150q0 21 -14.5 35.5t-35.5 14.5h-200 q-21 0 -35.5 -14.5t-14.5 -35.5v-150z" /> | ||||||
|  | <glyph unicode="🔔" d="M1062 400h17q20 0 33.5 -14.5t13.5 -35.5q0 -20 -13 -40t-31 -27q-22 -9 -63 -23t-167.5 -37t-251.5 -23t-245.5 20.5t-178.5 41.5l-58 20q-18 7 -31 27.5t-13 40.5q0 21 13.5 35.5t33.5 14.5h17l118 173l63 327q15 77 76 140t144 83l-18 32q-6 19 3 32t29 13h94 q20 0 29 -10.5t3 -29.5l-18 -37q83 -19 144 -82.5t76 -140.5l63 -327zM600 104q-54 0 -103 6q12 -49 40 -79.5t63 -30.5t63 30.5t39 79.5q-48 -6 -102 -6z" /> | ||||||
|  | <glyph unicode="🔖" d="M200 0l450 444l450 -443v1150q0 20 -14.5 35t-35.5 15h-800q-21 0 -35.5 -15t-14.5 -35v-1151z" /> | ||||||
|  | <glyph unicode="🔥" d="M400 755q2 -12 8 -41.5t8 -43t6 -39.5t3.5 -39.5t-1 -33.5t-6 -31.5t-13.5 -24t-21 -20.5t-31 -12q-38 -10 -67 13t-40.5 61.5t-15 81.5t10.5 75q-52 -46 -83.5 -101t-39 -107t-7.5 -85t5 -63q9 -56 44 -119.5t105 -108.5q31 -21 64 -16t62 23.5t57 49.5t48 61.5t35 60.5 q32 66 39 184.5t-13 157.5q79 -80 122 -164t26 -184q-5 -33 -20.5 -69.5t-37.5 -80.5q-10 -19 -14.5 -29t-12 -26t-9 -23.5t-3 -19t2.5 -15.5t11 -9.5t19.5 -5t30.5 2.5t42 8q57 20 91 34t87.5 44.5t87 64t65.5 88.5t47 122q38 172 -44.5 341.5t-246.5 278.5q22 -44 43 -129 q39 -159 -32 -154q-15 2 -33 9q-79 33 -120.5 100t-44 175.5t48.5 257.5q-13 -8 -34 -23.5t-72.5 -66.5t-88.5 -105.5t-60 -138t-8 -166.5z" /> | ||||||
|  | <glyph unicode="🔧" d="M948 778l251 126q13 -175 -151 -267q-123 -70 -253 -23l-596 -596q-15 -16 -36.5 -16t-36.5 16l-111 110q-15 15 -15 36.5t15 37.5l600 599q-33 101 6 201.5t135 154.5q164 92 306 -9l-259 -138z" /> | ||||||
|  | </font> | ||||||
|  | </defs></svg>  | ||||||
| After Width: | Height: | Size: 62 KiB | 
							
								
								
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.ttf
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.ttf
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.woff
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								nodejs/public/fonts/glyphicons-halflings-regular.woff
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										552
									
								
								nodejs/public/js/ICanHaz.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										552
									
								
								nodejs/public/js/ICanHaz.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,552 @@ | |||||||
|  | /*! | ||||||
|  | ICanHaz.js version 0.10.2 -- by @HenrikJoreteg | ||||||
|  | More info at: http://icanhazjs.com | ||||||
|  | */ | ||||||
|  | (function () { | ||||||
|  | /* | ||||||
|  |   mustache.js — Logic-less templates in JavaScript | ||||||
|  |  | ||||||
|  |   See http://mustache.github.com/ for more info. | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | var Mustache = function () { | ||||||
|  |   var _toString = Object.prototype.toString; | ||||||
|  |  | ||||||
|  |   Array.isArray = Array.isArray || function (obj) { | ||||||
|  |     return _toString.call(obj) == "[object Array]"; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var _trim = String.prototype.trim, trim; | ||||||
|  |  | ||||||
|  |   if (_trim) { | ||||||
|  |     trim = function (text) { | ||||||
|  |       return text == null ? "" : _trim.call(text); | ||||||
|  |     } | ||||||
|  |   } else { | ||||||
|  |     var trimLeft, trimRight; | ||||||
|  |  | ||||||
|  |     // IE doesn't match non-breaking spaces with \s. | ||||||
|  |     if ((/\S/).test("\xA0")) { | ||||||
|  |       trimLeft = /^[\s\xA0]+/; | ||||||
|  |       trimRight = /[\s\xA0]+$/; | ||||||
|  |     } else { | ||||||
|  |       trimLeft = /^\s+/; | ||||||
|  |       trimRight = /\s+$/; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     trim = function (text) { | ||||||
|  |       return text == null ? "" : | ||||||
|  |         text.toString().replace(trimLeft, "").replace(trimRight, ""); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var escapeMap = { | ||||||
|  |     "&": "&", | ||||||
|  |     "<": "<", | ||||||
|  |     ">": ">", | ||||||
|  |     '"': '"', | ||||||
|  |     "'": ''' | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   function escapeHTML(string) { | ||||||
|  |     return String(string).replace(/&(?!\w+;)|[<>"']/g, function (s) { | ||||||
|  |       return escapeMap[s] || s; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var regexCache = {}; | ||||||
|  |   var Renderer = function () {}; | ||||||
|  |  | ||||||
|  |   Renderer.prototype = { | ||||||
|  |     otag: "<<", | ||||||
|  |     ctag: ">>", | ||||||
|  |     pragmas: {}, | ||||||
|  |     buffer: [], | ||||||
|  |     pragmas_implemented: { | ||||||
|  |       "IMPLICIT-ITERATOR": true | ||||||
|  |     }, | ||||||
|  |     context: {}, | ||||||
|  |  | ||||||
|  |     render: function (template, context, partials, in_recursion) { | ||||||
|  |       // reset buffer & set context | ||||||
|  |       if (!in_recursion) { | ||||||
|  |         this.context = context; | ||||||
|  |         this.buffer = []; // TODO: make this non-lazy | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // fail fast | ||||||
|  |       if (!this.includes("", template)) { | ||||||
|  |         if (in_recursion) { | ||||||
|  |           return template; | ||||||
|  |         } else { | ||||||
|  |           this.send(template); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // get the pragmas together | ||||||
|  |       template = this.render_pragmas(template); | ||||||
|  |  | ||||||
|  |       // render the template | ||||||
|  |       var html = this.render_section(template, context, partials); | ||||||
|  |  | ||||||
|  |       // render_section did not find any sections, we still need to render the tags | ||||||
|  |       if (html === false) { | ||||||
|  |         html = this.render_tags(template, context, partials, in_recursion); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (in_recursion) { | ||||||
|  |         return html; | ||||||
|  |       } else { | ||||||
|  |         this.sendLines(html); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Sends parsed lines | ||||||
|  |     */ | ||||||
|  |     send: function (line) { | ||||||
|  |       if (line !== "") { | ||||||
|  |         this.buffer.push(line); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     sendLines: function (text) { | ||||||
|  |       if (text) { | ||||||
|  |         var lines = text.split("\n"); | ||||||
|  |         for (var i = 0; i < lines.length; i++) { | ||||||
|  |           this.send(lines[i]); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Looks for %PRAGMAS | ||||||
|  |     */ | ||||||
|  |     render_pragmas: function (template) { | ||||||
|  |       // no pragmas | ||||||
|  |       if (!this.includes("%", template)) { | ||||||
|  |         return template; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var that = this; | ||||||
|  |       var regex = this.getCachedRegex("render_pragmas", function (otag, ctag) { | ||||||
|  |         return new RegExp(otag + "%([\\w-]+) ?([\\w]+=[\\w]+)?" + ctag, "g"); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       return template.replace(regex, function (match, pragma, options) { | ||||||
|  |         if (!that.pragmas_implemented[pragma]) { | ||||||
|  |           throw({message: | ||||||
|  |             "This implementation of mustache doesn't understand the '" + | ||||||
|  |             pragma + "' pragma"}); | ||||||
|  |         } | ||||||
|  |         that.pragmas[pragma] = {}; | ||||||
|  |         if (options) { | ||||||
|  |           var opts = options.split("="); | ||||||
|  |           that.pragmas[pragma][opts[0]] = opts[1]; | ||||||
|  |         } | ||||||
|  |         return ""; | ||||||
|  |         // ignore unknown pragmas silently | ||||||
|  |       }); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Tries to find a partial in the curent scope and render it | ||||||
|  |     */ | ||||||
|  |     render_partial: function (name, context, partials) { | ||||||
|  |       name = trim(name); | ||||||
|  |       if (!partials || partials[name] === undefined) { | ||||||
|  |         throw({message: "unknown_partial '" + name + "'"}); | ||||||
|  |       } | ||||||
|  |       if (!context || typeof context[name] != "object") { | ||||||
|  |         return this.render(partials[name], context, partials, true); | ||||||
|  |       } | ||||||
|  |       return this.render(partials[name], context[name], partials, true); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Renders inverted (^) and normal (#) sections | ||||||
|  |     */ | ||||||
|  |     render_section: function (template, context, partials) { | ||||||
|  |       if (!this.includes("#", template) && !this.includes("^", template)) { | ||||||
|  |         // did not render anything, there were no sections | ||||||
|  |         return false; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var that = this; | ||||||
|  |  | ||||||
|  |       var regex = this.getCachedRegex("render_section", function (otag, ctag) { | ||||||
|  |         // This regex matches _the first_ section ({{#foo}}{{/foo}}), and captures the remainder | ||||||
|  |         return new RegExp( | ||||||
|  |           "^([\\s\\S]*?)" +         // all the crap at the beginning that is not {{*}} ($1) | ||||||
|  |  | ||||||
|  |           otag +                    // {{ | ||||||
|  |           "(\\^|\\#)\\s*(.+)\\s*" + //  #foo (# == $2, foo == $3) | ||||||
|  |           ctag +                    // }} | ||||||
|  |  | ||||||
|  |           "\n*([\\s\\S]*?)" +       // between the tag ($2). leading newlines are dropped | ||||||
|  |  | ||||||
|  |           otag +                    // {{ | ||||||
|  |           "\\/\\s*\\3\\s*" +        //  /foo (backreference to the opening tag). | ||||||
|  |           ctag +                    // }} | ||||||
|  |  | ||||||
|  |           "\\s*([\\s\\S]*)$",       // everything else in the string ($4). leading whitespace is dropped. | ||||||
|  |  | ||||||
|  |         "g"); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |       // for each {{#foo}}{{/foo}} section do... | ||||||
|  |       return template.replace(regex, function (match, before, type, name, content, after) { | ||||||
|  |         // before contains only tags, no sections | ||||||
|  |         var renderedBefore = before ? that.render_tags(before, context, partials, true) : "", | ||||||
|  |  | ||||||
|  |         // after may contain both sections and tags, so use full rendering function | ||||||
|  |             renderedAfter = after ? that.render(after, context, partials, true) : "", | ||||||
|  |  | ||||||
|  |         // will be computed below | ||||||
|  |             renderedContent, | ||||||
|  |  | ||||||
|  |             value = that.find(name, context); | ||||||
|  |  | ||||||
|  |         if (type === "^") { // inverted section | ||||||
|  |           if (!value || Array.isArray(value) && value.length === 0) { | ||||||
|  |             // false or empty list, render it | ||||||
|  |             renderedContent = that.render(content, context, partials, true); | ||||||
|  |           } else { | ||||||
|  |             renderedContent = ""; | ||||||
|  |           } | ||||||
|  |         } else if (type === "#") { // normal section | ||||||
|  |           if (Array.isArray(value)) { // Enumerable, Let's loop! | ||||||
|  |             renderedContent = that.map(value, function (row) { | ||||||
|  |               return that.render(content, that.create_context(row), partials, true); | ||||||
|  |             }).join(""); | ||||||
|  |           } else if (that.is_object(value)) { // Object, Use it as subcontext! | ||||||
|  |             renderedContent = that.render(content, that.create_context(value), | ||||||
|  |               partials, true); | ||||||
|  |           } else if (typeof value == "function") { | ||||||
|  |             // higher order section | ||||||
|  |             renderedContent = value.call(context, content, function (text) { | ||||||
|  |               return that.render(text, context, partials, true); | ||||||
|  |             }); | ||||||
|  |           } else if (value) { // boolean section | ||||||
|  |             renderedContent = that.render(content, context, partials, true); | ||||||
|  |           } else { | ||||||
|  |             renderedContent = ""; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return renderedBefore + renderedContent + renderedAfter; | ||||||
|  |       }); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Replace {{foo}} and friends with values from our view | ||||||
|  |     */ | ||||||
|  |     render_tags: function (template, context, partials, in_recursion) { | ||||||
|  |       // tit for tat | ||||||
|  |       var that = this; | ||||||
|  |  | ||||||
|  |       var new_regex = function () { | ||||||
|  |         return that.getCachedRegex("render_tags", function (otag, ctag) { | ||||||
|  |           return new RegExp(otag + "(=|!|>|&|\\{|%)?([^#\\^]+?)\\1?" + ctag + "+", "g"); | ||||||
|  |         }); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       var regex = new_regex(); | ||||||
|  |       var tag_replace_callback = function (match, operator, name) { | ||||||
|  |         switch(operator) { | ||||||
|  |         case "!": // ignore comments | ||||||
|  |           return ""; | ||||||
|  |         case "=": // set new delimiters, rebuild the replace regexp | ||||||
|  |           that.set_delimiters(name); | ||||||
|  |           regex = new_regex(); | ||||||
|  |           return ""; | ||||||
|  |         case ">": // render partial | ||||||
|  |           return that.render_partial(name, context, partials); | ||||||
|  |         case "{": // the triple mustache is unescaped | ||||||
|  |         case "&": // & operator is an alternative unescape method | ||||||
|  |           return that.find(name, context); | ||||||
|  |         default: // escape the value | ||||||
|  |           return escapeHTML(that.find(name, context)); | ||||||
|  |         } | ||||||
|  |       }; | ||||||
|  |       var lines = template.split("\n"); | ||||||
|  |       for(var i = 0; i < lines.length; i++) { | ||||||
|  |         lines[i] = lines[i].replace(regex, tag_replace_callback, this); | ||||||
|  |         if (!in_recursion) { | ||||||
|  |           this.send(lines[i]); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (in_recursion) { | ||||||
|  |         return lines.join("\n"); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     set_delimiters: function (delimiters) { | ||||||
|  |       var dels = delimiters.split(" "); | ||||||
|  |       this.otag = this.escape_regex(dels[0]); | ||||||
|  |       this.ctag = this.escape_regex(dels[1]); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     escape_regex: function (text) { | ||||||
|  |       // thank you Simon Willison | ||||||
|  |       if (!arguments.callee.sRE) { | ||||||
|  |         var specials = [ | ||||||
|  |           '/', '.', '*', '+', '?', '|', | ||||||
|  |           '(', ')', '[', ']', '{', '}', '\\' | ||||||
|  |         ]; | ||||||
|  |         arguments.callee.sRE = new RegExp( | ||||||
|  |           '(\\' + specials.join('|\\') + ')', 'g' | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |       return text.replace(arguments.callee.sRE, '\\$1'); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       find `name` in current `context`. That is find me a value | ||||||
|  |       from the view object | ||||||
|  |     */ | ||||||
|  |     find: function (name, context) { | ||||||
|  |       name = trim(name); | ||||||
|  |  | ||||||
|  |       // Checks whether a value is thruthy or false or 0 | ||||||
|  |       function is_kinda_truthy(bool) { | ||||||
|  |         return bool === false || bool === 0 || bool; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var value; | ||||||
|  |  | ||||||
|  |       // check for dot notation eg. foo.bar | ||||||
|  |       if (name.match(/([a-z_]+)\./ig)) { | ||||||
|  |         var childValue = this.walk_context(name, context); | ||||||
|  |         if (is_kinda_truthy(childValue)) { | ||||||
|  |           value = childValue; | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         if (is_kinda_truthy(context[name])) { | ||||||
|  |           value = context[name]; | ||||||
|  |         } else if (is_kinda_truthy(this.context[name])) { | ||||||
|  |           value = this.context[name]; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (typeof value == "function") { | ||||||
|  |         return value.apply(context); | ||||||
|  |       } | ||||||
|  |       if (value !== undefined) { | ||||||
|  |         return value; | ||||||
|  |       } | ||||||
|  |       // silently ignore unkown variables | ||||||
|  |       return ""; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     walk_context: function (name, context) { | ||||||
|  |       var path = name.split('.'); | ||||||
|  |       // if the var doesn't exist in current context, check the top level context | ||||||
|  |       var value_context = (context[path[0]] != undefined) ? context : this.context; | ||||||
|  |       var value = value_context[path.shift()]; | ||||||
|  |       while (value != undefined && path.length > 0) { | ||||||
|  |         value_context = value; | ||||||
|  |         value = value[path.shift()]; | ||||||
|  |       } | ||||||
|  |       // if the value is a function, call it, binding the correct context | ||||||
|  |       if (typeof value == "function") { | ||||||
|  |         return value.apply(value_context); | ||||||
|  |       } | ||||||
|  |       return value; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     // Utility methods | ||||||
|  |  | ||||||
|  |     /* includes tag */ | ||||||
|  |     includes: function (needle, haystack) { | ||||||
|  |       return haystack.indexOf(this.otag + needle) != -1; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     // by @langalex, support for arrays of strings | ||||||
|  |     create_context: function (_context) { | ||||||
|  |       if (this.is_object(_context)) { | ||||||
|  |         return _context; | ||||||
|  |       } else { | ||||||
|  |         var iterator = "."; | ||||||
|  |         if (this.pragmas["IMPLICIT-ITERATOR"]) { | ||||||
|  |           iterator = this.pragmas["IMPLICIT-ITERATOR"].iterator; | ||||||
|  |         } | ||||||
|  |         var ctx = {}; | ||||||
|  |         ctx[iterator] = _context; | ||||||
|  |         return ctx; | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     is_object: function (a) { | ||||||
|  |       return a && typeof a == "object"; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Why, why, why? Because IE. Cry, cry cry. | ||||||
|  |     */ | ||||||
|  |     map: function (array, fn) { | ||||||
|  |       if (typeof array.map == "function") { | ||||||
|  |         return array.map(fn); | ||||||
|  |       } else { | ||||||
|  |         var r = []; | ||||||
|  |         var l = array.length; | ||||||
|  |         for(var i = 0; i < l; i++) { | ||||||
|  |           r.push(fn(array[i])); | ||||||
|  |         } | ||||||
|  |         return r; | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     getCachedRegex: function (name, generator) { | ||||||
|  |       var byOtag = regexCache[this.otag]; | ||||||
|  |       if (!byOtag) { | ||||||
|  |         byOtag = regexCache[this.otag] = {}; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var byCtag = byOtag[this.ctag]; | ||||||
|  |       if (!byCtag) { | ||||||
|  |         byCtag = byOtag[this.ctag] = {}; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var regex = byCtag[name]; | ||||||
|  |       if (!regex) { | ||||||
|  |         regex = byCtag[name] = generator(this.otag, this.ctag); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return regex; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return({ | ||||||
|  |     name: "mustache.js", | ||||||
|  |     version: "0.4.0", | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |       Turns a template and view into HTML | ||||||
|  |     */ | ||||||
|  |     to_html: function (template, view, partials, send_fun) { | ||||||
|  |       var renderer = new Renderer(); | ||||||
|  |       if (send_fun) { | ||||||
|  |         renderer.send = send_fun; | ||||||
|  |       } | ||||||
|  |       renderer.render(template, view || {}, partials); | ||||||
|  |       if (!send_fun) { | ||||||
|  |         return renderer.buffer.join("\n"); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | }(); | ||||||
|  | /*! | ||||||
|  |   ICanHaz.js -- by @HenrikJoreteg | ||||||
|  | */ | ||||||
|  | /*global  */ | ||||||
|  | (function () { | ||||||
|  |     function trim(stuff) { | ||||||
|  |         if (''.trim) return stuff.trim(); | ||||||
|  |         else return stuff.replace(/^\s+/, '').replace(/\s+$/, ''); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Establish the root object, `window` in the browser, or `global` on the server. | ||||||
|  |     var root = this; | ||||||
|  |  | ||||||
|  |     var ich = { | ||||||
|  |         VERSION: "0.10.2", | ||||||
|  |         templates: {}, | ||||||
|  |  | ||||||
|  |         // grab jquery or zepto if it's there | ||||||
|  |         $: (typeof window !== 'undefined') ? window.jQuery || window.Zepto || null : null, | ||||||
|  |  | ||||||
|  |         // public function for adding templates | ||||||
|  |         // can take a name and template string arguments | ||||||
|  |         // or can take an object with name/template pairs | ||||||
|  |         // We're enforcing uniqueness to avoid accidental template overwrites. | ||||||
|  |         // If you want a different template, it should have a different name. | ||||||
|  |         addTemplate: function (name, templateString) { | ||||||
|  |             if (typeof name === 'object') { | ||||||
|  |                 for (var template in name) { | ||||||
|  |                     this.addTemplate(template, name[template]); | ||||||
|  |                 } | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             if (ich[name]) { | ||||||
|  |                 console.error("Invalid name: " + name + "."); | ||||||
|  |             } else if (ich.templates[name]) { | ||||||
|  |                 console.error("Template \"" + name + "  \" exists"); | ||||||
|  |             } else { | ||||||
|  |                 ich.templates[name] = templateString; | ||||||
|  |                 ich[name] = function (data, raw) { | ||||||
|  |                     data = data || {}; | ||||||
|  |                     var result = Mustache.to_html(ich.templates[name], data, ich.templates); | ||||||
|  |                     return (ich.$ && !raw) ? ich.$(trim(result)) : result; | ||||||
|  |                 }; | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         // clears all retrieval functions and empties cache | ||||||
|  |         clearAll: function () { | ||||||
|  |             for (var key in ich.templates) { | ||||||
|  |                 delete ich[key]; | ||||||
|  |             } | ||||||
|  |             ich.templates = {}; | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         // clears/grabs | ||||||
|  |         refresh: function () { | ||||||
|  |             ich.clearAll(); | ||||||
|  |             ich.grabTemplates(); | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         // grabs templates from the DOM and caches them. | ||||||
|  |         // Loop through and add templates. | ||||||
|  |         // Whitespace at beginning and end of all templates inside <script> tags will | ||||||
|  |         // be trimmed. If you want whitespace around a partial, add it in the parent, | ||||||
|  |         // not the partial. Or do it explicitly using <br/> or   | ||||||
|  |         grabTemplates: function () { | ||||||
|  |             var i, | ||||||
|  |                 l, | ||||||
|  |                 scripts = document.getElementsByTagName('script'), | ||||||
|  |                 script, | ||||||
|  |                 trash = []; | ||||||
|  |             for (i = 0, l = scripts.length; i < l; i++) { | ||||||
|  |                 script = scripts[i]; | ||||||
|  |                 if (script && script.innerHTML && script.id && (script.type === "text/html" || script.type === "text/x-icanhaz")) { | ||||||
|  |                     ich.addTemplate(script.id, trim(script.innerHTML)); | ||||||
|  |                     trash.unshift(script); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             for (i = 0, l = trash.length; i < l; i++) { | ||||||
|  |                 trash[i].parentNode.removeChild(trash[i]); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Export the ICanHaz object for **Node.js**, with | ||||||
|  |     // backwards-compatibility for the old `require()` API. If we're in | ||||||
|  |     // the browser, add `ich` as a global object via a string identifier, | ||||||
|  |     // for Closure Compiler "advanced" mode. | ||||||
|  |     if (typeof exports !== 'undefined') { | ||||||
|  |         if (typeof module !== 'undefined' && module.exports) { | ||||||
|  |             exports = module.exports = ich; | ||||||
|  |         } | ||||||
|  |         exports.ich = ich; | ||||||
|  |     } else { | ||||||
|  |         root['ich'] = ich; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (typeof document !== 'undefined') { | ||||||
|  |         if (ich.$) { | ||||||
|  |             ich.$(function () { | ||||||
|  |                 ich.grabTemplates(); | ||||||
|  |             }); | ||||||
|  |         } else { | ||||||
|  |             document.addEventListener('DOMContentLoaded', function () { | ||||||
|  |                 ich.grabTemplates(); | ||||||
|  |             }, true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | })(); | ||||||
|  | })(); | ||||||
							
								
								
									
										358
									
								
								nodejs/public/js/app.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										358
									
								
								nodejs/public/js/app.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,358 @@ | |||||||
|  | var app = {}; | ||||||
|  |  | ||||||
|  | app.api = (function(app){ | ||||||
|  | 	var baseURL = '/api/' | ||||||
|  |  | ||||||
|  | 	function post(url, data, callack){ | ||||||
|  | 		$.ajax({ | ||||||
|  | 			type: 'POST', | ||||||
|  | 			url: baseURL+url, | ||||||
|  | 			headers:{ | ||||||
|  | 				'auth-token': app.auth.getToken() | ||||||
|  | 			}, | ||||||
|  | 			data: JSON.stringify(data), | ||||||
|  | 			contentType: "application/json; charset=utf-8", | ||||||
|  | 			dataType: "json", | ||||||
|  | 			complete: function(res, text){ | ||||||
|  | 				callack( | ||||||
|  | 					text !== 'success' ? res.statusText : null, | ||||||
|  | 					JSON.parse(res.responseText), | ||||||
|  | 					res.status | ||||||
|  | 				) | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function put(url, data, callack){ | ||||||
|  | 		$.ajax({ | ||||||
|  | 			type: 'PUT', | ||||||
|  | 			url: baseURL+url, | ||||||
|  | 			headers:{ | ||||||
|  | 				'auth-token': app.auth.getToken() | ||||||
|  | 			}, | ||||||
|  | 			data: JSON.stringify(data), | ||||||
|  | 			contentType: "application/json; charset=utf-8", | ||||||
|  | 			dataType: "json", | ||||||
|  | 			complete: function(res, text){ | ||||||
|  | 				callack( | ||||||
|  | 					text !== 'success' ? res.statusText : null, | ||||||
|  | 					JSON.parse(res.responseText), | ||||||
|  | 					res.status | ||||||
|  | 				) | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function remove(url, callack, callack2){ | ||||||
|  | 		if(!$.isFunction(callack)) callack = callack2; | ||||||
|  | 		$.ajax({ | ||||||
|  | 			type: 'delete', | ||||||
|  | 			url: baseURL+url, | ||||||
|  | 			headers:{ | ||||||
|  | 				'auth-token': app.auth.getToken() | ||||||
|  | 			}, | ||||||
|  | 			contentType: "application/json; charset=utf-8", | ||||||
|  | 			dataType: "json", | ||||||
|  | 			complete: function(res, text){ | ||||||
|  | 				callack( | ||||||
|  | 					text !== 'success' ? res.statusText : null, | ||||||
|  | 					JSON.parse(res.responseText), | ||||||
|  | 					res.status | ||||||
|  | 				) | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function get(url, callack){ | ||||||
|  | 		$.ajax({ | ||||||
|  | 			type: 'GET', | ||||||
|  | 			url: baseURL+url, | ||||||
|  | 			headers:{ | ||||||
|  | 				'auth-token': app.auth.getToken() | ||||||
|  | 			}, | ||||||
|  | 			contentType: "application/json; charset=utf-8", | ||||||
|  | 			dataType: "json", | ||||||
|  | 			complete: function(res, text){ | ||||||
|  | 				callack( | ||||||
|  | 					text !== 'success' ? res.statusText : null, | ||||||
|  | 					JSON.parse(res.responseText), | ||||||
|  | 					res.status | ||||||
|  | 				) | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return {post: post, get: get, put: put, delete: remove} | ||||||
|  | })(app) | ||||||
|  |  | ||||||
|  | app.auth = (function(app) { | ||||||
|  | 	var user = {} | ||||||
|  | 	function setToken(token){ | ||||||
|  | 		localStorage.setItem('APIToken', token); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function getToken(){ | ||||||
|  | 		return localStorage.getItem('APIToken'); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function isLoggedIn(callack){ | ||||||
|  | 		if(getToken()){ | ||||||
|  | 			return app.api.get('user/me', function(error, data){ | ||||||
|  | 				if(!error) app.auth.user = data; | ||||||
|  | 				return callack(error, data); | ||||||
|  | 			}); | ||||||
|  | 		}else{ | ||||||
|  | 			callack(null, false); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function logIn(args, callack){ | ||||||
|  | 		app.api.post('auth/login', args, function(error, data){ | ||||||
|  | 			if(data.login){ | ||||||
|  | 				setToken(data.token); | ||||||
|  | 			} | ||||||
|  | 			callack(error, !!data.token); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function logOut(callack){ | ||||||
|  | 		localStorage.removeItem('APIToken'); | ||||||
|  | 		callack(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function makeUserFromInvite(args, callack){ | ||||||
|  | 		app.api.post('auth/invite/'+ args.token, args, function(error, data){ | ||||||
|  | 			if(data.login){ | ||||||
|  | 				callack(null, data); | ||||||
|  | 				setToken(data.token); | ||||||
|  | 			} | ||||||
|  | 			callack(error, !!data.token); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return { | ||||||
|  | 		getToken: getToken, | ||||||
|  | 		setToken: setToken, | ||||||
|  | 		isLoggedIn: isLoggedIn, | ||||||
|  | 		logIn: logIn, | ||||||
|  | 		logOut: logOut, | ||||||
|  | 		makeUserFromInvite: makeUserFromInvite, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | })(app); | ||||||
|  |  | ||||||
|  | app.user = (function(app){ | ||||||
|  | 	function list(callack){ | ||||||
|  | 		app.api.get('user/?detail=true', function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function add(args, callack){ | ||||||
|  | 		app.api.post('user/', args, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function remove(args, callack){ | ||||||
|  | 		if(!confirm('Delete '+ args.uid+ 'user?')) return false; | ||||||
|  | 		app.api.delete('user/'+ args.uid, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function changePassword(args, callack){ | ||||||
|  | 		app.api.put('users/'+ arg.uid || '', args, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function createInvite(callack){ | ||||||
|  | 		app.api.post('user/invite', {}, function(error, data, status){ | ||||||
|  | 			callack(error, data);	 | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function consumeInvite(args){ | ||||||
|  | 		app.api.post('/auth/invite/'+args.token, args, function(error, data){ | ||||||
|  | 			if(data.token){ | ||||||
|  | 				app.auth.setToken(data.token) | ||||||
|  | 				return callack(null, true) | ||||||
|  | 			} | ||||||
|  | 			callack(error) | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return {list, remove, createInvite}; | ||||||
|  |  | ||||||
|  | })(app); | ||||||
|  |  | ||||||
|  | app.host = (function(app){ | ||||||
|  | 	function list(callack){ | ||||||
|  | 		app.api.get('host/?detail=true', function(error, data){ | ||||||
|  | 			callack(error, data.hosts) | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function get(host, callack){ | ||||||
|  | 		app.api.get('host/' + host, function(error, data){ | ||||||
|  | 			callack(error, data) | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function add(args, callack){ | ||||||
|  | 		app.api.post('host/', args, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function edit(args, callack){ | ||||||
|  | 		app.api.put('host/' + args.edit_host, args, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function remove(args, callack){ | ||||||
|  | 		app.api.delete('host/'+ args.host, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return { | ||||||
|  | 		list: list, | ||||||
|  | 		get: get, | ||||||
|  | 		add: add, | ||||||
|  | 		edit: edit, | ||||||
|  | 		remove: remove, | ||||||
|  | 	} | ||||||
|  | })(app); | ||||||
|  |  | ||||||
|  | app.group = (function(app){ | ||||||
|  | 	function list(callack){ | ||||||
|  | 		app.api.get('group?detail=true', function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function remove(args, callack){ | ||||||
|  | 		app.api.delete('group/'+args.cn, function(error, data){ | ||||||
|  | 			callack(error, data); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return {list, remove} | ||||||
|  | })(app) | ||||||
|  |  | ||||||
|  | app.util = (function(app){ | ||||||
|  |  | ||||||
|  | 	function getUrlParameter(name) { | ||||||
|  | 	    name = name.replace(/[\[]/, '\\[').replace(/[\]]/, '\\]'); | ||||||
|  | 	    var regex = new RegExp('[\\?&]' + name + '=([^&#]*)'); | ||||||
|  | 	    var results = regex.exec(location.search); | ||||||
|  | 	    return results === null ? '' : decodeURIComponent(results[1].replace(/\+/g, ' ')); | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	function actionMessage(message, $target, type){ | ||||||
|  | 		message = message || ''; | ||||||
|  | 		$target = $target.closest('div.card').find('.actionMessage'); | ||||||
|  | 		type = type || 'info'; | ||||||
|  |  | ||||||
|  | 		if($target.html() === message) return; | ||||||
|  |  | ||||||
|  | 		if($target.html()){ | ||||||
|  | 			$target.slideUp('fast', function(){ | ||||||
|  | 				$target.html('') | ||||||
|  | 				$target.removeClass (function (index, className) { | ||||||
|  | 					return (className.match (/(^|\s)bg-\S+/g) || []).join(' '); | ||||||
|  | 				}); | ||||||
|  | 				if(message) actionMessage(message, $target, type); | ||||||
|  | 			}) | ||||||
|  | 			return; | ||||||
|  | 		}else{ | ||||||
|  | 			if(type) $target.addClass('bg-' + type); | ||||||
|  | 			$target.html(message).slideDown('fast'); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	$.fn.serializeObject = function() { | ||||||
|  | 	    var  | ||||||
|  | 	        arr = $(this).serializeArray(),  | ||||||
|  | 	        obj = {}; | ||||||
|  | 	     | ||||||
|  | 	    for(var i = 0; i < arr.length; i++) { | ||||||
|  | 	        if(obj[arr[i].name] === undefined) { | ||||||
|  | 	            obj[arr[i].name] = arr[i].value; | ||||||
|  | 	        } else { | ||||||
|  | 	            if(!(obj[arr[i].name] instanceof Array)) { | ||||||
|  | 	                obj[arr[i].name] = [obj[arr[i].name]]; | ||||||
|  | 	            } | ||||||
|  | 	            obj[arr[i].name].push(arr[i].value); | ||||||
|  | 	        } | ||||||
|  | 	    } | ||||||
|  | 	    return obj; | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	return { | ||||||
|  | 		getUrlParameter: getUrlParameter, | ||||||
|  | 		actionMessage: actionMessage | ||||||
|  | 	} | ||||||
|  | })(app); | ||||||
|  |  | ||||||
|  | $.holdReady( true ); | ||||||
|  | if(!location.pathname.includes('/login')){ | ||||||
|  | 	app.auth.isLoggedIn(function(error, isLoggedIn){ | ||||||
|  | 		if(error || !isLoggedIn){ | ||||||
|  | 			app.auth.logOut(function(){}) | ||||||
|  | 			location.replace('/login/?redirect='+location.pathname); | ||||||
|  | 		}else{ | ||||||
|  | 			$.holdReady( false ); | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | }else{ | ||||||
|  | 	$.holdReady( false ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | $( document ).ready( function () { | ||||||
|  |  | ||||||
|  | 	$( 'div.row' ).fadeIn( 'slow' ); //show the page | ||||||
|  |  | ||||||
|  | 	//panel button's | ||||||
|  | 	$( '.fa-arrows-v' ).click( function () { | ||||||
|  | 		$( this ).closest( '.card' ).find( '.card-body' ).slideToggle( 'fast' ); | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	$( '.glyphicon-refresh' ).each( function () { | ||||||
|  | 		$(this).click( function () { | ||||||
|  | 			tableAJAX(); | ||||||
|  | 		}); | ||||||
|  | 	}); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | //ajax form submit | ||||||
|  | function formAJAX( btn, del ) { | ||||||
|  | 	event.preventDefault(); // avoid to execute the actual submit of the form. | ||||||
|  | 	var $form = $(btn).closest( '[action]' ); // gets the 'form' parent | ||||||
|  | 	var formData = $form.find( '[name]' ).serializeObject(); // builds query formDataing | ||||||
|  | 	var method = $form.attr('method') || 'post'; | ||||||
|  |  | ||||||
|  | 	if( !$form.validate()) { | ||||||
|  | 		app.util.actionMessage('Please fix the form errors.', $form, 'danger') | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	app.util.actionMessage(  | ||||||
|  | 		'<div class="spinner-border" role="status"><span class="sr-only">Loading...</span></div>', | ||||||
|  | 		$form, | ||||||
|  | 		'info' | ||||||
|  | 	); | ||||||
|  |  | ||||||
|  | 	app.api[method]($form.attr('action'), formData, function(error, data){ | ||||||
|  | 		app.util.actionMessage(data.message, $form, error ? 'danger' : 'success'); //re-populate table | ||||||
|  | 		if(!error){ | ||||||
|  | 			$form.trigger("reset"); | ||||||
|  | 			eval($form.attr('evalAJAX')); //gets JS to run after completion | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										1999
									
								
								nodejs/public/js/bootstrap.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										1999
									
								
								nodejs/public/js/bootstrap.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								nodejs/public/js/bootstrap.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										6
									
								
								nodejs/public/js/bootstrap.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1
									
								
								nodejs/public/js/jquery-validate.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								nodejs/public/js/jquery-validate.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | (function(e){function r(e,t){var n=t.attr("validate").split(":"),r=n[1],i=t.val(),s=n[0];t.siblings("label").children("b").remove();t.parent().removeClass("has-error");if(isNaN(r)===false&&r&&i.length<r){return e.processValidation("Must be "+r+" characters",t)}if(isNaN(r)===false&&i.length===0){return}if(s in e.rule){return e.processValidation(e.rule[s].apply(this,[i,r]),t)}}var t={rule:{eq:function(t,n){var r=e("[name="+n.nameOpt+"]").val();if(t!=r){return"Miss-match"}}},form:{alertCount:false,alertCountMessage:" errors!"},processValidation:function(t,r){if(typeof t=="undefined"||t==true){return}e("<b>").html(" - "+t).appendTo(r.siblings("label"));r.parent().addClass("has-error");n++;return false}};var n=0;e.fn.validate=function(i){n=0;var s=false,o=e.extend({},t,i);if(this.is("[validate]")){r(o,this)}else{s=true;this.find("[validate]").each(function(){r(o,e(this))})}if(n===0){return true}else{if(o.form.alertCount&&s)alert(n+o.form.alertCountMessage);event.preventDefault();return false}};jQuery.extend({validateSettings:function(n){return t=e.extend({},t,n)}})})(jQuery);$.validateSettings({rule:{ip:function(e){e=e.split(".");if(e.length!=4){return"Malformed IP"}$.each(e,function(e,t){if(t>255||t<0){return"Malformed IP"}})},host:function(e){var t=/^(?=.{1,255}$)[0-9A-Za-z](?:(?:[0-9A-Za-z]|-){0,61}[0-9A-Za-z])?(?:\.[0-9A-Za-z](?:(?:[0-9A-Za-z]|-){0,61}[0-9A-Za-z])?)*\.?$/;if(t.test(e)===false){return"Invalid"}},user:function(e){var t=/^[a-z0-9\_\-\@\.]{1,32}$/;if(t.test(e)===false){return"Invalid"}},password:function(e){var t=/^(?=[^\d_].*?\d)\w(\w|[!@#$%]){1,48}/;if(t.test(e)===false){return"Try again"}}}}) | ||||||
							
								
								
									
										2
									
								
								nodejs/public/js/jquery.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								nodejs/public/js/jquery.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										23
									
								
								nodejs/public/js/jquery.syntaxhighlighter.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										23
									
								
								nodejs/public/js/jquery.syntaxhighlighter.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | /* | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  |  MIT License {@link http://creativecommons.org/licenses/MIT/} | ||||||
|  | */ | ||||||
|  | "undefined"===typeof window.console&&(window.console={}); | ||||||
|  | "undefined"===typeof window.console.emulated&&("function"===typeof window.console.log?window.console.hasLog=!0:("undefined"===typeof window.console.log&&(window.console.log=function(){}),window.console.hasLog=!1),"function"===typeof window.console.debug?window.console.hasDebug=!0:("undefined"===typeof window.console.debug&&(window.console.debug=!window.console.hasLog?function(){}:function(){for(var a=["console.debug:"],b=0;b<arguments.length;b++)a.push(arguments[b]);window.console.log.apply(window.console, | ||||||
|  | a)}),window.console.hasDebug=!1),"function"===typeof window.console.warn?window.console.hasWarn=!0:("undefined"===typeof window.console.warn&&(window.console.warn=!window.console.hasLog?function(){}:function(){for(var a=["console.warn:"],b=0;b<arguments.length;b++)a.push(arguments[b]);window.console.log.apply(window.console,a)}),window.console.hasWarn=!1),"function"===typeof window.console.error?window.console.hasError=!0:("undefined"===typeof window.console.error&&(window.console.error=function(){var a= | ||||||
|  | "An error has occured.";if(window.console.hasLog){for(var a=["console.error:"],b=0;b<arguments.length;b++)a.push(arguments[b]);window.console.log.apply(window.console,a);a="An error has occured. More information is available in your browser's javascript console."}for(b=0;b<arguments.length;++b){if(typeof arguments[b]!=="string")break;a=a+("\n"+arguments[b])}if(typeof Error!=="undefined")throw Error(a);throw a;}),window.console.hasError=!1),"function"===typeof window.console.trace?window.console.hasTrace= | ||||||
|  | !0:("undefined"===typeof window.console.trace&&(window.console.trace=function(){window.console.error("console.trace does not exist")}),window.console.hasTrace=!1),window.console.emulated=!0); | ||||||
|  | (function(a){a.appendStylesheet=a.appendStylesheet||function(b,c){if(!document.body){setTimeout(function(){a.appendStylesheet.apply(a,[b,c])},500);return a}var d="stylesheet-"+b.replace(/[^a-zA-Z0-9]/g,""),e=a("#"+d);typeof c==="undefined"&&(c=false);if(e.length===1)if(c)e.remove();else return a;var e=document.getElementsByTagName(a.browser.safari?"head":"body")[0],f=document.createElement("link");f.type="text/css";f.rel="stylesheet";f.media="screen";f.href=b;f.id=d;e.appendChild(f);return a};a.appendScript= | ||||||
|  | a.appendScript||function(b,c){if(!document.body){setTimeout(function(){a.appendScript.apply(a,[b,c])},500);return a}var d="script-"+b.replace(/[^a-zA-Z0-9]/g,""),e=a("#"+d);typeof c==="undefined"&&(c=false);if(e.length===1)if(c)e.remove();else return a;var e=document.getElementsByTagName(a.browser.safari?"head":"body")[0],f=document.createElement("script");f.type="text/javascript";f.src=b;f.id=d;e.appendChild(f);return a}})(jQuery); | ||||||
|  | (function(a){a.fn.findAndSelf=a.fn.findAndSelf||function(b){return a(this).find(b).andSelf().filter(b)};Number.prototype.replace=Number.prototype.replace||function(){return(""+this).replace.apply(this,arguments)};a.SyntaxHighlighter?window.console.warn("SyntaxHighlighter has already been defined..."):a.SyntaxHighlighter={config:{load:true,highlight:true,debug:false,wrapLines:true,lineNumbers:true,stripEmptyStartFinishLines:true,stripInitialWhitespace:true,alternateLines:false,defaultClassname:"highlight", | ||||||
|  | theme:"balupton",themes:["balupton"],addSparkleExtension:true,prettifyBaseUrl:"http://balupton.github.com/jquery-syntaxhighlighter/prettify",baseUrl:"http://balupton.github.com/jquery-syntaxhighlighter"},init:function(b){var c=this.config,d=c.baseUrl;if(d[d.length-1]==="/")c.baseUrl=d.substr(0,d.length-2);delete d;a.extend(true,this.config,b||{});a.Sparkle&&a.Sparkle.addExtension("syntaxhighlighter",function(){a(this).syntaxHighlight()});a.fn.syntaxHighlight=a.fn.SyntaxHighlight=this.fn;c.load&&this.load(); | ||||||
|  | c.highlight&&this.highlight();return this},load:function(){var b=this.config,c=b.prettifyBaseUrl,d=b.baseUrl,b=b.themes;if(!this.loaded()){a.appendScript(c+"/prettify.min.js");a.appendStylesheet(c+"/prettify.min.css");a.appendStylesheet(d+"/styles/style.min.css");a.each(b,function(b,c){a.appendStylesheet(d+"/styles/theme-"+c+".min.css")});a.browser.msie&&a.appendStylesheet(d+"/styles/ie.min.css");this.loadedExtras=true}return this},loadedExtras:false,loaded:function(){return typeof prettyPrint!== | ||||||
|  | "undefined"&&this.loadedExtras},determineLanguage:function(a){for(var c=null,d=/lang(uage)?-([a-z0-9]+)/g,e=d.exec(a);e!==null;){c=e[2];e=d.exec(a)}return c},fn:function(){var b=a(this);a.SyntaxHighlighter.highlight({el:b});return this},highlight:function(b){typeof b!=="object"&&(b={});var c=this,d=c.config,e=b.el||false;e instanceof jQuery||(e=a("body"));if(c.loaded()){var f=d.defaultClassname,g="";if(typeof f==="array"){g="."+f.join(",.");f=f.join(" ")}else{f=""+f;g="."+f.replace(" ",",.")}if(g=== | ||||||
|  | "."||!f){window.console.error("SyntaxHighlighter.highlight: Invalid defaultClassname.",[this,arguments],[d.defaultClassname]);window.console.trace()}e=e.findAndSelf("code,pre").filter("[class*=lang],"+g).filter(":not(.prettyprint)");e.css({"overflow-y":"visible","overflow-x":"visible","white-space":"pre"}).addClass("prettyprint "+f).each(function(){var b=a(this),d=b.attr("class"),d=c.determineLanguage(d);b.addClass("lang-"+d)});d.lineNumbers&&e.addClass("linenums");d.theme&&e.addClass("theme-"+d.theme); | ||||||
|  | d.alternateLines&&e.addClass("alternate");prettyPrint();d.stripEmptyStartFinishLines&&e.find("li:first-child > :first-child, li:last-child > :first-child").each(function(){var b=a(this),d=/^([\r\n\s\t]|\ )*$/.test(b.html()),c=b.parent(),c=b.siblings();if(d&&(c.length===0||c.length===1&&c.filter(":last").is("br"))){c=b.parent();b=c.val();c.next().val(b);c.remove()}});d.stripInitialWhitespace&&e.find("li:first-child > :first-child").each(function(){var b=a(this),c=(b.html().match(/^(([\r\n\s\t]|\ )+)/)|| | ||||||
|  | [])[1]||"";c.length&&b.parent().siblings().children(":first-child").add(b).each(function(){var b=a(this),d=b.html(),d=d.replace(RegExp("^"+c,"gm"),"");b.html(d)})});d.wrapLines?e.css({"overflow-x":"hidden","overflow-y":"hidden","white-space":"pre-wrap","max-height":"none"}):e.css({"overflow-x":"auto","overflow-y":"auto","white-space":"normal","max-height":"500px"});return this}d.debug&&window.console.debug("SyntaxHighlighter.highlight: Chosen SyntaxHighlighter is not yet defined. Waiting 1200 ms then trying again."); | ||||||
|  | setTimeout(function(){c.highlight.apply(c,[b])},1200)}}})(jQuery); | ||||||
							
								
								
									
										4602
									
								
								nodejs/public/js/moment.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4602
									
								
								nodejs/public/js/moment.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1
									
								
								nodejs/public/js/mustache.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								nodejs/public/js/mustache.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										156
									
								
								nodejs/public/js/val.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										156
									
								
								nodejs/public/js/val.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,156 @@ | |||||||
|  | ( function( $ ) { | ||||||
|  | 	var settings = { | ||||||
|  | 		rule: { | ||||||
|  | 			eq: function( value, options ) { | ||||||
|  | 				var compare = $( '[name=' + options + ']' ).val(); | ||||||
|  | 	  | ||||||
|  | 				if ( value != compare ) { | ||||||
|  | 					return "Miss-match"; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		form: { | ||||||
|  | 			alertCount: false, //pop-up with error count | ||||||
|  | 			alertCountMessage: " errors!" | ||||||
|  | 		}, | ||||||
|  | 		 | ||||||
|  | 		processValidation: function ( error_message, $input ) { | ||||||
|  | 			if ( typeof error_message == 'undefined' || error_message == true ) { | ||||||
|  | 				return; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			$( '<b style="color:#dc3545" class="jvalidate-error">' ).html( ' - ' + error_message ).appendTo( $input.parents('.form-group').children( 'label' ) ); | ||||||
|  | 			$input.addClass("is-invalid"); | ||||||
|  | 			failedCount++; | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	var failedCount = 0; | ||||||
|  |  | ||||||
|  | 	function processRule( thisSettings, $input ) { | ||||||
|  | 		var attr = $input.attr( 'validate' ).split( ':' ), //array of params | ||||||
|  | 			requirement = attr[1], | ||||||
|  | 			value = $input.val(), //link to input value | ||||||
|  | 			rule = attr[0]; | ||||||
|  |  | ||||||
|  | 		$input.parents('.form-group').children( 'label' ).children( '.jvalidate-error' ).remove(); //removes old error | ||||||
|  | 		$input.removeClass( "is-invalid" ); //removes is-invalid class | ||||||
|  |  | ||||||
|  | 		//checks if field is required, and length  | ||||||
|  | 		if (isNaN(requirement) === false && requirement && value.length < requirement) { | ||||||
|  | 			return thisSettings.processValidation( 'Must be ' + requirement + ' characters', $input ); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		//checks if empty to stop processing  | ||||||
|  | 		if ( isNaN( requirement ) === false && value.length === 0 ) { | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if ( rule in thisSettings.rule ) { | ||||||
|  | 			return thisSettings.processValidation( thisSettings.rule[rule].apply( this, [value, requirement] ), $input ); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 		 | ||||||
|  | 	$.fn.validate = function( settingsObj, event ) { | ||||||
|  | 		event = event || window.event; | ||||||
|  | 		 | ||||||
|  | 		failedCount = 0; | ||||||
|  | 		var thisForm = false, | ||||||
|  | 			thisSettings = $.extend( true, settings, settingsObj ); | ||||||
|  |  | ||||||
|  | 		if ( this.is( '[validate]' ) ) { | ||||||
|  | 			processRule( thisSettings, this ); | ||||||
|  | 		} else { | ||||||
|  | 			thisForm = true; | ||||||
|  | 			this.find( '[validate]' ).each( function () { | ||||||
|  | 				if(!processRule( thisSettings, $( this ) )){ | ||||||
|  | 					// failedCount++; | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		this.attr('isValid', !failedCount); | ||||||
|  | 		if ( failedCount === 0 ) { //no errors | ||||||
|  | 			return true; | ||||||
|  | 		} else { //errors | ||||||
|  | 			if ( thisForm ){ | ||||||
|  | 				if(thisSettings.form.alertCount){ | ||||||
|  | 					alert( failedCount + thisSettings.form.alertCountMessage ); | ||||||
|  | 				} | ||||||
|  | 				/* if(event) event.returnValue = false; | ||||||
|  | 				if(event) event.preventDefault(); | ||||||
|  | 				return false; | ||||||
|  | 				 | ||||||
|  | 				if(event.preventDefault) if(event)*/  | ||||||
|  | 				//event.returnValue = false; | ||||||
|  | 				event.preventDefault(); | ||||||
|  | 				event.defaultPrevented; | ||||||
|  |  | ||||||
|  | 			} | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	jQuery.extend({ | ||||||
|  | 		validateSettings: function( settingsObj ) { | ||||||
|  | 			$.extend( true, settings, settingsObj ); | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		validateInit: function( ettingsObj ) { | ||||||
|  | 			$( '[action]' ).on( 'submit', function ( event, settingsObj ){ | ||||||
|  | 				$( this ).validate( settingsObj, event ); | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | }( jQuery )); | ||||||
|  |  | ||||||
|  | $.validateSettings({ | ||||||
|  | 	rule:{ | ||||||
|  | 		ip: function( value ) { | ||||||
|  | 			value = value.split( '.' ); | ||||||
|  |   | ||||||
|  | 			if ( value.length != 4 ) { | ||||||
|  | 				return "Malformed IP"; | ||||||
|  | 			} | ||||||
|  |   | ||||||
|  | 			$.each( value, function( key, value ) { | ||||||
|  | 				if( value > 255 || value < 0 ) { | ||||||
|  | 					return "Malformed IP"; | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		host: function( value ) { | ||||||
|  | 			var reg = /^(?=.{1,255}$)[0-9A-Za-z](?:(?:[0-9A-Za-z]|-){0,61}[0-9A-Za-z])?(?:\.[0-9A-Za-z](?:(?:[0-9A-Za-z]|-){0,61}[0-9A-Za-z])?)*\.?$/; | ||||||
|  | 			if ( reg.test( value ) === false ) { | ||||||
|  | 				return "Invalid"; | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		user: function( value ) { | ||||||
|  | 			var reg = /^[a-z0-9\_\-\@\.]{1,32}$/; | ||||||
|  | 			if ( reg.test( value ) === false ) { | ||||||
|  | 				return "Invalid"; | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  |   | ||||||
|  | 		password: function( value ) { | ||||||
|  | 			var reg = /^(?=[^\d_].*?\d)\w(\w|[!@#$%]){1,48}/; | ||||||
|  | 			if ( reg.test( value ) === false ) { | ||||||
|  | 				return "Weak password, Try again"; | ||||||
|  | 			} | ||||||
|  | 		}, | ||||||
|  | 		email: function( value ){ | ||||||
|  |  | ||||||
|  | 			//validated email address | ||||||
|  | 			//more testing | ||||||
|  | 			var pattern = new RegExp(/^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?$/i); | ||||||
|  | 				if( !pattern.test( value ) ){ | ||||||
|  | 					return 'Invalid'; | ||||||
|  | 				} | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | }); | ||||||
							
								
								
									
										8
									
								
								nodejs/public/js/waypoints-sticky.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										8
									
								
								nodejs/public/js/waypoints-sticky.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | // Generated by CoffeeScript 1.6.2 | ||||||
|  | /* | ||||||
|  | Sticky Elements Shortcut for jQuery Waypoints - v2.0.3 | ||||||
|  | Copyright (c) 2011-2013 Caleb Troughton | ||||||
|  | Dual licensed under the MIT license and GPL license. | ||||||
|  | https://github.com/imakewebthings/jquery-waypoints/blob/master/licenses.txt | ||||||
|  | */ | ||||||
|  | (function(){(function(t,n){if(typeof define==="function"&&define.amd){return define(["jquery","waypoints"],n)}else{return n(t.jQuery)}})(this,function(t){var n,s;n={wrapper:'<div class="sticky-wrapper" />',stuckClass:"stuck"};s=function(t,n){t.wrap(n.wrapper);return t.parent()};t.waypoints("extendFn","sticky",function(e){var i,r,a;r=t.extend({},t.fn.waypoint.defaults,n,e);i=s(this,r);a=r.handler;r.handler=function(n){var s,e;s=t(this).children(":first");e=n==="down"||n==="right";s.toggleClass(r.stuckClass,e);i.height(e?s.outerHeight():"");if(a!=null){return a.call(this,n)}};i.waypoint(r);return this.data("stuckClass",r.stuckClass)});return t.waypoints("extendFn","unsticky",function(){this.parent().waypoint("destroy");this.unwrap();return this.removeClass(this.data("stuckClass"))})})}).call(this); | ||||||
							
								
								
									
										8
									
								
								nodejs/public/js/waypoints.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										8
									
								
								nodejs/public/js/waypoints.min.js
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										116
									
								
								nodejs/routes/auth.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										116
									
								
								nodejs/routes/auth.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,116 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const router = require('express').Router(); | ||||||
|  | const {User} = require('../models/user'); | ||||||
|  | const {Auth, AuthToken} = require('../models/auth'); | ||||||
|  | const {PasswordResetToken} = require('../models/token'); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | router.post('/login', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let auth = await Auth.login(req.body); | ||||||
|  | 		return res.json({ | ||||||
|  | 			login: true, | ||||||
|  | 			token: auth.token.token, | ||||||
|  | 			message:`${req.body.uid} logged in!`, | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.all('/logout', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		if(req.user){ | ||||||
|  | 			await req.user.logout(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		res.json({message: 'Bye'}) | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/resetpassword', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let sent = await User.passwordReset(`${req.protocol}://${req.hostname}`, req.body.mail); | ||||||
|  |  | ||||||
|  | 		console.info('resetpassword for', req.body.mail, 'sent') | ||||||
|  |  | ||||||
|  | 		return res.json({ | ||||||
|  | 			message: 'If the emaill address is in our system, you will receive a message.' | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/resetpassword/:token', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await PasswordResetToken.get(req.params.token); | ||||||
|  |  | ||||||
|  | 		if(token.is_valid && 86400000+Number(token.created_on) > (new Date).getTime()){ | ||||||
|  | 			let user = await User.get(token.created_by); | ||||||
|  | 			await user.setPassword(req.body); | ||||||
|  | 			token.update({is_valid: false}); | ||||||
|  | 			return res.json({ | ||||||
|  | 				message: 'Password has been changed.' | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/invite/:token/:mailToken', async function(req, res, next) { | ||||||
|  | 	try{ | ||||||
|  | 		req.body.token = req.params.token; | ||||||
|  | 		req.body.mailToken = req.params.mailToken; | ||||||
|  | 		let user = await User.addByInvite(req.body); | ||||||
|  | 		let token = await AuthToken.add(user); | ||||||
|  |  | ||||||
|  | 		return res.json({ | ||||||
|  | 			user: user.uid, | ||||||
|  | 			token: token.token | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/invite/:token', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let data = { | ||||||
|  | 			token: req.params.token, | ||||||
|  | 			url: `${req.protocol}://${req.hostname}`, | ||||||
|  | 			mail: req.body.mail, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		await User.verifyEmail(data); | ||||||
|  | 		return res.send({message: 'sent'}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error) | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports = router; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | 	verify public ssh key | ||||||
|  | */ | ||||||
|  | // router.post('/verifykey', async function(req, res){ | ||||||
|  | // 	let key = req.body.key; | ||||||
|  |  | ||||||
|  | // 	try{ | ||||||
|  | // 		return res.json({ | ||||||
|  | // 			info: await Users.verifyKey(key) | ||||||
|  | // 		}); | ||||||
|  | // 	}catch(error){ | ||||||
|  | // 		return res.status(400).json({ | ||||||
|  | // 			message: 'Key is not a public key file!' | ||||||
|  | // 		}); | ||||||
|  | // 	} | ||||||
|  | 	 | ||||||
|  | // }); | ||||||
							
								
								
									
										82
									
								
								nodejs/routes/group.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								nodejs/routes/group.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,82 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const router = require('express').Router(); | ||||||
|  | const {User} = require('../models/user_ldap'); | ||||||
|  | const {Group} = require('../models/group_ldap');  | ||||||
|  |  | ||||||
|  | router.get('/', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let member = req.query.member ? await User.get(req.query.member) : {} | ||||||
|  |  | ||||||
|  | 		console.log('member', member) | ||||||
|  |  | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await Group[req.query.detail ? "listDetail" : "list"](member.dn) | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		req.body.owner = req.user.dn; | ||||||
|  | 		return res.json({ | ||||||
|  | 			results: await Group.add(req.body), | ||||||
|  | 			message: `${req.body.name} was added!` | ||||||
|  | 		}) | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/:name', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await Group.get(req.params.name) | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.put('/:name/:uid', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		var group = await Group.get(req.params.name); | ||||||
|  | 		var user = await User.get(req.params.uid); | ||||||
|  | 		return res.json({ | ||||||
|  | 			results: group.addMember(user), | ||||||
|  | 			message: `Added user ${req.params.uid} to ${req.params.name} group.` | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.delete('/:name/:uid', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		var group = await Group.get(req.params.name); | ||||||
|  | 		var user = await User.get(req.params.uid); | ||||||
|  | 		return res.json({ | ||||||
|  | 			results: group.removeMember(user), | ||||||
|  | 			message: `Removed user ${req.params.uid} from ${req.params.name} group.` | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.delete('/:name', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		var group = await Group.get(req.params.name); | ||||||
|  | 		return res.json({ | ||||||
|  | 			removed: await group.remove(), | ||||||
|  | 			results: group, | ||||||
|  | 			message: `Group ${req.params.name} Deleted` | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports = router; | ||||||
							
								
								
									
										75
									
								
								nodejs/routes/index.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										75
									
								
								nodejs/routes/index.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,75 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | var express = require('express'); | ||||||
|  | var router = express.Router(); | ||||||
|  | const moment = require('moment'); | ||||||
|  | const {InviteToken, PasswordResetToken} = require('./../models/token'); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* GET home page. */ | ||||||
|  | router.get('/', async function(req, res, next) { | ||||||
|  |   res.render('home', { title: 'Express' }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | /* GET home page. */ | ||||||
|  | router.get('/users', function(req, res, next) { | ||||||
|  |   res.render('users', { title: 'Express' }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/users/:uid', function(req, res, next) { | ||||||
|  |   res.render('home', { title: 'Express' }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/groups', function(req, res, next) { | ||||||
|  |   res.render('groups', { title: 'Express' }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |              | ||||||
|  | router.get('/login/resetpassword/:token', async function(req, res, next){ | ||||||
|  | 	let token = await PasswordResetToken.get(req.params.token); | ||||||
|  |  | ||||||
|  | 	if(token.is_valid && 86400000+Number(token.created_on) > (new Date).getTime()){ | ||||||
|  | 		res.render('reset_password', {token:token}); | ||||||
|  | 	}else{ | ||||||
|  | 		next({message: 'token not found', status: 404}); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/login/invite/:token/:mailToken', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		 | ||||||
|  | 		let token = await InviteToken.get(req.params.token); | ||||||
|  |  | ||||||
|  | 		if(token.is_valid && token.mail !== '__NONE__' && token.mail_token === req.params.mailToken){ | ||||||
|  | 			token.created_on = moment(token.created_on, 'x').fromNow(); | ||||||
|  |   			res.render('invite', { title: 'Express', invite: token }); | ||||||
|  | 		}else{ | ||||||
|  | 			next({message: 'token not found', status: 404}); | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/login/invite/:token', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await InviteToken.get(req.params.token); | ||||||
|  | 		token.created_on = moment(token.created_on, 'x').fromNow(); | ||||||
|  |  | ||||||
|  | 		if(token.is_valid){ | ||||||
|  |   			res.render('invite_email', { title: 'Express', invite: token }); | ||||||
|  | 		}else{ | ||||||
|  | 			next({message: 'token not found', status: 404}); | ||||||
|  | 		} | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* GET home page. */ | ||||||
|  | router.get('/login', function(req, res, next) { | ||||||
|  |   res.render('login', {redirect: req.query.redirect}); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports = router; | ||||||
							
								
								
									
										63
									
								
								nodejs/routes/token.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								nodejs/routes/token.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const router = require('express').Router(); | ||||||
|  | const {AuthToken} = require('../models/auth'); | ||||||
|  | const {Token, InviteToken} = require('../models/token'); | ||||||
|  |  | ||||||
|  | const tokens  = { | ||||||
|  | 	auth: AuthToken, | ||||||
|  | 	invite: InviteToken | ||||||
|  | } | ||||||
|  |  | ||||||
|  | router.get('/:name', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		console.log(tokens, req.params.name) | ||||||
|  |  | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await tokens[req.params.name][req.query.detail ? "listDetail" : "list"]() | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | router.get('/:name/:token', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await tokens[req.params.name].get(req.params.token) | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // router.delete('/:username', async function(req, res, next){ | ||||||
|  | // 	try{ | ||||||
|  | // 		let user = await User.get(req.params.username); | ||||||
|  |  | ||||||
|  | // 		return res.json({username: req.params.username, results: await user.remove()}) | ||||||
|  | // 	}catch(error){ | ||||||
|  | // 		next(error); | ||||||
|  | // 	} | ||||||
|  | // }); | ||||||
|  |  | ||||||
|  | module.exports = router; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | 	verify public ssh key | ||||||
|  | */ | ||||||
|  | // router.post('/verifykey', async function(req, res){ | ||||||
|  | // 	let key = req.body.key; | ||||||
|  |  | ||||||
|  | // 	try{ | ||||||
|  | // 		return res.json({ | ||||||
|  | // 			info: await Users.verifyKey(key) | ||||||
|  | // 		}); | ||||||
|  | // 	}catch(error){ | ||||||
|  | // 		return res.status(400).json({ | ||||||
|  | // 			message: 'Key is not a public key file!' | ||||||
|  | // 		}); | ||||||
|  | // 	} | ||||||
|  | 	 | ||||||
|  | // }); | ||||||
							
								
								
									
										118
									
								
								nodejs/routes/user.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										118
									
								
								nodejs/routes/user.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,118 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const router = require('express').Router(); | ||||||
|  | const {User} = require('../models/user');  | ||||||
|  |  | ||||||
|  | router.get('/', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await User[req.query.detail ? "listDetail" : "list"]() | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		req.body.created_by = req.user.uid | ||||||
|  |  | ||||||
|  | 		return res.json({results: await User.add(req.body)}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.delete('/:uid', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await User.get(req.params.uid); | ||||||
|  |  | ||||||
|  | 		return res.json({uid: req.params.uid, results: await user.remove()}) | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.put('/:uid', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await User.get(req.params.uid); | ||||||
|  |  | ||||||
|  | 		// console.log('update user', user); | ||||||
|  |  | ||||||
|  | 		return res.json({ | ||||||
|  | 			results: await user.update(req.body), | ||||||
|  | 			message: `Updated ${req.params.uid} user` | ||||||
|  |  | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/me', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		return res.json(await User.get({uid: req.user.uid})); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.put('/password', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		return res.json({results: await req.user.setPassword(req.body)}) | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.put('/:uid/password', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let user = await User.get(req.params.uid); | ||||||
|  | 		return res.json({ | ||||||
|  | 			results: await user.setPassword(req.body), | ||||||
|  | 			message: `User ${user.uid} password changed.` | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/invite', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let token = await req.user.invite(); | ||||||
|  |  | ||||||
|  | 		return res.json({token: token.token}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.post('/key', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		let added = await User.addSSHkey({ | ||||||
|  | 			uid: req.user.uid, | ||||||
|  | 			key: req.body.key | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 		return res.status(added === true ? 200 : 400).json({ | ||||||
|  | 			message: added | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | router.get('/:uid', async function(req, res, next){ | ||||||
|  | 	try{ | ||||||
|  | 		return res.json({ | ||||||
|  | 			results:  await User.get(req.params.uid), | ||||||
|  | 		}); | ||||||
|  | 	}catch(error){ | ||||||
|  | 		next(error); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports = router; | ||||||
							
								
								
									
										80
									
								
								nodejs/utils/object_validate.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								nodejs/utils/object_validate.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,80 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const process_type = { | ||||||
|  | 	number: function(key, value){ | ||||||
|  | 		if(key.min && value < key.min) return `is to small, min ${key.min}.` | ||||||
|  | 		if(key.max && value > key.max) return `is to large, max ${key.max}.` | ||||||
|  | 	}, | ||||||
|  | 	string: function(key, value){ | ||||||
|  | 		if(key.min && value.length < key.min) return `is too short, min ${key.min}.` | ||||||
|  | 		if(key.max && value.length > key.max) return `is too short, max ${key.max}.` | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function returnOrCall(value){ | ||||||
|  | 	return typeof(value) === 'function' ? value() : value; | ||||||
|  | }  | ||||||
|  |  | ||||||
|  | function processKeys(map, data, partial){ | ||||||
|  | 	let errors = []; | ||||||
|  | 	let out = {}; | ||||||
|  |  | ||||||
|  | 	for(let key of Object.keys(map)){ | ||||||
|  | 		if(!map[key].always && partial && !data.hasOwnProperty(key)) continue; | ||||||
|  |  | ||||||
|  | 		if(!partial && map[key].isRequired && !data.hasOwnProperty(key)){ | ||||||
|  | 			errors.push({key, message:`${key} is required.`}); | ||||||
|  | 			continue; | ||||||
|  | 		}  | ||||||
|  |  | ||||||
|  | 		if(data.hasOwnProperty(key) && map[key].type && typeof(data[key]) !== map[key].type){ | ||||||
|  | 			errors.push({key, message:`${key} is not ${map[key].type} type.`}); | ||||||
|  | 			continue; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		out[key] = data.hasOwnProperty(key) && data[key] !== undefined ? data[key] : returnOrCall(map[key].default); | ||||||
|  |  | ||||||
|  | 		if(data.hasOwnProperty(key) && process_type[map[key].type]){ | ||||||
|  | 			let typeError = process_type[map[key].type](map[key], data[key]); | ||||||
|  | 			if(typeError){ | ||||||
|  | 				errors.push({key, message:`${key} ${typeError}`}); | ||||||
|  | 				 | ||||||
|  | 				continue; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if(errors.length !== 0){ | ||||||
|  | 		throw new ObjectValidateError(errors); | ||||||
|  | 		return {__errors__: errors}; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return out; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function parseFromString(map, data){ | ||||||
|  | 	let types = { | ||||||
|  | 		boolean: function(value){ return value === 'false' ? false : true }, | ||||||
|  | 		number: Number, | ||||||
|  | 		string: String, | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	for(let key of Object.keys(data)){ | ||||||
|  | 		if(map[key] && map[key].type){ | ||||||
|  | 			data[key] = types[map[key].type](data[key]); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return data; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function ObjectValidateError(message) { | ||||||
|  | 	this.name = 'ObjectValidateError'; | ||||||
|  | 	this.message = (message || {}); | ||||||
|  | 	this.status = 422; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ObjectValidateError.prototype = Error.prototype; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | module.exports = {processKeys, parseFromString, ObjectValidateError}; | ||||||
							
								
								
									
										27
									
								
								nodejs/utils/redis.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										27
									
								
								nodejs/utils/redis.js
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const {createClient} = require('redis'); | ||||||
|  | const {promisify} = require('util'); | ||||||
|  |  | ||||||
|  | const config = { | ||||||
|  | 	prefix: 'sso_' | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function client() { | ||||||
|  | 	return createClient(config); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const _client = client(); | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  | 	client: client, | ||||||
|  | 	HGET: promisify(_client.HGET).bind(_client), | ||||||
|  | 	HDEL: promisify(_client.HDEL).bind(_client), | ||||||
|  | 	SADD: promisify(_client.SADD).bind(_client), | ||||||
|  | 	SREM: promisify(_client.SREM).bind(_client), | ||||||
|  | 	DEL: promisify(_client.DEL).bind(_client), | ||||||
|  | 	HSET: promisify(_client.HSET).bind(_client), | ||||||
|  | 	HGETALL: promisify(_client.HGETALL).bind(_client), | ||||||
|  | 	SMEMBERS: promisify(_client.SMEMBERS).bind(_client), | ||||||
|  | 	RENAME: promisify(_client.RENAME).bind(_client), | ||||||
|  | }; | ||||||
							
								
								
									
										190
									
								
								nodejs/utils/redis_model.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								nodejs/utils/redis_model.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,190 @@ | |||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const client = require('../utils/redis'); | ||||||
|  | const objValidate = require('../utils/object_validate'); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | let table = {}; | ||||||
|  |  | ||||||
|  | table.get = async function(data){ | ||||||
|  | 	try{ | ||||||
|  | 		// if the data argument was passed as the index key value, make a data | ||||||
|  | 		// object and add the index key to it. | ||||||
|  | 		if(typeof data !== 'object'){ | ||||||
|  | 			let key = data; | ||||||
|  | 			data = {}; | ||||||
|  | 			data[this._key] = key; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Get all the hash keys for the passed index key. | ||||||
|  | 		let res = await client.HGETALL(`${this._name}_${data[this._key]}`); | ||||||
|  |  | ||||||
|  | 		// If the redis query resolved to something, prepare the data. | ||||||
|  | 		if(res){ | ||||||
|  |  | ||||||
|  | 			// Redis always returns strings, use the keyMap schema to turn them | ||||||
|  | 			// back to native values. | ||||||
|  | 			res = objValidate.parseFromString(this._keyMap, res); | ||||||
|  |  | ||||||
|  | 			// Make sure the index key in in the returned object. | ||||||
|  | 			res[this._key] = data[this._key]; | ||||||
|  |  | ||||||
|  | 			// Create a instance for this redis entry. | ||||||
|  | 			var entry = Object.create(this); | ||||||
|  |  | ||||||
|  | 			// Insert the redis response into the instance. | ||||||
|  | 			Object.assign(entry, res); | ||||||
|  |  | ||||||
|  | 			// Return the instance to the caller. | ||||||
|  | 			return entry; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	let error = new Error('EntryNotFound'); | ||||||
|  | 	error.name = 'EntryNotFound'; | ||||||
|  | 	error.message = `${this._name}:${data[this._key]} does not exists`; | ||||||
|  | 	error.status = 404; | ||||||
|  | 	throw error; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.exists = async function(data){ | ||||||
|  | 	// Return true or false if the requested entry exists ignoring error's. | ||||||
|  | 	try{ | ||||||
|  | 		await this.get(data); | ||||||
|  |  | ||||||
|  | 		return true | ||||||
|  | 	}catch(error){ | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.list = async function(){ | ||||||
|  | 	// return a list of all the index keys for this table. | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		return await client.SMEMBERS(this._name); | ||||||
|  |  | ||||||
|  | 	}catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.listDetail = async function(){ | ||||||
|  | 	// Return a list of the entries as instances. | ||||||
|  | 	let out = []; | ||||||
|  |  | ||||||
|  | 	for(let entry of await this.list()){ | ||||||
|  | 		out.push(await this.get(entry)); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return out | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.add = async function(data){ | ||||||
|  | 	// Add a entry to this redis table. | ||||||
|  | 	try{ | ||||||
|  |  | ||||||
|  | 		// Validate the passed data by the keyMap schema. | ||||||
|  |  | ||||||
|  | 		data = objValidate.processKeys(this._keyMap, data); | ||||||
|  |  | ||||||
|  | 		// Do not allow the caller to overwrite an existing index key, | ||||||
|  | 		if(data[this._key] && await this.exists(data)){ | ||||||
|  | 			let error = new Error('EntryNameUsed'); | ||||||
|  | 			error.name = 'EntryNameUsed'; | ||||||
|  | 			error.message = `${this._name}:${data[this._key]} already exists`; | ||||||
|  | 			error.status = 409; | ||||||
|  |  | ||||||
|  | 			throw error; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Add the key to the members for this redis table | ||||||
|  | 		await client.SADD(this._name, data[this._key]); | ||||||
|  |  | ||||||
|  | 		// Add the values for this entry. | ||||||
|  | 		for(let key of Object.keys(data)){ | ||||||
|  | 			await client.HSET(`${this._name}_${data[this._key]}`, key, data[key]); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// return the created redis entry as entry instance. | ||||||
|  | 		return await this.get(data[this._key]); | ||||||
|  | 	} catch(error){ | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.update = async function(data, key){ | ||||||
|  | 	// Update an existing entry. | ||||||
|  | 	try{ | ||||||
|  | 		// If an index key is passed, we assume is passed, assume we are not | ||||||
|  | 		// part of an entry instance. Make one and recall this from from a entry | ||||||
|  | 		// instance, | ||||||
|  | 		if(key) return await (await this.get(key)).update(data); | ||||||
|  |  | ||||||
|  | 		// Check to see if entry name changed. | ||||||
|  | 		if(data[this._key] && data[this._key] !== this[this._key]){ | ||||||
|  |  | ||||||
|  | 			// Merge the current data into with the updated data  | ||||||
|  | 			let newData = Object.assign({}, this, data); | ||||||
|  |  | ||||||
|  | 			// Remove the updated failed so it doesnt keep it | ||||||
|  | 			delete newData.updated; | ||||||
|  |  | ||||||
|  | 			// Create a new record for the updated entry. If that succeeds, | ||||||
|  | 			// delete the old recored | ||||||
|  | 			if(await this.add(newData)) await this.remove(); | ||||||
|  |  | ||||||
|  | 		}else{ | ||||||
|  | 			// Update what ever fields that where passed. | ||||||
|  |  | ||||||
|  | 			// Validate the passed data, ignoring required fields. | ||||||
|  | 			data = objValidate.processKeys(this._keyMap, data, true); | ||||||
|  | 			 | ||||||
|  | 			// Loop over the data fields and apply them to redis | ||||||
|  | 			for(let key of Object.keys(data)){ | ||||||
|  | 				this[key] = data[key]; | ||||||
|  | 				await client.HSET(`${this._name}_${this[this._key]}`, key, data[key]); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return this; | ||||||
|  | 	 | ||||||
|  | 	} catch(error){ | ||||||
|  | 		// Pass any error to the calling function | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | table.remove = async function(data){ | ||||||
|  | 	// Remove an entry from this table. | ||||||
|  |  | ||||||
|  | 	data = data || this; | ||||||
|  | 	try{ | ||||||
|  | 		// Remove the index key from the tables members list. | ||||||
|  | 		await client.SREM(this._name, data[this._key]); | ||||||
|  |  | ||||||
|  | 		// Remove the entries hash values. | ||||||
|  | 		let count = await client.DEL(`${this._name}_${data[this._key]}`); | ||||||
|  |  | ||||||
|  | 		// Return the number of removed values to the caller. | ||||||
|  | 		return count; | ||||||
|  |  | ||||||
|  | 	} catch(error) { | ||||||
|  | 		throw error; | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | function Table(data){ | ||||||
|  | 	// Create a table instance. | ||||||
|  | 	let instance = Object.create(data); | ||||||
|  | 	Object.assign(instance, table); | ||||||
|  |  | ||||||
|  | 	// Return the table instance to the caller. | ||||||
|  | 	return Object.create(instance); | ||||||
|  |  | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = Table; | ||||||
							
								
								
									
										3
									
								
								nodejs/views/bottom.ejs
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										3
									
								
								nodejs/views/bottom.ejs
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | 		</div> | ||||||
|  |     </body> | ||||||
|  | </html> | ||||||
							
								
								
									
										25
									
								
								nodejs/views/email_templates/reset_link.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								nodejs/views/email_templates/reset_link.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | module.exports = { | ||||||
|  | 	subject: 'Password reset for Theta 42 account', | ||||||
|  | 	message: ` | ||||||
|  | <h2> Theta 42 account</h2> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	Hello {{ user.givenName }}, | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	You have asked to reset the password for user name <b>{{ user.uid }}</b> . Please | ||||||
|  | 	click the link below to complete this request. If this was done in errror, | ||||||
|  | 	please ignore this email. | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	{{ link }} | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | </p> | ||||||
|  | 	Thank you,<br /> | ||||||
|  | 	Theta 42 | ||||||
|  | </p> | ||||||
|  | ` | ||||||
|  | }; | ||||||
							
								
								
									
										24
									
								
								nodejs/views/email_templates/validate_link.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								nodejs/views/email_templates/validate_link.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,24 @@ | |||||||
|  | module.exports = { | ||||||
|  | 	subject: 'Validate email for Theta 42 account', | ||||||
|  | 	message: ` | ||||||
|  | <h2> Theta 42 account</h2> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	Welcome, | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	We need to verify the provided email address in order to continue. Please | ||||||
|  | 	follow the link below to verify this email address: | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	{{ link }} | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | </p> | ||||||
|  | 	Thank you,<br /> | ||||||
|  | 	Theta 42 | ||||||
|  | </p> | ||||||
|  | ` | ||||||
|  | }; | ||||||
							
								
								
									
										34
									
								
								nodejs/views/email_templates/welcome.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								nodejs/views/email_templates/welcome.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | module.exports = { | ||||||
|  | 	subject: 'Welcome to Theta 42!', | ||||||
|  | 	message: ` | ||||||
|  | <p> | ||||||
|  | 	Welcome {{user.givenName}}, | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	Your new Theta 42 Single sign-on account is ready to use. Here is some | ||||||
|  | 	information to get you started. | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	Your username is <b>{{user.uid}}</b> | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	You can manage your account at https://sso.theta42.com | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	You account is ready to be used now, test it by SSHing into the Theta 42 | ||||||
|  | 	jump host \`ssh {{user.uid}}@718it.biz\` | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  | 	The SSO service is still in beta, so please report any bugs you may find! | ||||||
|  | 	You will be notified of new features and services as they become available. | ||||||
|  | </p> | ||||||
|  | 	Thank you,<br /> | ||||||
|  | 	Theta 42 | ||||||
|  | </p> | ||||||
|  | ` | ||||||
|  | }; | ||||||
							
								
								
									
										134
									
								
								nodejs/views/groups.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										134
									
								
								nodejs/views/groups.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,134 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script id="rowTemplate" type="text/html"> | ||||||
|  | 	<p> | ||||||
|  | 	<div class="card shadow"> | ||||||
|  | 		<div class="card-header"> | ||||||
|  | 			<i class="fad fa-users-class"></i> | ||||||
|  | 			Group: {{ cn }} | ||||||
|  | 		</div> | ||||||
|  | 		<div class="card-body"> | ||||||
|  | 			<p> | ||||||
|  | 				{{ description }} | ||||||
|  | 			</p> | ||||||
|  | 			<p> | ||||||
|  | 				<ul class="list-group"> | ||||||
|  | 				{{ #member }} | ||||||
|  | 					<li class="list-group-item shadow"> | ||||||
|  | 						<i class="fad fa-user"></i> {{ uid }} | ||||||
|  | 						<button type="button" action="group/{{groupCN}}/{{uid}}" method="delete" onclick="formAJAX(this)" evalAJAX="tableAJAX(data.message)" class="btn btn-sm btn-danger float-right"> | ||||||
|  | 							<i class="fad fa-user-slash"></i> | ||||||
|  | 						</button> | ||||||
|  | 					</li> | ||||||
|  | 				{{ /member }} | ||||||
|  | 				</ul> | ||||||
|  | 			</p> | ||||||
|  |  | ||||||
|  | 			<button class="btn btn-secondary dropdown-toggle" type="button" id="dropdownMenuButton" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"> | ||||||
|  | 				<i class="fad fa-user-plus"></i> | ||||||
|  | 			</button> | ||||||
|  | 			<div class="dropdown-menu shadow-lg" aria-labelledby="dropdownMenuButton"> | ||||||
|  | 			{{ #toAdd }}{{#.}} | ||||||
|  | 				<a class="dropdown-item" action="group/{{groupCN}}/{{uid}}" method="put" onclick="formAJAX(this)" evalAJAX="tableAJAX(data.message)"> | ||||||
|  | 					<i class="fad fa-user"></i> {{uid}} | ||||||
|  | 				</a> | ||||||
|  | 			{{/.}}{{ /toAdd }} | ||||||
|  | 			 | ||||||
|  | 			</div> | ||||||
|  |  | ||||||
|  | 			<button type="button" onclick="app.group.remove({cn: '{{cn}}'}, function(){tableAJAX('Group {{cn}} deleted.')})" class="btn btn-danger float-right"> | ||||||
|  | 				<i class="fad fa-trash"></i> | ||||||
|  | 			</button> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	</p> | ||||||
|  | </script> | ||||||
|  | <script type="text/javascript"> | ||||||
|  |  | ||||||
|  | 	var userlist; | ||||||
|  |  | ||||||
|  | 	function getUserList(callback){ | ||||||
|  | 		app.user.list(function(error, data){ | ||||||
|  | 			userlist = data.results; | ||||||
|  | 			callback() | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	function tableAJAX(actionMessage){ | ||||||
|  | 		var rowTemplate = $('#rowTemplate').html(); | ||||||
|  | 		var $target = $('#tableAJAX'); | ||||||
|  |  | ||||||
|  | 		$target.html('').hide(); | ||||||
|  | 		app.util.actionMessage('Refreshing user list...', $target); | ||||||
|  |  | ||||||
|  | 		app.group.list(function(error, data){ | ||||||
|  | 			$.each( data.results, function(key, value) { | ||||||
|  |  | ||||||
|  | 				// console.log(value.member) | ||||||
|  |  | ||||||
|  | 				value.toAdd = userlist.map(function(user){ | ||||||
|  | 					if(!value.member.includes(user.dn)) return user; | ||||||
|  | 				}) | ||||||
|  | 				value.member = value.member.map(function(user){ | ||||||
|  | 					return { | ||||||
|  | 						dn: user, | ||||||
|  | 						uid: user.match(/cn=[a-zA-Z0-9\_\-\@\.]+/)[0].replace('cn=', '') | ||||||
|  | 					} | ||||||
|  | 				}) | ||||||
|  | 				value.groupCN = value.cn; | ||||||
|  | 				user_row = Mustache.render(rowTemplate, value); | ||||||
|  | 				$target.append(user_row); | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			$target.fadeIn('slow'); | ||||||
|  |  | ||||||
|  | 			app.util.actionMessage(actionMessage || '', $target, 'info'); | ||||||
|  |  | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 		getUserList(tableAJAX); | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  | <div class="row" style="display:none"> | ||||||
|  | 	<div class="col-md-4"> | ||||||
|  | 		<div class="card shadow-lg"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				<i class="fas fa-layer-plus"></i> | ||||||
|  | 				Add new group | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"></div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<form action="group/" method="post" onsubmit="formAJAX(this)" evalAJAX="tableAJAX('')"> | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">Name</label> | ||||||
|  | 						<input type="text" class="form-control shadow" name="name" placeholder="app_gitea_admin" validate=":3" /> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">Description</label> | ||||||
|  | 						<textarea class="form-control shadow" name="description" placeholder="Admin group for gitea app" validate=":3"></textarea> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<button type="submit" class="btn btn-outline-dark">Add</button> | ||||||
|  | 				</form> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="col-md-8"> | ||||||
|  | 		<div class="card shadow-lg"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				<i class="fad fa-users-class"></i> | ||||||
|  | 				Group list | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<div class="" id="tableAJAX"> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										321
									
								
								nodejs/views/home.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										321
									
								
								nodejs/views/home.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,321 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script id="profileTemplate" type="text/html"> | ||||||
|  | 	<h2><i>User Name:</i> <b>{{uid}}</b></h2> | ||||||
|  | 	<i>Name:</i> <b>{{givenName}} {{sn}}</b><br /> | ||||||
|  | 	<i>Email:</i> <b>{{mail}} </b><br /> | ||||||
|  | 	<i>Joined:</i> <b>{{createTimestamp}} </b>, <i>Edited:</i> <b>{{modifyTimestamp}} </b><br /> | ||||||
|  | 	<i>Phone:</i> <b>{{mobile}} </b><br /> | ||||||
|  | 	<i>LDAP DN:</i> <b>{{dn}} </b><br /> | ||||||
|  | 	<i>Home Directory:</i> <b>{{homeDirectory}} </b><br /> | ||||||
|  | 	<i>Login Shell:</i> <b>{{loginShell}} </b><br /> | ||||||
|  | 	<i>SSH Public Key:</i> <b>{{sshPublicKey}}</b><br /> | ||||||
|  | 	<i>Unix User ID:</i> <b>{{uidNumber}} </b><br /> | ||||||
|  | 	<i>Unix Group ID:</i> <b>{{gidNumber}} </b><br /> | ||||||
|  | 	<i>Description:</i><br> | ||||||
|  | 	<p> | ||||||
|  | 		{{description}} | ||||||
|  | 	</p> | ||||||
|  | 	<img id="profile_photo" > | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <script id="groupRowTemplate" type="text/html"> | ||||||
|  | 	<tr> | ||||||
|  | 		<td>{{cn}}</td> | ||||||
|  | 		<td>{{description}}</td> | ||||||
|  | 	</tr> | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <script id="userEditTemplate" type="text/html"> | ||||||
|  | 	<h3>Editing {{uid}}</h3> | ||||||
|  | 	<form action="user/{{uid}}" method="put" onsubmit="formAJAX(this)" evalAJAX="editUserSeccess()"> | ||||||
|  | 		<div class="form-group"> | ||||||
|  | 			<label class="control-label">SSH Public Key</label> | ||||||
|  | 			<input type="text" class="form-control" name="sshPublicKey" placeholder="ssh-rsa AAAAB3NzaC1yc2EAAAADAQ..." value="{{sshPublicKey}}" /> | ||||||
|  | 		</div> | ||||||
|  |  | ||||||
|  | 		<div class="form-group"> | ||||||
|  | 			<label class="control-label">Mobile Phone</label> | ||||||
|  | 			<input type="text" class="form-control" name="mobile" placeholder="9175551234" validate=":9" value="{{mobile}}" /> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="form-group"> | ||||||
|  | 			<label class="control-label">User Description (Optional)</label> | ||||||
|  | 			<textarea class="form-control" name="description" placeholder="Admin group for gitea app">{{description}}</textarea> | ||||||
|  | 		</div> | ||||||
|  | 		<button type="submit" class="btn btn-outline-dark btn-warning">Change</button> | ||||||
|  |  | ||||||
|  | 	</form> | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <script id="paswordResetTemplate" type="text/html"> | ||||||
|  | 	<h3> | ||||||
|  | 		Reset Password for {{uid}} | ||||||
|  | 	</h3> | ||||||
|  | 	<form action="user/{{uid}}/password" method="put" onsubmit="formAJAX(this)" class="form-group"> | ||||||
|  | 		<div class="form-group"> | ||||||
|  | 			<label class="control-label">Password</label> | ||||||
|  | 			<div class="input-group mb-3 shadow"> | ||||||
|  | 				<div class="input-group-prepend"> | ||||||
|  | 					<span class="input-group-text" ><i class="fad fa-key"></i></span> | ||||||
|  | 				</div> | ||||||
|  | 				<input type="password" name="userPassword" class="form-control" placeholder="hunter123!" validate=":3" /> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="form-group"> | ||||||
|  | 			<label class="control-label">Again</label> | ||||||
|  | 			<div class="input-group mb-3 shadow"> | ||||||
|  | 				<div class="input-group-prepend"> | ||||||
|  | 					<span class="input-group-text" ><i class="fad fa-key"></i></span> | ||||||
|  | 				</div> | ||||||
|  | 				<input type="password" name="password" class="form-control" placeholder="hunter123!" validate="eq:userPassword" /> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<button type="submit" class="btn btn-outline-secondary shadow">Change</button> | ||||||
|  | 	</form> | ||||||
|  |  | ||||||
|  | </script> | ||||||
|  | <script type="text/javascript"> | ||||||
|  | 	var currentUser; | ||||||
|  |  | ||||||
|  | 	function getInvite(){ | ||||||
|  | 		app.user.createInvite(function(error, data){ | ||||||
|  | 			$('#invite_token').html(location.origin+"/login/invite/"+data.token); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function renderProfile(user){ | ||||||
|  | 		var profileTemplate = $('#profileTemplate').html(); | ||||||
|  | 		var paswordResetTemplate = $('#paswordResetTemplate').html();  | ||||||
|  | 		var $target = $('#userProfile div.card-body'); | ||||||
|  |  | ||||||
|  | 		// data.photo = unescape(encodeURIComponent(data.jpegPhoto)); | ||||||
|  | 		user.createTimestamp = moment(user.createTimestamp, "YYYYMMDDHHmmssZ").fromNow(); | ||||||
|  | 		user.modifyTimestamp = moment(user.modifyTimestamp, "YYYYMMDDHHmmssZ").fromNow(); | ||||||
|  |  | ||||||
|  | 		$target.html(Mustache.render(profileTemplate, user)); | ||||||
|  |  | ||||||
|  | 		$('#passwordReset').html(Mustache.render(paswordResetTemplate, user)) | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	function renderUserGroups(user){ | ||||||
|  | 		app.api.get('group/?detail=true&member='+user.uid, function(error, data){ | ||||||
|  | 			var groupRowTemplate = $('#groupRowTemplate').html(); | ||||||
|  | 			var $target = $('#mygroups'); | ||||||
|  |  | ||||||
|  | 			$target.html('') | ||||||
|  | 			if(error){ | ||||||
|  | 				app.utils.actionMessage(data.message, $target, 'danger'); | ||||||
|  | 			}else{ | ||||||
|  | 				for(var group of data.results){ | ||||||
|  | 					$target.append(Mustache.render(groupRowTemplate, group)); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function determinUser(callback){ | ||||||
|  | 		if(location.pathname.includes('/users/')){ | ||||||
|  | 			var uid = location.pathname.replace('/users/', ''); | ||||||
|  |  | ||||||
|  | 			app.api.get('user/'+uid, function(err, res){ | ||||||
|  | 				callback(res.results) | ||||||
|  | 			}) | ||||||
|  | 		}else{ | ||||||
|  | 			callback(app.auth.user) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function editUser(){ | ||||||
|  | 		determinUser(function(user){ | ||||||
|  | 			var $profileCard = $('#userProfile'); | ||||||
|  | 			var $editCard = $('#editProfile'); | ||||||
|  | 			var userEditTemplate = $('#userEditTemplate').html() | ||||||
|  |  | ||||||
|  | 			$editCard.find('.card-body').html(Mustache.render(userEditTemplate, user)) | ||||||
|  |  | ||||||
|  | 			$profileCard.slideUp(); | ||||||
|  | 			$editCard.slideDown(); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	function editUserSeccess(){ | ||||||
|  | 		console.log('edit done') | ||||||
|  | 		$('#editProfile').slideUp(); | ||||||
|  | 		determinUser(function(user){ | ||||||
|  | 			currentUser = user; | ||||||
|  | 			app.auth.user = user; | ||||||
|  | 			renderProfile(user); | ||||||
|  | 			renderUserGroups(user); | ||||||
|  | 			$('#userProfile').slideDown(); | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  |  | ||||||
|  | 		determinUser(function(user){ | ||||||
|  | 			currentUser = user; | ||||||
|  | 			renderProfile(user); | ||||||
|  | 			renderUserGroups(user); | ||||||
|  | 			$('#username').text(user.uid); | ||||||
|  | 		}); | ||||||
|  |  | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  | <style type="text/css"> | ||||||
|  | 	.services-list i{ | ||||||
|  | 		padding-right: .5em; | ||||||
|  | 	} | ||||||
|  | </style> | ||||||
|  | <div class="row" style="display:none"> | ||||||
|  | 	<div class="col-md-4"> | ||||||
|  | 		<div class="shadow-lg card mb-3 card-default"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				<i class="fas fa-user-plus"></i> | ||||||
|  | 				Invite User | ||||||
|  | 				<div class="float-right"> | ||||||
|  | 					<i class="far fa-arrows-v"></i> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  |  | ||||||
|  | 				<button onclick="getInvite(this)" class="btn btn-outline-dark shadow">New Invite Token</button> | ||||||
|  | 				<div> | ||||||
|  | 					<b id="invite_token"></b> | ||||||
|  | 				</div> | ||||||
|  |  | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="shadow-lg card mb-3 card-default"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 					<div class="float-right"> | ||||||
|  | 						<i class="far fa-arrows-v"></i> | ||||||
|  | 					</div> | ||||||
|  | 					<i class="fad fa-th-list"></i> | ||||||
|  | 					Services | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<ul class="list-group text-dark services-list"> | ||||||
|  | 					<li class="list-group-item text-dark"> | ||||||
|  | 							<i class="fad fa-terminal"></i> | ||||||
|  | 							SSH <b id="username"></b>@718it.biz:22 | ||||||
|  | 					</li> | ||||||
|  | 					<a href="https://emby.718it.biz/" target="_blank" class="text-dark"> | ||||||
|  | 						<li class="list-group-item text-dark"> | ||||||
|  | 								<i class="fad fa-film"></i> | ||||||
|  | 								Emby | ||||||
|  | 						</li> | ||||||
|  | 					</a> | ||||||
|  | 					<a href="https://git.theta42.com" target="_blank" class="text-dark"> | ||||||
|  | 						<li class="list-group-item text-dark"> | ||||||
|  | 								<i class="fab fa-git"></i> | ||||||
|  | 								Git server | ||||||
|  | 						</li> | ||||||
|  | 					</a> | ||||||
|  | 					<a href="https://rdp.vm42.us" target="_blank" class="text-dark"> | ||||||
|  | 						<li class="list-group-item text-dark"> | ||||||
|  | 								<i class="fad fa-desktop"></i> | ||||||
|  | 								Virtual Desk Top | ||||||
|  | 						</li> | ||||||
|  | 					</a> | ||||||
|  | 					<a href="https://pve.admin.vm42.us" target="_blank" class="text-dark"> | ||||||
|  | 						<li class="list-group-item text-dark"> | ||||||
|  | 								<i class="fad fa-server"></i> | ||||||
|  | 								Promox (contact wmanlty for access) | ||||||
|  | 						</li> | ||||||
|  | 					</a> | ||||||
|  | 				</ul> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="shadow-lg card"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				<i class="fad fa-undo-alt"></i> | ||||||
|  | 				Password Reset | ||||||
|  | 				<div class="float-right"> | ||||||
|  | 					<i class="far fa-arrows-v"></i> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div id="passwordReset" class="card-body"> | ||||||
|  |  | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="col-md-8"> | ||||||
|  | 		<div id="userProfile" class="shadow-lg card card-default mb-8"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				<i class="fad fa-id-card"></i> | ||||||
|  | 				Profile | ||||||
|  | 				<div class="float-right"> | ||||||
|  | 					<i class="far fa-arrows-v"></i> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<!-- User profile inserted by JS --> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-footer"> | ||||||
|  | 				<button type="button" onclick="editUser()" class="btn btn-warning btn-lg shadow "> | ||||||
|  | 					<i class="fad fa-user-edit"></i> | ||||||
|  | 				</button> | ||||||
|  | 				<button type="button" onclick="app.user.remove({uid: currentUser.uid}, function(){app.util.actionMessage('username {{uid}} delete.', $(this), 'danger')})" class="shadow btn btn-danger btn-lg float-right"> | ||||||
|  | 					<i class="fad fa-user-slash"></i> | ||||||
|  | 				</button> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  |  | ||||||
|  | 		<div id="editProfile" class="shadow-lg card card-default mb-8" style="display:none"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				<i class="fad fa-id-card"></i> | ||||||
|  | 				Edit Profile | ||||||
|  | 				<div class="float-right"> | ||||||
|  | 					<i class="far fa-arrows-v"></i> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<div id="tableAJAX"> | ||||||
|  |  | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  |  | ||||||
|  | 		<br /> | ||||||
|  | 		<div class="shadow-lg card card-default mb-8"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				<i class="fad fa-id-card"></i> | ||||||
|  | 				My groups | ||||||
|  | 				<div class="float-right"> | ||||||
|  | 					<i class="far fa-arrows-v"></i> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body" style="padding-bottom:0"> | ||||||
|  | 				<div class="table-responsive"> | ||||||
|  | 					<table class="table"> | ||||||
|  | 						<thead> | ||||||
|  | 							<th> | ||||||
|  | 								Name | ||||||
|  | 							</th> | ||||||
|  | 							<th> | ||||||
|  | 								Description | ||||||
|  | 							</th> | ||||||
|  | 						</thead> | ||||||
|  | 						<tbody id="mygroups"> | ||||||
|  | 							 | ||||||
|  | 						</tbody> | ||||||
|  | 					</table> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<br /> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										2
									
								
								nodejs/views/index.ejs
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								nodejs/views/index.ejs
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | <% include top %> | ||||||
|  | <% include bottom %> | ||||||
							
								
								
									
										39
									
								
								nodejs/views/invite.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								nodejs/views/invite.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script type="text/javascript"> | ||||||
|  | 	function tableAJAX(message){ | ||||||
|  | 		app.util.actionMessage(message); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 		$('form').attr('action', 'auth/invite/<%= invite.token %>/<%= invite.mail_token %>').attr('evalAJAX', 'location.replace("/login");') | ||||||
|  | 		$('[name="mail"').val('<%= invite.mail %>').prop("disabled", true); | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style type="text/css"> | ||||||
|  | 	div.form-group:hover { | ||||||
|  | 	    -ms-transform: scale(1.02); | ||||||
|  | 	    -webkit-transform: scale(1.02); | ||||||
|  | 	    transform: scale(1.02); | ||||||
|  | 	} | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | <div class="row" style="display:none"> | ||||||
|  | 		<div class="col-md-12"> | ||||||
|  | 		<div class="card shadow-lg"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				Add new user | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<p> | ||||||
|  | 					Invited By: <b><%= invite.created_by %></b>, <%= invite.created_on %> | ||||||
|  | 				</p> | ||||||
|  | 				<%- include('user_form') %> | ||||||
|  |  | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										59
									
								
								nodejs/views/invite_email.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								nodejs/views/invite_email.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script type="text/javascript"> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	var emailSent = function(){ | ||||||
|  | 		$('#email_card .card-body').html("<h1>Thank you!</h1><p>Check your mail</p>") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <style type="text/css"> | ||||||
|  | 	div.form-group:hover { | ||||||
|  | 		-ms-transform: scale(1.02); | ||||||
|  | 		-webkit-transform: scale(1.02); | ||||||
|  | 		transform: scale(1.02); | ||||||
|  | 	} | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | <div class="row"> | ||||||
|  | 	<div id="email_card" class="card-deck"> | ||||||
|  | 		<div class="shadow-lg card mb-3"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				Validate Email | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<p> | ||||||
|  | 					Invited By: <b><%= invite.created_by %></b>, <%= invite.created_on %>. | ||||||
|  | 				</p> | ||||||
|  | 				<p> | ||||||
|  | 					Please enter a valid email address. A link will be sent to | ||||||
|  | 					the supplied address to complete the registration process. | ||||||
|  | 				</p> | ||||||
|  | 				<p> | ||||||
|  | 					The supplied email will also be used as the linked email for | ||||||
|  | 					the new user. | ||||||
|  | 				</p> | ||||||
|  | 				<form action="auth/invite/<%= invite.token %>" onsubmit="formAJAX(this)" evalAJAX="emailSent()"> | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">Email</label> | ||||||
|  | 						<div class="input-group mb-3"> | ||||||
|  | 						  <div class="input-group-prepend"> | ||||||
|  | 							<span class="input-group-text" ><i class="fad fa-at"></i></span> | ||||||
|  | 						  </div> | ||||||
|  | 							<input type="email" name="mail" class="form-control" placeholder="jsmith@gmail.com" validate="email:3" /> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<button type="submit" class="btn btn-outline-dark"><i class="fad fa-paper-plane"></i> Send It!</button> | ||||||
|  | 				</form> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										100
									
								
								nodejs/views/login.ejs
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										100
									
								
								nodejs/views/login.ejs
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,100 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script type="text/javascript"> | ||||||
|  |  | ||||||
|  | 	app.auth.isLoggedIn(function(error, isLoggedIn){ | ||||||
|  | 		if(isLoggedIn){ | ||||||
|  | 			window.location.href = app.util.getUrlParameter('redirect') || '/'; | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  |  | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  |  | ||||||
|  | <div class="row"> | ||||||
|  | 	<div class="card-deck"> | ||||||
|  | 		<div class="shadow-lg card mb-3"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				Password Log in | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<form action="auth/login" onsubmit="formAJAX(this)" evalAJAX="app.auth.setToken(data.token);window.location.href = app.util.getUrlParameter('redirect') || '/';"> | ||||||
|  | 					<input type="hidden" name="redirect" value="<%= redirect %>"> | ||||||
|  |  | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">User name</label> | ||||||
|  | 						<div class="input-group mb-3 shadow"> | ||||||
|  | 						  <div class="input-group-prepend"> | ||||||
|  | 						    <span class="input-group-text" ><i class="fad fa-user"></i></span> | ||||||
|  | 						  </div> | ||||||
|  | 							<input type="text" name="uid" class="form-control" placeholder="jsmith" validate="user:3" /> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">Password</label> | ||||||
|  | 						<div class="input-group mb-3 shadow"> | ||||||
|  | 						  <div class="input-group-prepend"> | ||||||
|  | 						    <span class="input-group-text" ><i class="fad fa-key"></i></span> | ||||||
|  | 						  </div> | ||||||
|  | 							<input type="password" name="password" class="form-control" placeholder="hunter123!" validate=":3" /> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<button type="submit" class="btn btn-outline-dark"><i class="fad fa-sign-in"></i> Log in</button> | ||||||
|  | 				</form> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="shadow-lg card border-danger mb-3"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				Social Login | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<h3>Coming soon!</h3> | ||||||
|  | 				<p> | ||||||
|  | 					<ul class="list-group"> | ||||||
|  | 						<li class="list-group-item"><i class="fab fa-google"></i> Login with google OATH</li> | ||||||
|  | 						<li class="list-group-item"><i class="fab fa-github"></i> Login with github OATH</li> | ||||||
|  | 						<li class="list-group-item"><i class="fab fa-facebook"></i> Login with facebook OATH</li> | ||||||
|  | 					</ul> | ||||||
|  | 				</p> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="shadow-lg card mb-3"> | ||||||
|  | 			<div class="card-header shadow"> | ||||||
|  | 				Password Reset | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<p> | ||||||
|  | 					Forgot your password? Or your user name? No problem! Just | ||||||
|  | 					enter you email address below and if you are in our system, | ||||||
|  | 					we will email with the required information to get back up | ||||||
|  | 					and running! | ||||||
|  | 				</p> | ||||||
|  | 				<form action="auth/resetpassword" onsubmit="formAJAX(this)"> | ||||||
|  | 					<input type="hidden" name="redirect" value="<%= redirect %>"> | ||||||
|  |  | ||||||
|  | 					<div class="form-group"> | ||||||
|  | 						<label class="control-label">Email</label> | ||||||
|  | 						<div class="input-group mb-3 shadow"> | ||||||
|  | 						  <div class="input-group-prepend"> | ||||||
|  | 						    <span class="input-group-text" ><i class="fad fa-at"></i></span> | ||||||
|  | 						  </div> | ||||||
|  | 							<input type="email" name="mail" class="form-control" placeholder="jsmith@gmail.com" validate="email:3" /> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<button type="submit" class="btn btn-outline-dark"><i class="fad fa-question"></i> Help me!</button> | ||||||
|  | 				</form> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										43
									
								
								nodejs/views/reset_password.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								nodejs/views/reset_password.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script type="text/javascript"> | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  |  | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  | <div class="row" style="display:none"> | ||||||
|  | 	<div class="col-md-12"> | ||||||
|  | 		<div class="card mb-3 shadow-lg"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				Password reset | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header shadow actionMessage" style="display:none"> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<p> | ||||||
|  | 					<form action="auth/resetpassword/<%= token.token%>" method="post" onsubmit="formAJAX(this)"> | ||||||
|  | 						<div class="form-group"> | ||||||
|  | 							<label class="control-label">Password</label> | ||||||
|  | 							<div class="input-group mb-3 shadow"> | ||||||
|  | 							  <div class="input-group-prepend"> | ||||||
|  | 							    <span class="input-group-text" ><i class="fad fa-key"></i></span> | ||||||
|  | 							  </div> | ||||||
|  | 								<input type="password" name="userPassword" class="form-control" placeholder="hunter123!" validate=":3" /> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="form-group"> | ||||||
|  | 							<label class="control-label">Again</label> | ||||||
|  | 							<div class="input-group mb-3 shadow"> | ||||||
|  | 							  <div class="input-group-prepend"> | ||||||
|  | 							    <span class="input-group-text" ><i class="fad fa-key"></i></span> | ||||||
|  | 							  </div> | ||||||
|  | 								<input type="password" name="password" class="form-control" placeholder="hunter123!" validate="eq:userPassword" /> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<button type="submit" class="btn btn-outline-dark">Change</button> | ||||||
|  | 					</form> | ||||||
|  | 				</p> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
							
								
								
									
										42
									
								
								nodejs/views/top.ejs
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										42
									
								
								nodejs/views/top.ejs
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | <!doctype html> | ||||||
|  | <html lang="en"> | ||||||
|  |   <head> | ||||||
|  |     <meta charset="utf-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"> | ||||||
|  | 		<title>SSO Manager - Theta 42</title> | ||||||
|  | 		<!-- CSS are placed here --> | ||||||
|  | 		<!-- <link rel='stylesheet' href='/static/css/bootstrap.min.css' /> --> | ||||||
|  | 		<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.4.1/css/bootstrap.min.css" integrity="sha384-Vkoo8x4CGsO3+Hhxv8T/Q5PaXtkKtu6ug5TOeNV6gBiFeWPGFN9MuhOf23Q9Ifjh" crossorigin="anonymous"> | ||||||
|  | 		<link rel='stylesheet' href='/static/css/styles.css' /> | ||||||
|  | 		<!-- Scripts are placed here --> | ||||||
|  | 		<script src="https://code.jquery.com/jquery-3.5.0.min.js"></script> | ||||||
|  | 		<!-- <script type="text/javascript" src='/static/js/jquery.min.js'></script> --> | ||||||
|  | 		<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/popper.js@1.16.0/dist/umd/popper.min.js"></script> | ||||||
|  | 		<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.4.1/js/bootstrap.min.js" integrity="sha384-wfSDF2E50Y2D1uUdj0O3uMBJnjuUD4Ih7YwaYd1iqfktj0Uod8GCExl3Og8ifwB6" crossorigin="anonymous"></script> | ||||||
|  | 		<!-- <script type="text/javascript" src='/static/js/bootstrap.min.js'></script> --> | ||||||
|  | 		<script src="https://kit.fontawesome.com/4625ee80a2.js" crossorigin="anonymous"></script> | ||||||
|  | 		<script type="text/javascript" src='/static/js/mustache.min.js'></script> | ||||||
|  | 		<script type="text/javascript" src="/static/js/app.js"></script> | ||||||
|  | 		<script type="text/javascript" src="/static/js/val.js"></script> | ||||||
|  | 		<script type="text/javascript" src="/static/js/moment.js"></script> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		<!-- HTML5 shim, for IE6-8 support of HTML5 elements --> | ||||||
|  | 		<!--[if lt IE 9]> | ||||||
|  | 		<script src="http://html5shim.googlecode.com/svn/trunk/html5.js"></script> | ||||||
|  | 		<![endif]--> | ||||||
|  | 	</head> | ||||||
|  | 	<body> | ||||||
|  | 		<header class="shadow d-flex flex-column flex-md-row align-items-center p-3 px-md-4 mb-3 bg-white border-bottom shadow-sm"> | ||||||
|  | 			<h5 class="hover-effect my-0 mr-md-auto font-weight-normal">SSO Manager - Theta 42</h5> | ||||||
|  | 			<nav class="my-2 my-md-0 mr-md-3"> | ||||||
|  | 				<a class="text-dark hover-effect" href="/"><i class="fad fa-tachometer-alt-fastest"></i> Home</a> | ||||||
|  | 				<a class="text-dark hover-effect" href="/users"><i class="fad fa-users"></i> Users</a> | ||||||
|  | 				<a class="text-dark hover-effect" href="/groups"><i class="fad fa-users-class"></i> Groups</a> | ||||||
|  | 			</nav> | ||||||
|  | 			<a class="hover-effect btn btn-outline-primary" onclick="app.auth.logOut(e => window.location.href='/')"><i class="fas fa-sign-out"></i> Log Out</a> | ||||||
|  |  | ||||||
|  | 		</header> | ||||||
|  |  | ||||||
|  | 		<!-- Container --> | ||||||
|  | 		<div class="container"> | ||||||
							
								
								
									
										41
									
								
								nodejs/views/user_form.ejs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								nodejs/views/user_form.ejs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | <form action="user/" method="post" onsubmit="formAJAX(this)"> | ||||||
|  | 	<input type="hidden" class="form-control" name="delete" value="false" /> | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">First name</label> | ||||||
|  | 		<input type="text" class="form-control shadow" name="givenName" placeholder="John" validate=":3" /> | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">Last name</label> | ||||||
|  | 		<input type="text" class="form-control shadow" name="sn" placeholder="smith" validate=":3" /> | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">Email</label> | ||||||
|  | 		<input type="text" class="form-control shadow" name="mail" placeholder="jsmith@gmail.com" validate="email:3" /> | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">SSH Public Key</label> | ||||||
|  | 		<input type="text" class="form-control shadow" name="sshPublicKey" placeholder="ssh-rsa AAAAB3NzaC1yc2EAAAADAQ..." /> | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">Mobile Phone</label> | ||||||
|  | 		<input type="text" class="form-control shadow" name="mobile" placeholder="9175551234" validate=":9" /> | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">Password</label> | ||||||
|  | 		<input type="password" class="form-control shadow" name="userPassword" placeholder="Atleast 5 char. long" validate="password:5"/> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">Again</label> | ||||||
|  | 		<input type="password" class="form-control shadow" name="passwordMatch" placeholder="Retype password" validate="eq:userPassword"/> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="form-group"> | ||||||
|  | 		<label class="control-label">User Description (Optional)</label> | ||||||
|  | 		<textarea class="form-control shadow" name="description" placeholder="Admin group for gitea app"></textarea> | ||||||
|  | 	</div> | ||||||
|  | 	<button type="submit" class="btn btn-outline-dark">Add</button> | ||||||
|  | </form> | ||||||
							
								
								
									
										94
									
								
								nodejs/views/users.ejs
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										94
									
								
								nodejs/views/users.ejs
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,94 @@ | |||||||
|  | <%- include('top') %> | ||||||
|  | <script id="rowTemplate" type="text/html"> | ||||||
|  | 	<tr class="" action="user/password/{{ username }}" method="put" evalAJAX="$form.trigger('reset')"> | ||||||
|  | 		<td> | ||||||
|  | 			{{ uidNumber }} | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			<a href='/users/{{uid}}'>{{ uid }}</a>	 | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			{{givenName}} {{sn}}  | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			{{mail}} | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			{{#sudoUser}}<i class="fad fa-check-square"></i>{{/sudoUser}} | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			{{#sshPublicKey}}<i class="fad fa-check-square"></i>{{/sshPublicKey}} | ||||||
|  | 		</td> | ||||||
|  | 		<td> | ||||||
|  | 			<button type="button" onclick="app.user.remove({uid: '{{uid}}'}, function(){renderUsers('username {{uid}} delete.')})" class="btn btn-sm btn-danger"> | ||||||
|  | 				<i class="fad fa-user-slash"></i> | ||||||
|  | 			</button> | ||||||
|  | 		</td> | ||||||
|  | 	</tr> | ||||||
|  | </script> | ||||||
|  | <script type="text/javascript"> | ||||||
|  |  | ||||||
|  | 	function renderUsers(actionMessage){ | ||||||
|  | 		var rowTemplate = $('#rowTemplate').html(); | ||||||
|  | 		var $target = $('#tableAJAX'); | ||||||
|  |  | ||||||
|  | 		$target.html('').hide(); | ||||||
|  | 		app.util.actionMessage('Refreshing user list...', $target); | ||||||
|  |  | ||||||
|  | 		app.user.list(function(error, data){ | ||||||
|  | 			$.each( data.results, function(key, value) { | ||||||
|  | 				if(value.uidNumber<1500) return; | ||||||
|  | 				user_row = Mustache.render(rowTemplate, value); | ||||||
|  | 				$target.append(user_row); | ||||||
|  | 			}); | ||||||
|  |  | ||||||
|  | 			$target.fadeIn('slow'); | ||||||
|  |  | ||||||
|  | 			app.util.actionMessage(actionMessage || '', $target, 'info'); | ||||||
|  |  | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	$(document).ready(function(){ | ||||||
|  | 		renderUsers(); //populate the table | ||||||
|  | 		$('form[action="user/"]').attr('evalAJAX', 'renderUsers()') | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  | <div class="row" style="display:none"> | ||||||
|  | 	<div class="col-md-4"> | ||||||
|  | 		<div class="card shadow-lg"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				<i class="fas fa-user-plus"></i> | ||||||
|  | 				Add new user | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"></div> | ||||||
|  | 			<div class="card-body"> | ||||||
|  | 				<%- include('user_form') %> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="col-md-8"> | ||||||
|  | 		<div class="card shadow"> | ||||||
|  | 			<div class="card-header"> | ||||||
|  | 				<i class="fad fa-th-list"></i> | ||||||
|  | 				User List | ||||||
|  | 			</div> | ||||||
|  | 			<div class="card-header actionMessage" style="display:none"></div> | ||||||
|  | 			<table class="card-body table table-striped" style="margin-bottom:0"> | ||||||
|  | 				<thead> | ||||||
|  | 					<th>ID</th> | ||||||
|  | 					<th>User Name</th> | ||||||
|  | 					<th>Name</th> | ||||||
|  | 					<th>eMail</th> | ||||||
|  | 					<th>Sudo</th> | ||||||
|  | 					<th>Key</th> | ||||||
|  | 					<th></th> | ||||||
|  | 				</thead> | ||||||
|  | 				<tbody id="tableAJAX"> | ||||||
|  | 					<!-- ajax loaded table --> | ||||||
|  | 				</tbody> | ||||||
|  | 			</table> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <%- include('bottom') %> | ||||||
| @ -1 +1 @@ | |||||||
| package 'libpam0g-dev' | # package 'libpam0g-dev' | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ run_list( | |||||||
| 	"recipe[t42-common::python]", | 	"recipe[t42-common::python]", | ||||||
| 	"recipe[t42-common::nodejs]", | 	"recipe[t42-common::nodejs]", | ||||||
| 	# "recipe[t42-common::apache]", | 	# "recipe[t42-common::apache]", | ||||||
| 	"recipe[t42-common::openresty]", | 	# "recipe[t42-common::openresty]", | ||||||
| 	# "recipe[t42-common::php]", | 	# "recipe[t42-common::php]", | ||||||
| 	# "recipe[t42-common::mysql]", | 	# "recipe[t42-common::mysql]", | ||||||
| ) | ) | ||||||
		Reference in New Issue
	
	Block a user