diff --git a/migrations/6.sql b/migrations/6.sql new file mode 100644 index 0000000..f8ec76a --- /dev/null +++ b/migrations/6.sql @@ -0,0 +1 @@ +ALTER TABLE `configs` ADD COLUMN `provider` varchar(64) NOT NULL AFTER `name`; \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index e9b6962..da946a4 100644 --- a/package-lock.json +++ b/package-lock.json @@ -783,6 +783,26 @@ "toidentifier": "1.0.0" } }, + "i18n": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/i18n/-/i18n-0.9.0.tgz", + "integrity": "sha512-/Sie5k0xz8VoB0dilo0OPJic5gukT6j+/24aprosU700iWgHcyP5GqDwgNuNscwK2YkGHRE3fO/wn5d2Nc7dBw==", + "requires": { + "debug": "*", + "make-plural": "^6.0.1", + "math-interval-parser": "^2.0.1", + "messageformat": "^2.3.0", + "mustache": "*", + "sprintf-js": "^1.1.2" + }, + "dependencies": { + "sprintf-js": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", + "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==" + } + } + }, "iconv-lite": { "version": "0.4.24", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", @@ -1005,6 +1025,16 @@ "yallist": "^3.0.2" } }, + "make-plural": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/make-plural/-/make-plural-6.2.1.tgz", + "integrity": "sha512-AmkruwJ9EjvyTv6AM8MBMK3TAeOJvhgTv5YQXzF0EP2qawhpvMjDpHvsdOIIT0Vn+BB0+IogmYZ1z+Ulm/m0Fg==" + }, + "math-interval-parser": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/math-interval-parser/-/math-interval-parser-2.0.1.tgz", + "integrity": "sha512-VmlAmb0UJwlvMyx8iPhXUDnVW1F9IrGEd9CIOmv+XL8AErCUUuozoDMrgImvnYt2A+53qVX/tPW6YJurMKYsvA==" + }, "media-typer": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", @@ -1015,6 +1045,36 @@ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" }, + "messageformat": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/messageformat/-/messageformat-2.3.0.tgz", + "integrity": "sha512-uTzvsv0lTeQxYI2y1NPa1lItL5VRI8Gb93Y2K2ue5gBPyrbJxfDi/EYWxh2PKv5yO42AJeeqblS9MJSh/IEk4w==", + "requires": { + "make-plural": "^4.3.0", + "messageformat-formatters": "^2.0.1", + "messageformat-parser": "^4.1.2" + }, + "dependencies": { + "make-plural": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/make-plural/-/make-plural-4.3.0.tgz", + "integrity": "sha512-xTYd4JVHpSCW+aqDof6w/MebaMVNTVYBZhbB/vi513xXdiPT92JMVCo0Jq8W2UZnzYRFeVbQiQ+I25l13JuKvA==", + "requires": { + "minimist": "^1.2.0" + } + } + } + }, + "messageformat-formatters": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/messageformat-formatters/-/messageformat-formatters-2.0.1.tgz", + "integrity": "sha512-E/lQRXhtHwGuiQjI7qxkLp8AHbMD5r2217XNe/SREbBlSawe0lOqsFb7rflZJmlQFSULNLIqlcjjsCPlB3m3Mg==" + }, + "messageformat-parser": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/messageformat-parser/-/messageformat-parser-4.1.3.tgz", + "integrity": "sha512-2fU3XDCanRqeOCkn7R5zW5VQHWf+T3hH65SzuqRvjatBK7r4uyFa5mEX+k6F9Bd04LVM5G4/BHBTUJsOdW7uyg==" + }, "methods": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", diff --git a/package.json b/package.json index ded5578..e94a0fc 100644 --- a/package.json +++ b/package.json @@ -12,8 +12,9 @@ "license": "ISC", "dependencies": { "express": "^4.17.1", - "moment-timezone": "^0.5.28", "express-session": "^1.17.1", + "i18n": "^0.9.0", + "moment-timezone": "^0.5.28", "multer": "^1.4.2", "mustache": "^4.0.1", "mustache-express": "^1.3.0", diff --git a/src/config.example.json b/src/config.example.json index a0e6153..611e8f9 100644 --- a/src/config.example.json +++ b/src/config.example.json @@ -4,8 +4,11 @@ "port": 9991, "locale": "en", "style": "dark", - "logging": true, "secret": "-/!sup3rr4nd0m70p53cr3t70k3ny0u5h0u1dch4ng3!/-", + "logging": { + "enabled": true, + "max_size": 5 + }, "db": { "host": "127.0.0.1", "port": 3306, diff --git a/src/index.js b/src/index.js index 4c422e1..66db5a7 100644 --- a/src/index.js +++ b/src/index.js @@ -6,10 +6,13 @@ const session = require('express-session'); const bodyParser = require('body-parser'); const app = express(); const mustacheExpress = require('mustache-express'); +const i18n = require('i18n'); const config = require('./config.json'); +const utils = require('./utils.js'); const Device = require('./models/device.js'); const Config = require('./models/config.js'); +const Log = require('./models/log.js'); const Migrator = require('./migrator.js'); const ScheduleManager = require('./models/schedule-manager.js'); const apiRoutes = require('./routes/api.js'); @@ -18,30 +21,74 @@ const timezones = require('../static/data/timezones.json'); // TODO: Create route classes // TODO: Fix devices scroll with DataTables -// TODO: Secure /api/config/:uuid endpoint with token -// TODO: Provider option to show/hide config options +// TODO: Delete all logs button +// TODO: Secure /api/config endpoint with token // TODO: Accomodate for # in uuid name +// TODO: Fix schedule end time +// TODO: Center align data in table columns +// TODO: Change require to import -const defaultData = { - title: config.title, - locale: config.locale, - style: config.style == 'dark' ? 'dark' : '', - logging: config.logging -}; +const providers = [ + { name: 'GoCheats' }, + { name: 'Kevin' }, +]; -// Start database migrator -var dbMigrator = new Migrator(); -dbMigrator.load(); +run(); -// Middleware +async function run() { + // Start database migrator + var dbMigrator = new Migrator(); + dbMigrator.load(); + while (dbMigrator.done === false) { + await utils.snooze(1000); + } + app.listen(config.port, config.interface, () => console.log(`Listening on port ${config.port}...`)); +} + +i18n.configure({ + locales:['en', 'es', 'de'], + directory: path.resolve(__dirname, '../static/locales') +}); + +// View engine app.set('view engine', 'mustache'); app.set('views', path.resolve(__dirname, 'views')); app.engine('mustache', mustacheExpress()); + +// Static paths +app.use(express.static(path.resolve(__dirname, '../static'))); +//app.use('/logs', express.static(path.resolve(__dirname, '../logs'))); +app.use('/screenshots', express.static(path.resolve(__dirname, '../screenshots'))); + +//app.use(express.cookieParser()); +app.use(i18n.init); + +// register helper as a locals function wrapped as mustache expects +app.use(function (req, res, next) { + // mustache helper + res.locals.__ = function() { + /* eslint-disable no-unused-vars */ + return function(text, render) { + return i18n.__.apply(req, arguments); + }; + /* eslint-disable no-unused-vars */ + }; + next(); +}); + +// Default mustache data shared between pages +const defaultData = require('../static/locales/' + config.locale + '.json'); +defaultData.title = config.title; +defaultData.locale = config.locale; +defaultData.style = config.style == 'dark' ? 'dark' : ''; +defaultData.logging = config.logging.enabled; + +i18n.setLocale(config.locale); + +// Body parser middlewares app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: false, limit: '50mb' })); // for parsing application/x-www-form-urlencoded //app.use(bodyParser.raw({ type: 'application/x-www-form-urlencoded' })); -app.use(express.static(path.resolve(__dirname, '../static'))); -app.use('/screenshots', express.static(path.resolve(__dirname, '../screenshots'))); // Sessions middleware app.use(session({ @@ -74,12 +121,14 @@ app.get(['/', '/index'], async function(req, res) { var configs = await Config.getAll(); var schedules = ScheduleManager.getAll(); var metadata = await Migrator.getEntries(); + var logsSize = Log.getTotalSize(); var data = defaultData; data.metadata = metadata; data.devices = devices.length; data.configs = configs.length; data.schedules = Object.keys(schedules).length; data.username = username; + data.logs_size = utils.formatBytes(logsSize); res.render('index', data); } }); @@ -110,6 +159,8 @@ app.get('/devices', function(req, res) { }); app.get('/device/new', async function(req, res) { + var data = defaultData; + data.configs = await Config.getAll(); res.render('device-new', defaultData); }); @@ -173,7 +224,9 @@ app.get('/config/assign/:uuid', async function(req, res) { }); app.get('/config/new', function(req, res) { - res.render('config-new', defaultData); + var data = defaultData; + data.providers = providers; + res.render('config-new', data); }); app.get('/config/edit/:name', async function(req, res) { @@ -183,6 +236,11 @@ app.get('/config/edit/:name', async function(req, res) { data.title = config.title; data.old_name = name; data.name = c.name; + data.providers = providers; + data.providers.forEach(function(provider) { + provider.selected = provider.name === c.provider; + }); + data.gocheats_selected = c.provider === data.providers[0].name; data.backend_url = c.backendUrl; data.data_endpoints = c.dataEndpoints; data.token = c.token; @@ -280,9 +338,8 @@ app.get('/settings', function(req, res) { data.languages.forEach(function(locale) { locale.selected = locale.name === config.locale; }); - data.logging = config.logging ? 'checked' : ''; + data.logging = config.logging.enabled ? 'checked' : ''; + data.max_size = config.logging.max_size; console.log('Settings:', data); res.render('settings', data); -}); - -app.listen(config.port, config.interface, () => console.log(`Listening on port ${config.port}...`)); \ No newline at end of file +}); \ No newline at end of file diff --git a/src/migrator.js b/src/migrator.js index d068a59..334f63d 100644 --- a/src/migrator.js +++ b/src/migrator.js @@ -9,6 +9,7 @@ const migrationsDir = path.resolve(__dirname, '../migrations'); class Migrator { constructor() { + this.done = false; } async load() { var count = 1; @@ -123,11 +124,12 @@ class Migrator { process.exit(-1); }); console.log('[DBController] Migration successful'); - if (newVersion === toVersion) { - console.log('[DBController] Migration done'); - } this.migrate(newVersion, toVersion); } + if (fromVersion === toVersion) { + console.log('[DBController] Migration done'); + this.done = true; + } } backup() { // TODO: Migrator backup diff --git a/src/models/config.js b/src/models/config.js index 11e73ed..d281868 100644 --- a/src/models/config.js +++ b/src/models/config.js @@ -3,9 +3,10 @@ const query = require('../db.js'); class Config { - constructor(name, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, + constructor(name, provider, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, accountManager, deployEggs, nearbyTracker, autoLogin, isDefault) { this.name = name; + this.provider = provider; this.backendUrl = backendUrl; this.dataEndpoints = dataEndpoints; this.token = token; @@ -23,7 +24,7 @@ class Config { } static async getByName(name) { var sql = ` - SELECT backend_url, data_endpoints, token, heartbeat_max_time, min_delay_logout, + SELECT backend_url, provider, data_endpoints, token, heartbeat_max_time, min_delay_logout, account_manager, deploy_eggs, nearby_tracker, auto_login, is_default FROM configs WHERE name = ? @@ -36,6 +37,7 @@ class Config { var c = result[0]; var data = new Config( name, + c.provider, c.backend_url, c.data_endpoints, c.token, @@ -49,13 +51,13 @@ class Config { ); return data; } - static async create(name, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, + static async create(name, provider, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, accountManager, deployEggs, nearbyTracker, autoLogin, isDefault) { var sql = ` - INSERT INTO configs (name, backend_url, data_endpoints, token, heartbeat_max_time, min_delay_logout, + INSERT INTO configs (name, provider, backend_url, data_endpoints, token, heartbeat_max_time, min_delay_logout, account_manager, deploy_eggs, nearby_tracker, auto_login, is_default) - VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`; - var args = [name, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, + VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`; + var args = [name, provider, backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, accountManager, deployEggs, nearbyTracker, autoLogin, isDefault]; var result = await query(sql, args); return result.affectedRows === 1; @@ -68,7 +70,7 @@ class Config { } static async getDefault() { var sql = ` - SELECT name, backend_url, data_endpoints, token, heartbeat_max_time, min_delay_logout, + SELECT name, provider, backend_url, data_endpoints, token, heartbeat_max_time, min_delay_logout, account_manager, deploy_eggs, nearby_tracker, auto_login, is_default FROM configs WHERE is_default = 1 @@ -80,6 +82,7 @@ class Config { var c = result[0]; var data = new Config( c.name, + c.provider, c.backend_url, c.data_endpoints, c.token, @@ -95,7 +98,7 @@ class Config { } static async setDefault(name) { var sql = ` - UPDATE dcm.configs + UPDATE configs SET is_default = IF(name = ?, 1, 0);`; var args = [name]; var result = await query(sql, args); @@ -104,11 +107,12 @@ class Config { async save(oldName) { var sql = ` UPDATE configs - SET name=?, backend_url=?, data_endpoints=?, token=?, heartbeat_max_time=?, min_delay_logout=?, + SET name=?, provider=?, backend_url=?, data_endpoints=?, token=?, heartbeat_max_time=?, min_delay_logout=?, account_manager=?, deploy_eggs=?, nearby_tracker=?, auto_login=?, is_default=? WHERE name=?`; var args = [ this.name, + this.provider, this.backendUrl, this.dataEndpoints, this.token, diff --git a/src/models/log.js b/src/models/log.js index 21f755d..4ce64fd 100644 --- a/src/models/log.js +++ b/src/models/log.js @@ -3,6 +3,7 @@ const fs = require('fs'); const path = require('path'); const utils = require('../utils.js'); +const config = require('../config.json'); const logsDir = path.resolve(__dirname, '../../logs'); if (!fs.existsSync(logsDir)) { @@ -38,6 +39,13 @@ class Log { static create(uuid, message) { var name = uuid + '.log'; var logFile = path.resolve(logsDir, name); + if (fs.existsSync(logFile)) { + var size = fs.statSync(logFile).size || 0; + var maxSize = (config.logging.max_size || 5) * 1024 * 1024; + if (size >= maxSize) { + this.delete(uuid); + } + } var msg = { message: message, timestamp: new Date() / 1000, @@ -67,6 +75,19 @@ class Log { }); }); } + static getTotalSize() { + var total = 0; + if (!fs.existsSync(logsDir)) { + return total; + } + var logs = fs.readdirSync(logsDir); + logs.forEach(function(log) { + var logFile = path.join(logsDir, log); + var stats = fs.statSync(logFile); + total += stats.size; + }); + return total; + } } module.exports = Log; \ No newline at end of file diff --git a/src/routes/api.js b/src/routes/api.js index 5074523..e4ad098 100644 --- a/src/routes/api.js +++ b/src/routes/api.js @@ -76,7 +76,8 @@ router.post('/settings/change_ui', function(req, res) { newConfig.title = data.title; newConfig.locale = data.locale; newConfig.style = data.style; - newConfig.logging = data.logging === 'on' ? 1 : 0; + newConfig.logging.enabled = data.logging === 'on'; + newConfig.logging.max_size = data.max_size; fs.writeFileSync(path.resolve(__dirname, '../config.json'), JSON.stringify(newConfig, null, 2)); res.redirect('/settings'); }); @@ -269,6 +270,7 @@ router.post('/config', async function(req, res) { res.send(JSON.stringify(noConfigData2)); return; } + // Build json config var json = utils.buildConfig( c.backendUrl, @@ -302,8 +304,9 @@ router.post('/config/assign/:uuid', async function(req, res) { router.post('/config/new', async function(req, res) { var data = req.body; - var result = await Config.create( + var cfg = await Config.create( data.name, + data.provider, data.backend_url, data.data_endpoints, data.token, @@ -315,8 +318,12 @@ router.post('/config/new', async function(req, res) { data.auto_login === 'on' ? 1 : 0, data.is_default === 'on' ? 1 : 0 ); - if (result) { + if (cfg) { console.log('Config inserted'); + if (cfg.isDefault) { + console.log('Setting default config:', data.name); + await Config.setDefault(data.name); + } } else { console.error('Failed to create new config'); } @@ -328,6 +335,7 @@ router.post('/config/edit/:name', async function(req, res) { var data = req.body; var c = await Config.getByName(oldName); c.name = data.name; + c.provider = data.provider; c.backendUrl = data.backend_url; c.dataEndpoints = data.data_endpoints; c.token = data.token; @@ -339,9 +347,11 @@ router.post('/config/edit/:name', async function(req, res) { c.autoLogin = data.auto_login === 'on' ? 1 : 0; c.isDefault = data.is_default === 'on' ? 1 : 0; if (await c.save(oldName)) { + console.log('Config saved'); // Success if (c.isDefault) { - await Config.setDefault(oldName); + console.log('Setting default config:', c.name); + await Config.setDefault(c.name); } } res.redirect('/configs'); @@ -444,7 +454,7 @@ router.get('/logs/:uuid', function(req, res) { }); router.post('/log/new', function(req, res) { - if (config.logging === false) { + if (config.logging.enabled === false) { // Logs are disabled res.send('OK'); return; @@ -472,6 +482,16 @@ router.get('/log/delete/:uuid', function(req, res) { res.redirect('/device/logs/' + uuid); }); +router.get('/log/export/:uuid', function(req, res) { + var uuid = req.params.uuid; + var logs = Log.getByDevice(uuid); + var logText = ''; + logs.forEach(function(log) { + logText += `${log.timestamp} ${log.uuid} ${log.message}\n`; + }); + res.send(logText); +}); + router.get('/logs/delete_all', function(req, res) { var result = Log.deleteAll(); if (result) { diff --git a/src/utils.js b/src/utils.js index 1be3e7c..de96926 100644 --- a/src/utils.js +++ b/src/utils.js @@ -19,6 +19,7 @@ function getDateTime(timestamp) { function buildConfig(backendUrl, dataEndpoints, token, heartbeatMaxTime, minDelayLogout, accountManager, deployEggs, nearbyTracker, autoLogin) { + // TODO: Check provider against keys needed (or just assume?) and only return those keys. var obj = { 'backend_url': backendUrl, 'data_endpoints': (dataEndpoints || '').split(',') || [], @@ -34,9 +35,29 @@ function buildConfig(backendUrl, dataEndpoints, token, heartbeatMaxTime, minDela return json; } +function saveDataAsImage(name, imgData) { + var data = imgData.replace(/^data:image\/\w+;base64,/, ''); + var buf = new Buffer(data, 'base64'); + fs.writeFileSync('../screenshots/' + name, buf); +} + +function formatBytes(bytes) { + const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']; + if (bytes === 0) { + return 0; + } + const index = parseInt(Math.floor(Math.log(bytes) / Math.log(1024))); + if (index == 0) { + return bytes + ' ' + sizes[index]; + } + return (bytes / Math.pow(1024, index)).toFixed(1) + ' ' + sizes[index]; +} + module.exports = { readFile, snooze, getDateTime, - buildConfig + buildConfig, + saveDataAsImage, + formatBytes }; \ No newline at end of file diff --git a/src/views/account.mustache b/src/views/account.mustache index dbfa6e1..a2305da 100644 --- a/src/views/account.mustache +++ b/src/views/account.mustache @@ -3,28 +3,28 @@
{{> navbar}}