rebased
This commit is contained in:
commit
f7cee0239e
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]",
|
||||||
)
|
)
|
Loading…
x
Reference in New Issue
Block a user