import * as Bluebird from 'bluebird'; import { EventEmitter } from 'events'; import { Transaction } from 'knex'; import * as _ from 'lodash'; import { generateUniqueKey } from 'resin-register-device'; import { Either } from 'fp-ts/lib/Either'; import * as t from 'io-ts'; import ConfigJsonConfigBackend from './configJson'; import * as FnSchema from './functions'; import * as Schema from './schema'; import { SchemaReturn, SchemaTypeKey, schemaTypes } from './schema-type'; import DB from '../db'; import { ConfigurationValidationError, InternalInconsistencyError, } from '../lib/errors'; interface ConfigOpts { db: DB; configPath: string; } type ConfigMap = { [key in T]: SchemaReturn }; export class Config extends EventEmitter { private db: DB; private configJsonBackend: ConfigJsonConfigBackend; public constructor({ db, configPath }: ConfigOpts) { super(); this.db = db; this.configJsonBackend = new ConfigJsonConfigBackend( Schema.schema, configPath, ); } public init(): Bluebird { return this.configJsonBackend.init().then(() => { return this.generateRequiredFields(); }); } public get( key: T, trx?: Transaction, ): Bluebird> { const db = trx || this.db.models.bind(this.db); return Bluebird.try(() => { if (Schema.schema.hasOwnProperty(key)) { const schemaKey = key as Schema.SchemaKey; return this.getSchema(schemaKey, db).then(value => { if (value == null) { const defaultValue = schemaTypes[key].default; if (defaultValue instanceof t.Type) { // The only reason that this would be the case in a non-function // schema key is for the meta nullOrUndefined value. We check this // by first decoding the value undefined with the default type, and // then return undefined const decoded = (defaultValue as t.Type).decode(undefined); this.checkValueDecode(decoded, key, undefined); return decoded.value; } return defaultValue as SchemaReturn; } const decoded = this.decodeSchema(schemaKey, value); this.checkValueDecode(decoded, key, value); return decoded.value; }); } else if (FnSchema.fnSchema.hasOwnProperty(key)) { const fnKey = key as FnSchema.FnSchemaKey; // Cast the promise as something that produes an unknown, and this means that // we can validate the output of the function as well, ensuring that the type matches const promiseValue = FnSchema.fnSchema[fnKey](this) as Bluebird< unknown >; return promiseValue.then((value: unknown) => { const decoded = schemaTypes[key].type.decode(value); this.checkValueDecode(decoded, key, value); return decoded.value as SchemaReturn; }); } else { throw new Error(`Unknown config value ${key}`); } }); } public getMany( keys: T[], trx?: Transaction, ): Bluebird<{ [key in T]: SchemaReturn }> { return Bluebird.map(keys, (key: T) => this.get(key, trx)).then(values => { return _.zipObject(keys, values); }); } public set( keyValues: ConfigMap, trx?: Transaction, ): Bluebird { const setValuesInTransaction = (tx: Transaction) => { const configJsonVals: Dictionary = {}; const dbVals: Dictionary = {}; _.each(keyValues, (v, k: T) => { const schemaKey = k as Schema.SchemaKey; const source = Schema.schema[schemaKey].source; switch (source) { case 'config.json': configJsonVals[schemaKey] = v; break; case 'db': dbVals[schemaKey] = v; break; default: throw new Error( `Unknown configuration source: ${source} for config key: ${k}`, ); break; } }); const dbKeys = _.keys(dbVals) as T[]; return this.getMany(dbKeys, tx) .then(oldValues => { return Bluebird.map(dbKeys, (key: T) => { const value = dbVals[key]; // if we have anything other than a string, it must be converted to // a string before being stored in the db const strValue = Config.valueToString(value); if (oldValues[key] !== value) { return this.db.upsertModel( 'config', { key, value: strValue }, { key }, tx, ); } }); }) .then(() => { if (!_.isEmpty(configJsonVals)) { return this.configJsonBackend.set(configJsonVals as { [key in Schema.SchemaKey]: unknown }); } }); }; return Bluebird.try(() => { // Firstly validate all of the types as they are being set this.validateConfigMap(keyValues); if (trx != null) { return setValuesInTransaction(trx).return(); } else { return this.db .transaction((tx: Transaction) => setValuesInTransaction(tx)) .return(); } }).then(() => { this.emit('change', keyValues); }); } public remove(key: T): Bluebird { return Bluebird.try(() => { if (Schema.schema[key] == null || !Schema.schema[key].mutable) { throw new Error( `Attempt to delete non-existent or immutable key ${key}`, ); } if (Schema.schema[key].source === 'config.json') { return this.configJsonBackend.remove(key); } else if (Schema.schema[key].source === 'db') { return this.db .models('config') .del() .where({ key }); } else { throw new Error( `Unknown or unsupported config backend: ${Schema.schema[key].source}`, ); } }); } public regenerateRegistrationFields(): Bluebird { return this.set({ uuid: this.newUniqueKey(), deviceApiKey: this.newUniqueKey(), }); } public newUniqueKey(): string { return generateUniqueKey(); } private async getSchema( key: T, db: Transaction, ): Promise { let value: unknown; switch (Schema.schema[key].source) { case 'config.json': value = await this.configJsonBackend.get(key); break; case 'db': value = await db('config') .select('value') .where({ key }) .then(([conf]: [{ value: string }]) => { if (conf != null) { return conf.value; } return; }); break; } return value; } private decodeSchema( key: T, value: unknown, ): Either> { return schemaTypes[key].type.decode(value); } private validateConfigMap(configMap: ConfigMap) { // Just loop over every value, run the decode function, and // throw if any value fails verification _.map(configMap, (value, key) => { if ( !Schema.schema.hasOwnProperty(key) || !Schema.schema[key as Schema.SchemaKey].mutable ) { throw new Error( `Attempt to set value for non-mutable schema key: ${key}`, ); } // If the default entry in the schema is a type and not a value, // use this in the validation of the value const schemaTypesEntry = schemaTypes[key as SchemaTypeKey]; let type: t.Type; if (schemaTypesEntry.default instanceof t.Type) { type = t.union([schemaTypesEntry.type, schemaTypesEntry.default]); } else { type = schemaTypesEntry.type; } const decoded = type.decode(value); if (decoded.isLeft()) { throw new Error( `Cannot set value for ${key}, as value failed validation: ${ decoded.value }`, ); } }); } private generateRequiredFields() { return this.getMany([ 'uuid', 'deviceApiKey', 'apiSecret', 'unmanaged', ]).then(({ uuid, deviceApiKey, apiSecret, unmanaged }) => { // These fields need to be set regardless if (uuid == null || apiSecret == null) { uuid = uuid || this.newUniqueKey(); apiSecret = apiSecret || this.newUniqueKey(); } return this.set({ uuid, apiSecret }).then(() => { if (unmanaged) { return; } if (!deviceApiKey) { return this.set({ deviceApiKey: this.newUniqueKey() }); } }); }); } private static valueToString(value: unknown) { switch (typeof value) { case 'object': return JSON.stringify(value); case 'number': case 'string': case 'boolean': return value.toString(); default: throw new InternalInconsistencyError( `Could not convert configuration value to string for storage, value: ${value}, type: ${typeof value}`, ); } } private checkValueDecode( decoded: Either, key: string, value: unknown, ): void { if (decoded.isLeft()) { throw new ConfigurationValidationError(key, value); } } } export default Config;