swaf/dist/db/Model.js

291 lines
37 KiB
JavaScript

var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
import MysqlConnectionManager, { query } from "./MysqlConnectionManager";
import Validator from "./Validator";
import Query from "./Query";
import Pagination from "../Pagination";
export default class Model {
constructor(data) {
this.properties = [];
this.relations = {};
this.defineProperty('id', new Validator());
this.defineProperties();
this.updateWithData(data);
}
static getById(id) {
return __awaiter(this, void 0, void 0, function* () {
const cachedModel = ModelCache.get(this.table, id);
if ((cachedModel === null || cachedModel === void 0 ? void 0 : cachedModel.constructor) === this) {
return cachedModel;
}
const models = yield this.models(this.select().where('id', id).first());
return models.length > 0 ? models[0] : null;
});
}
static paginate(request, perPage = 20) {
return __awaiter(this, void 0, void 0, function* () {
let page = request.params.page ? parseInt(request.params.page) : 1;
let query = this.select().limit(perPage, (page - 1) * perPage).withTotalRowCount();
if (request.params.sortBy) {
const dir = request.params.sortDirection;
query = query.sortBy(request.params.sortBy, dir === 'ASC' || dir === 'DESC' ? dir : undefined);
}
else {
query = query.sortBy('id');
}
const models = yield this.models(query);
// @ts-ignore
models.pagination = new Pagination(models, page, perPage, models.totalCount);
return models;
});
}
static select(...fields) {
return Query.select(this.table, ...fields);
}
static update(data) {
return Query.update(this.table, data);
}
static delete() {
return Query.delete(this.table);
}
static models(query) {
return __awaiter(this, void 0, void 0, function* () {
const results = yield query.execute();
const models = [];
const factory = this.getFactory();
for (const result of results.results) {
const cachedModel = ModelCache.get(this.table, result.id);
if (cachedModel && cachedModel.constructor === this) {
cachedModel.updateWithData(result);
models.push(cachedModel);
}
else {
models.push(factory(result));
}
}
// @ts-ignore
models.totalCount = results.foundRows;
return models;
});
}
static loadRelation(models, relation, model, localField) {
return __awaiter(this, void 0, void 0, function* () {
const loadMap = {};
const ids = models.map(m => {
m.relations[relation] = null;
if (m[localField])
loadMap[m[localField]] = v => m.relations[relation] = v;
return m[localField];
}).filter(id => id);
for (const v of yield model.models(model.select().whereIn('id', ids))) {
loadMap[v.id](v);
}
});
}
static getFactory(factory) {
if (factory === undefined) {
factory = this.FACTORY;
if (factory === undefined)
factory = data => new this(data);
}
return factory;
}
defineProperty(name, validator) {
if (validator === undefined)
validator = new Validator();
if (validator instanceof RegExp) {
const regexp = validator;
validator = new Validator().regexp(regexp);
}
const prop = new ModelProperty(name, validator);
this.properties.push(prop);
Object.defineProperty(this, name, {
get: () => prop.value,
set: (value) => prop.value = value,
});
}
updateWithData(data) {
this.id = data['id'];
for (const prop of this.properties) {
if (data[prop.name] !== undefined) {
this[prop.name] = data[prop.name];
}
}
}
beforeSave(exists, connection) {
return __awaiter(this, void 0, void 0, function* () {
});
}
afterSave() {
return __awaiter(this, void 0, void 0, function* () {
});
}
save(connection, postHook) {
return __awaiter(this, void 0, void 0, function* () {
yield this.validate(false, connection);
const exists = yield this.exists();
let needs_full_update = false;
if (connection) {
needs_full_update = yield this.saveTransaction(connection, exists, needs_full_update);
}
else {
needs_full_update = yield MysqlConnectionManager.wrapTransaction((connection) => __awaiter(this, void 0, void 0, function* () { return this.saveTransaction(connection, exists, needs_full_update); }));
}
const callback = () => __awaiter(this, void 0, void 0, function* () {
if (needs_full_update) {
this.updateWithData((yield this.constructor.select().where('id', this.id).first().execute()).results[0]);
}
if (!exists) {
this.cache();
}
yield this.afterSave();
});
if (connection) {
postHook(callback);
}
else {
yield callback();
}
});
}
saveTransaction(connection, exists, needs_full_update) {
return __awaiter(this, void 0, void 0, function* () {
// Before save
yield this.beforeSave(exists, connection);
if (exists && this.hasOwnProperty('updated_at')) {
this.updated_at = new Date();
}
const props = [];
const values = [];
if (exists) {
for (const prop of this.properties) {
if (prop.value !== undefined) {
props.push(prop.name + '=?');
values.push(prop.value);
}
else {
needs_full_update = true;
}
}
values.push(this.id);
yield query(`UPDATE ${this.table} SET ${props.join(',')} WHERE id=?`, values, connection);
}
else {
const props_holders = [];
for (const prop of this.properties) {
if (prop.value !== undefined) {
props.push(prop.name);
props_holders.push('?');
values.push(prop.value);
}
else {
needs_full_update = true;
}
}
const result = yield query(`INSERT INTO ${this.table} (${props.join(', ')}) VALUES(${props_holders.join(', ')})`, values, connection);
this.id = result.other.insertId;
}
return needs_full_update;
});
}
static get table() {
return this.name
.replace(/(?:^|\.?)([A-Z])/g, (x, y) => '_' + y.toLowerCase())
.replace(/^_/, '')
+ 's';
}
get table() {
// @ts-ignore
return this.constructor.table;
}
exists() {
return __awaiter(this, void 0, void 0, function* () {
if (!this.id)
return false;
const result = yield query(`SELECT 1 FROM ${this.table} WHERE id=? LIMIT 1`, [
this.id,
]);
return result.results.length > 0;
});
}
delete() {
return __awaiter(this, void 0, void 0, function* () {
if (!(yield this.exists()))
throw new Error('This model instance doesn\'t exist in DB.');
yield query(`DELETE FROM ${this.table} WHERE id=?`, [
this.id,
]);
ModelCache.forget(this);
this.id = undefined;
});
}
validate(onlyFormat = false, connection) {
return __awaiter(this, void 0, void 0, function* () {
return yield Promise.all(this.properties.map(prop => prop.validate(onlyFormat, connection)));
});
}
cache() {
ModelCache.cache(this);
}
relation(name) {
if (this.relations[name] === undefined)
throw new Error('Model not loaded');
return this.relations[name];
}
}
class ModelProperty {
constructor(name, validator) {
this.name = name;
this.validator = validator;
}
validate(onlyFormat, connection) {
return __awaiter(this, void 0, void 0, function* () {
return yield this.validator.execute(this.name, this.value, onlyFormat, connection);
});
}
get value() {
return this.val;
}
set value(val) {
this.val = val;
}
}
export class ModelCache {
static cache(instance) {
if (instance.id === undefined)
throw new Error('Cannot cache an instance with an undefined id.');
let tableCache = this.caches[instance.table];
if (!tableCache)
tableCache = this.caches[instance.table] = {};
if (!tableCache[instance.id])
tableCache[instance.id] = instance;
}
static forget(instance) {
if (instance.id === undefined)
throw new Error('Cannot forget an instance with an undefined id.');
let tableCache = this.caches[instance.table];
if (!tableCache)
return;
if (tableCache[instance.id])
delete tableCache[instance.id];
}
static all(table) {
return this.caches[table];
}
static get(table, id) {
const tableCache = this.all(table);
if (!tableCache)
return undefined;
return tableCache[id];
}
}
ModelCache.caches = {};
export const EMAIL_REGEX = /^[a-zA-Z0-9.!#$%&'*+\\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+$/;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Model.js","sourceRoot":"./","sources":["db/Model.ts"],"names":[],"mappings":"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","sourcesContent":["import MysqlConnectionManager, {query} from \"./MysqlConnectionManager\";\nimport Validator from \"./Validator\";\nimport {Connection} from \"mysql\";\nimport Query from \"./Query\";\nimport {Request} from \"express\";\nimport Pagination from \"../Pagination\";\n\nexport default abstract class Model {\n    public static async getById<T extends Model>(id: number): Promise<T | null> {\n        const cachedModel = ModelCache.get(this.table, id);\n        if (cachedModel?.constructor === this) {\n            return <T>cachedModel;\n        }\n\n        const models = await this.models<T>(this.select().where('id', id).first());\n        return models.length > 0 ? models[0] : null;\n    }\n\n    public static async paginate<T extends Model>(request: Request, perPage: number = 20): Promise<T[]> {\n        let page = request.params.page ? parseInt(request.params.page) : 1;\n        let query: Query = this.select().limit(perPage, (page - 1) * perPage).withTotalRowCount();\n        if (request.params.sortBy) {\n            const dir = request.params.sortDirection;\n            query = query.sortBy(request.params.sortBy, dir === 'ASC' || dir === 'DESC' ? dir : undefined);\n        } else {\n            query = query.sortBy('id');\n        }\n        const models = await this.models<T>(query);\n        // @ts-ignore\n        models.pagination = new Pagination(models, page, perPage, models.totalCount);\n        return models;\n    }\n\n    protected static select(...fields: string[]): Query {\n        return Query.select(this.table, ...fields);\n    }\n\n    protected static update(data: { [key: string]: any }): Query {\n        return Query.update(this.table, data);\n    }\n\n    protected static delete(): Query {\n        return Query.delete(this.table);\n    }\n\n    protected static async models<T extends Model>(query: Query): Promise<T[]> {\n        const results = await query.execute();\n        const models: T[] = [];\n        const factory = this.getFactory<T>();\n        for (const result of results.results) {\n            const cachedModel = ModelCache.get(this.table, result.id);\n            if (cachedModel && cachedModel.constructor === this) {\n                cachedModel.updateWithData(result);\n                models.push(<T>cachedModel);\n            } else {\n                models.push(factory(result));\n            }\n        }\n        // @ts-ignore\n        models.totalCount = results.foundRows;\n        return models;\n    }\n\n    public static async loadRelation<T extends Model>(models: T[], relation: string, model: Function, localField: string) {\n        const loadMap: { [p: number]: (model: T) => void } = {};\n        const ids = models.map(m => {\n            m.relations[relation] = null;\n            if (m[localField]) loadMap[m[localField]] = v => m.relations[relation] = v;\n            return m[localField];\n        }).filter(id => id);\n        for (const v of await (<any>model).models((<any>model).select().whereIn('id', ids))) {\n            loadMap[v.id!](v);\n        }\n    }\n\n    private static getFactory<T extends Model>(factory?: ModelFactory<T>): ModelFactory<T> {\n        if (factory === undefined) {\n            factory = (<any>this).FACTORY;\n            if (factory === undefined) factory = data => new (<any>this)(data);\n        }\n        return factory;\n    }\n\n\n    protected readonly properties: ModelProperty<any>[] = [];\n    private readonly relations: { [p: string]: (Model | null) } = {};\n    public id?: number;\n\n    [key: string]: any;\n\n    public constructor(data: any) {\n        this.defineProperty<number>('id', new Validator());\n        this.defineProperties();\n        this.updateWithData(data);\n    }\n\n    protected abstract defineProperties(): void;\n\n    protected defineProperty<T>(name: string, validator?: Validator<T> | RegExp) {\n        if (validator === undefined) validator = new Validator();\n        if (validator instanceof RegExp) {\n            const regexp = validator;\n            validator = new Validator().regexp(regexp);\n        }\n\n        const prop = new ModelProperty<T>(name, validator);\n        this.properties.push(prop);\n        Object.defineProperty(this, name, {\n            get: () => prop.value,\n            set: (value: T) => prop.value = value,\n        });\n    }\n\n    private updateWithData(data: any) {\n        this.id = data['id'];\n\n        for (const prop of this.properties) {\n            if (data[prop.name] !== undefined) {\n                this[prop.name] = data[prop.name];\n            }\n        }\n    }\n\n    protected async beforeSave(exists: boolean, connection: Connection): Promise<void> {\n    }\n\n    protected async afterSave(): Promise<void> {\n    }\n\n    public async save(connection?: Connection, postHook?: (callback: () => Promise<void>) => void): Promise<void> {\n        await this.validate(false, connection);\n\n        const exists = await this.exists();\n        let needs_full_update = false;\n\n        if (connection) {\n            needs_full_update = await this.saveTransaction(connection, exists, needs_full_update);\n        } else {\n            needs_full_update = await MysqlConnectionManager.wrapTransaction(async connection => this.saveTransaction(connection, exists, needs_full_update));\n        }\n\n        const callback = async () => {\n            if (needs_full_update) {\n                this.updateWithData((await (<Model><unknown>this.constructor).select().where('id', this.id!).first().execute()).results[0]);\n            }\n\n            if (!exists) {\n                this.cache();\n            }\n\n            await this.afterSave();\n        };\n\n        if (connection) {\n            postHook!(callback);\n        } else {\n            await callback();\n        }\n    }\n\n    private async saveTransaction(connection: Connection, exists: boolean, needs_full_update: boolean): Promise<boolean> {\n        // Before save\n        await this.beforeSave(exists, connection);\n        if (exists && this.hasOwnProperty('updated_at')) {\n            this.updated_at = new Date();\n        }\n\n        const props = [];\n        const values = [];\n\n        if (exists) {\n            for (const prop of this.properties) {\n                if (prop.value !== undefined) {\n                    props.push(prop.name + '=?');\n                    values.push(prop.value);\n                } else {\n                    needs_full_update = true;\n                }\n            }\n            values.push(this.id);\n            await query(`UPDATE ${this.table} SET ${props.join(',')} WHERE id=?`, values, connection);\n        } else {\n            const props_holders = [];\n            for (const prop of this.properties) {\n                if (prop.value !== undefined) {\n                    props.push(prop.name);\n                    props_holders.push('?');\n                    values.push(prop.value);\n                } else {\n                    needs_full_update = true;\n                }\n            }\n            const result = await query(`INSERT INTO ${this.table} (${props.join(', ')}) VALUES(${props_holders.join(', ')})`, values, connection);\n\n            this.id = result.other.insertId;\n        }\n\n        return needs_full_update;\n    }\n\n    public static get table(): string {\n        return this.name\n                .replace(/(?:^|\\.?)([A-Z])/g, (x, y) => '_' + y.toLowerCase())\n                .replace(/^_/, '')\n            + 's';\n    }\n\n    public get table(): string {\n        // @ts-ignore\n        return this.constructor.table;\n    }\n\n    public async exists(): Promise<boolean> {\n        if (!this.id) return false;\n\n        const result = await query(`SELECT 1 FROM ${this.table} WHERE id=? LIMIT 1`, [\n            this.id,\n        ]);\n        return result.results.length > 0;\n    }\n\n    public async delete(): Promise<void> {\n        if (!(await this.exists())) throw new Error('This model instance doesn\\'t exist in DB.');\n\n        await query(`DELETE FROM ${this.table} WHERE id=?`, [\n            this.id,\n        ]);\n        ModelCache.forget(this);\n        this.id = undefined;\n    }\n\n    public async validate(onlyFormat: boolean = false, connection?: Connection): Promise<void[]> {\n        return await Promise.all(this.properties.map(prop => prop.validate(onlyFormat, connection)));\n    }\n\n    private cache() {\n        ModelCache.cache(this);\n    }\n\n    protected relation<T extends Model>(name: string): T | null {\n        if (this.relations[name] === undefined) throw new Error('Model not loaded');\n        return <T | null>this.relations[name];\n    }\n}\n\nexport interface ModelFactory<T extends Model> {\n    (data: any): T;\n}\n\nclass ModelProperty<T> {\n    public readonly name: string;\n    private readonly validator: Validator<T>;\n    private val?: T;\n\n    constructor(name: string, validator: Validator<T>) {\n        this.name = name;\n        this.validator = validator;\n    }\n\n    public async validate(onlyFormat: boolean, connection?: Connection): Promise<void> {\n        return await this.validator.execute(this.name, this.value, onlyFormat, connection);\n    }\n\n    public get value(): T | undefined {\n        return this.val;\n    }\n\n    public set value(val: T | undefined) {\n        this.val = val;\n    }\n}\n\nexport class ModelCache {\n    private static readonly caches: {\n        [key: string]: {\n            [key: number]: Model\n        }\n    } = {};\n\n    public static cache(instance: Model) {\n        if (instance.id === undefined) throw new Error('Cannot cache an instance with an undefined id.');\n\n        let tableCache = this.caches[instance.table];\n        if (!tableCache) tableCache = this.caches[instance.table] = {};\n\n        if (!tableCache[instance.id]) tableCache[instance.id] = instance;\n    }\n\n    public static forget(instance: Model) {\n        if (instance.id === undefined) throw new Error('Cannot forget an instance with an undefined id.');\n\n        let tableCache = this.caches[instance.table];\n        if (!tableCache) return;\n\n        if (tableCache[instance.id]) delete tableCache[instance.id];\n    }\n\n    public static all(table: string): {\n        [key: number]: Model\n    } | undefined {\n        return this.caches[table];\n    }\n\n    public static get(table: string, id: number): Model | undefined {\n        const tableCache = this.all(table);\n        if (!tableCache) return undefined;\n        return tableCache[id];\n    }\n}\n\nexport const EMAIL_REGEX = /^[a-zA-Z0-9.!#$%&'*+\\\\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+$/;"]}