'use strict'; const BaseMapConfigProvider = require('./base-mapconfig-adapter'); const crypto = require('crypto'); const dot = require('dot'); const MapConfig = require('windshaft').model.MapConfig; const templateName = require('../../../backends/template_maps').templateName; // Configure bases for cache keys suitable for string interpolation const baseKey = '{{=it.dbname}}:{{=it.user}}:{{=it.templateName}}'; const rendererKey = baseKey + ':{{=it.authToken}}:{{=it.configHash}}:{{=it.format}}:{{=it.layer}}:{{=it.scale_factor}}'; const baseKeyTpl = dot.template(baseKey); const rendererKeyTpl = dot.template(rendererKey); module.exports = class NamedMapMapConfigProvider extends BaseMapConfigProvider { constructor ( templateMaps, pgConnection, metadataBackend, userLimitsBackend, mapConfigAdapter, affectedTablesCache, user, templateId, config, authToken, params ) { super(); this.templateMaps = templateMaps; this.pgConnection = pgConnection; this.metadataBackend = metadataBackend; this.userLimitsBackend = userLimitsBackend; this.mapConfigAdapter = mapConfigAdapter; this.user = user; this.templateName = templateName(templateId); this.config = config; this.authToken = authToken; this.params = params; this.cacheBuster = Date.now(); // use template after call to mapConfig this.template = null; this.affectedTablesCache = affectedTablesCache; // providing this.mapConfig = null; this.rendererParams = null; this.context = {}; this.analysesResults = []; } getMapConfig (callback) { if (this.mapConfig !== null) { return callback(null, this.mapConfig, this.rendererParams, this.context); } this.getContext((err, context) => { if (err) { return callback(err); } let templateParams = {}; if (this.config) { try { templateParams = Object.prototype.toString.call(this.config) === '[object String]' ? JSON.parse(this.config) : this.config; } catch (e) { const err = new Error('malformed config parameter, should be a valid JSON'); return callback(err); } } context.templateParams = templateParams; this.getTemplate((err, template) => { if (err) { return callback(err); } let requestMapConfig; try { requestMapConfig = this.templateMaps.instance(template, templateParams); } catch (err) { return callback(err); } const { user, rendererParams } = this; this.mapConfigAdapter.getMapConfig( user, requestMapConfig, rendererParams, context, (err, mapConfig, stats = {}) => { if (err) { return callback(err); } this.mapConfig = (mapConfig === null) ? null : new MapConfig(mapConfig, context.datasource); this.analysesResults = context.analysesResults || []; return callback(null, this.mapConfig, this.rendererParams, this.context, stats); }); }); }); } getContext (callback) { this.getDBParams(this.user, (err, rendererParams) => { if (err) { return callback(err); } this.rendererParams = rendererParams; this.metadataBackend.getUserMapKey(this.user, (err, apiKey) => { if (err) { return callback(err); } const context = {}; context.analysisConfiguration = { user: this.user, db: { host: rendererParams.dbhost, port: rendererParams.dbport, dbname: rendererParams.dbname, user: rendererParams.dbuser, pass: rendererParams.dbpassword }, batch: { username: this.user, apiKey: apiKey } }; this.userLimitsBackend.getRenderLimits(this.user, this.params.api_key, (err, renderLimits) => { if (err) { return callback(err); } context.limits = renderLimits || {}; this.context = context; return callback(null, context); }); }); }); } getTemplate (callback) { if (this.template !== null) { return callback(null, this.template); } this.templateMaps.getTemplate(this.user, this.templateName, (err, tpl) => { if (err) { return callback(err); } if (!tpl) { const error = new Error(`Template '${this.templateName}' of user '${this.user}' not found`); error.http_status = 404; return callback(error); } let authorized = false; try { authorized = this.templateMaps.isAuthorized(tpl, this.authToken); } catch (err) { const error = new Error('Failed to authorize template'); error.http_status = 403; return callback(error); } if (!authorized) { const error = new Error('Unauthorized template instantiation'); error.http_status = 403; return callback(error); } this.template = tpl; return callback(null, this.template); }); } getKey () { return this.createKey(false); } getCacheBuster () { return this.cacheBuster; } reset () { this.template = null; this.affectedTables = null; this.mapConfig = null; this.cacheBuster = Date.now(); } filter (key) { const regex = new RegExp('^' + this.createKey(true) + '.*'); return key && key.match(regex); } createKey (base) { const { dbname = '', user = this.user, templateName = this.templateName, authToken = this.authToken || '', configHash = createConfigHash(this.config), layer = '', scale_factor = 1 } = this.params; const tplValues = { dbname, user, templateName, authToken, configHash, layer, scale_factor }; return (base) ? baseKeyTpl(tplValues) : rendererKeyTpl(tplValues); } getDBParams (cdbuser, callback) { const dbParams = Object.assign({ user: this.user }, this.params); this.pgConnection.getDatabaseParams(cdbuser, (err, databaseParams) => { if (err) { return callback(err); } dbParams.dbuser = databaseParams.dbuser; dbParams.dbpass = databaseParams.dbpass; dbParams.dbhost = databaseParams.dbhost; dbParams.dbport = databaseParams.dbport; dbParams.dbname = databaseParams.dbname; return callback(null, dbParams); }); } getTemplateName () { return this.templateName; } }; function createConfigHash(config) { if (!config) { return ''; } return crypto.createHash('md5').update(JSON.stringify(config)).digest('hex').substring(0,8); } module.exports.configHash = createConfigHash;