const path = require('path'); const {loadConfig, toUpperCamelCase} = require('./util'); const MultiMap = require('./multi-map'); /** * Loads model configuration. */ module.exports = class ModelLoader { init(logger) { const configDir = path.join(__dirname, '..'); const conf = loadConfig(configDir, 'logs'); const schemaMap = new MultiMap(); const logMap = new Map(); Object.assign(this, { logger, conf }); Object.assign(logger, { schemaMap, logMap }); for (const logName in conf.logs) { const logConf = conf.logs[logName]; const schema = logConf.schema || logger.conf.dstDb.database; const logInfo = { name: logName, conf: logConf, schema, table: parseTable(logConf.logTable, schema), mainTable: parseTable(logConf.mainTable, schema) }; logMap.set(logName, logInfo); const mainTable = addTable(logConf.mainTable, logInfo); mainTable.isMain = true; if (logConf.tables) for (const tableConf of logConf.tables){ const table = addTable(tableConf, logInfo); if (table !== mainTable) { Object.assign(table, { main: mainTable, isMain: false }); } } } function addTable(tableConf, logInfo) { if (typeof tableConf == 'string') tableConf = {name: tableConf}; const table = parseTable(tableConf.name, logInfo.schema); let tableInfo = schemaMap.get(table.schema, table.name); if (!tableInfo) { tableInfo = { conf: tableConf, logInfo }; schemaMap.set(table.schema, table.name, tableInfo); } let modelName = tableConf.modelName; if (!modelName) { modelName = conf.upperCaseTable ? toUpperCamelCase(table.name) : table.name; } Object.assign(tableInfo, { conf: tableConf, modelName, relation: tableConf.relation }); return tableInfo; } } async loadSchema() { const {db, schemaMap} = this.logger; const {conf} = this; const excludeFields = new Set(conf.excludeFields); const excludeRegex = conf.excludeRegex ? new RegExp(conf.excludeRegex) : null; const localProps = [ 'idName', 'showField', 'logFields' ]; const globalProps = [ 'userField', 'rowExcludeField', 'reasonField' ]; for (const [schema, table, tableInfo] of schemaMap) { const tableConf = tableInfo.conf; for (const prop of localProps) tableInfo[prop] = tableConf[prop]; for (const prop of globalProps) tableInfo[prop] = tableConf[prop] !== undefined ? tableConf[prop] : conf[prop]; // Fetch columns & types const columns = new Set(); Object.assign (tableInfo, { castTypes: new Map(), columns }); if (tableConf.types) for (const col in tableConf.types) tableInfo.castTypes.set(col, tableConf.types[col]); const [dbCols] = await db.query( `SELECT COLUMN_NAME \`col\`, DATA_TYPE \`type\` FROM information_schema.\`COLUMNS\` WHERE TABLE_NAME = ? AND TABLE_SCHEMA = ?`, [table, schema] ); const exclude = new Set(tableConf.exclude); exclude.add(tableInfo.userField); exclude.add(tableInfo.reasonField); for (const {col, type} of dbCols) { const isExcluded = excludeFields.has(col) || (excludeRegex && excludeRegex.test(col)) || exclude.has(col); if (!isExcluded) columns.add(col); const castType = conf.castTypes[type]; if (castType && !tableInfo.castTypes.has(col)) tableInfo.castTypes.set(col, castType); } // Fetch primary key const {idName} = tableInfo; if (!idName) { const [pks] = await db.query( `SELECT COLUMN_NAME pk FROM information_schema.KEY_COLUMN_USAGE WHERE CONSTRAINT_NAME = 'PRIMARY' AND TABLE_NAME = ? AND TABLE_SCHEMA = ?`, [table, schema] ); if (!pks.length) throw new Error(`Primary key not found: ${schema}.${table}`); if (pks.length > 1) throw new Error(`Only one column primary is supported: ${schema}.${table}`); const [{pk}] = pks; tableInfo.idName = pk; } else if (!columns.has(idName)) throw new Error(`Primary column not found: ${schema}.${table}.${idName}`); // Get show field const {showField} = tableInfo; if (showField !== null) { if (showField === undefined) { if (!tableInfo.isMain || conf.logMainShowField) for (const field of conf.showFields) { if (columns.has(field)) { tableInfo.showField = field; break; } } } else if (!columns.has(showField)) throw new Error(`Show column not found: ${schema}.${table}.${showField}`); } } for (const [schema, table, tableInfo] of schemaMap) { // Fetch relation to main table if (!tableInfo.conf.relation && !tableInfo.isMain) { const mainTable = tableInfo.logInfo.mainTable; const mainInfo = schemaMap.get(mainTable.schema, mainTable.name); const [mainRelations] = await db.query( `SELECT COLUMN_NAME relation FROM information_schema.KEY_COLUMN_USAGE WHERE TABLE_NAME = ? AND TABLE_SCHEMA = ? AND REFERENCED_TABLE_NAME = ? AND REFERENCED_TABLE_SCHEMA = ? AND REFERENCED_COLUMN_NAME = ?`, [ table, schema, mainTable.name, mainTable.schema, mainInfo.idName ] ); if (!mainRelations.length) throw new Error(`No relation to main table found for table: ${schema}.${table}`); if (mainRelations.length > 1) throw new Error(`Found more multiple relations to main table: ${schema}.${table}`); for (const {relation} of mainRelations) tableInfo.relation = relation; } // Set instance columns const {columns} = tableInfo; const cols = new Set(columns); if (!conf.logId) cols.delete(tableInfo.idName); if (!conf.logRelation) cols.delete(tableInfo.relation); tableInfo.instanceColumns = [...cols.keys()]; } } } function parseTable(tableString, defaultSchema) { let name, schema; const split = tableString.split('.'); if (split.length == 1) { name = split[0]; schema = defaultSchema; } else { [schema, name] = split; } return {name, schema}; }