mylogger/lib/model-loader.js

248 lines
6.7 KiB
JavaScript

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};
}