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 { query } from "./MysqlConnectionManager"; export default class Query { constructor(type, table, fields) { this._where = []; this._foundRows = false; this.type = type; this.table = table; this.fields = fields || []; } static select(table, ...fields) { return new Query(QueryType.SELECT, table, fields.length > 0 ? fields : ['*']); } static update(table, data) { const fields = []; for (let key in data) { if (data.hasOwnProperty(key)) { fields.push(new UpdateFieldValue(key, data[key])); } } return new Query(QueryType.UPDATE, table, fields); } static delete(table) { return new Query(QueryType.DELETE, table); } where(field, value, operator = WhereOperator.AND, test = WhereTest.EQUALS) { this._where.push(new WhereFieldValue(field, value, operator, test)); return this; } whereNot(field, value, operator = WhereOperator.AND) { return this.where(field, value, operator, WhereTest.DIFFERENT); } orWhere(field, value) { return this.where(field, value, WhereOperator.OR); } whereIn(field, value) { return this.where(field, value, WhereOperator.AND, WhereTest.IN); } limit(limit, offset = 0) { this._limit = limit; this._offset = offset; return this; } first() { return this.limit(1); } sortBy(field, direction = 'ASC') { this._sortBy = field; this._sortDirection = direction; return this; } withTotalRowCount() { this._foundRows = true; return this; } toString(final = false) { var _a; let query = ''; let fields = (_a = this.fields) === null || _a === void 0 ? void 0 : _a.join(','); let where = ''; if (this._where.length > 0) { where = `WHERE ${this._where[0]}`; for (let i = 1; i < this._where.length; i++) { where += this._where[i].toString(false); } } let limit = ''; if (typeof this._limit === 'number') { limit = `LIMIT ${this._limit}`; if (typeof this._offset === 'number' && this._offset !== 0) { limit += ` OFFSET ${this._offset}`; } } let orderBy = ''; if (typeof this._sortBy === 'string') { orderBy = `ORDER BY ${this._sortBy} ${this._sortDirection}`; } switch (this.type) { case QueryType.SELECT: query = `SELECT ${this._foundRows ? 'SQL_CALC_FOUND_ROWS' : ''} ${fields} FROM ${this.table} ${where} ${orderBy} ${limit}`; break; case QueryType.UPDATE: query = `UPDATE ${this.table} SET ${fields} ${where} ${orderBy} ${limit}`; break; case QueryType.DELETE: query = `DELETE FROM ${this.table} ${where} ${orderBy} ${limit}`; break; } return final ? query : `(${query})`; } build() { return this.toString(true); } get variables() { var _a; const variables = []; (_a = this.fields) === null || _a === void 0 ? void 0 : _a.filter(v => v instanceof FieldValue).flatMap(v => v.variables).forEach(v => variables.push(v)); this._where.flatMap(v => v.variables) .forEach(v => variables.push(v)); return variables; } isCacheable() { return this.type === QueryType.SELECT && this.fields.length === 1 && this.fields[0] === '*'; } execute(connection) { return __awaiter(this, void 0, void 0, function* () { const queryResult = yield query(this.build(), this.variables, connection); if (this._foundRows) { const foundRows = yield query('SELECT FOUND_ROWS() as r', undefined, connection); queryResult.foundRows = foundRows.results[0].r; } return queryResult; }); } } export var QueryType; (function (QueryType) { QueryType[QueryType["SELECT"] = 0] = "SELECT"; QueryType[QueryType["UPDATE"] = 1] = "UPDATE"; QueryType[QueryType["DELETE"] = 2] = "DELETE"; })(QueryType || (QueryType = {})); var WhereOperator; (function (WhereOperator) { WhereOperator["AND"] = "AND"; WhereOperator["OR"] = "OR"; })(WhereOperator || (WhereOperator = {})); var WhereTest; (function (WhereTest) { WhereTest["EQUALS"] = "="; WhereTest["DIFFERENT"] = "!="; WhereTest["IN"] = " IN "; })(WhereTest || (WhereTest = {})); class FieldValue { constructor(field, value) { this.field = field; this.value = value; } toString(first = true) { return `${!first ? ',' : ''}${this.field}${this.test}${this.value instanceof Query ? this.value : (Array.isArray(this.value) ? '(?)' : '?')}`; } get test() { return '='; } get variables() { return this.value instanceof Query ? this.value.variables : [this.value]; } } class SelectFieldValue extends FieldValue { toString(first = true) { return `(${this.value instanceof Query ? this.value : '?'}) AS ${this.field}`; } } class UpdateFieldValue extends FieldValue { } class WhereFieldValue extends FieldValue { constructor(field, value, operator, test) { super(field, value); this.operator = operator; this._test = test; } toString(first = true) { return (!first ? ` ${this.operator} ` : '') + super.toString(true); } get test() { return this._test; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Query.js","sourceRoot":"./","sources":["db/Query.ts"],"names":[],"mappings":"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","sourcesContent":["import {query, QueryResult} from \"./MysqlConnectionManager\";\nimport {Connection} from \"mysql\";\n\nexport default class Query {\n    public static select(table: string, ...fields: string[]): Query {\n        return new Query(QueryType.SELECT, table, fields.length > 0 ? fields : ['*']);\n    }\n\n    public static update(table: string, data: {\n        [key: string]: any\n    }) {\n        const fields = [];\n        for (let key in data) {\n            if (data.hasOwnProperty(key)) {\n                fields.push(new UpdateFieldValue(key, data[key]));\n            }\n        }\n        return new Query(QueryType.UPDATE, table, fields);\n    }\n\n    public static delete(table: string) {\n        return new Query(QueryType.DELETE, table);\n    }\n\n    private readonly type: QueryType;\n    private readonly table: string;\n    private readonly fields: (string | SelectFieldValue | UpdateFieldValue)[];\n    private _where: WhereFieldValue[] = [];\n    private _limit?: number;\n    private _offset?: number;\n    private _sortBy?: string;\n    private _sortDirection?: 'ASC' | 'DESC';\n    private _foundRows: boolean = false;\n\n    private constructor(type: QueryType, table: string, fields?: (string | SelectFieldValue | UpdateFieldValue)[]) {\n        this.type = type;\n        this.table = table;\n        this.fields = fields || [];\n    }\n\n    public where(field: string, value: string | Date | Query | any, operator: WhereOperator = WhereOperator.AND, test: WhereTest = WhereTest.EQUALS): Query {\n        this._where.push(new WhereFieldValue(field, value, operator, test));\n        return this;\n    }\n\n    public whereNot(field: string, value: string | Date | Query | any, operator: WhereOperator = WhereOperator.AND): Query {\n        return this.where(field, value, operator, WhereTest.DIFFERENT);\n    }\n\n    public orWhere(field: string, value: string | Date | Query | any): Query {\n        return this.where(field, value, WhereOperator.OR);\n    }\n\n    public whereIn(field: string, value: any[]): Query {\n        return this.where(field, value, WhereOperator.AND, WhereTest.IN);\n    }\n\n    public limit(limit: number, offset: number = 0): Query {\n        this._limit = limit;\n        this._offset = offset;\n        return this;\n    }\n\n    public first(): Query {\n        return this.limit(1);\n    }\n\n    public sortBy(field: string, direction: 'ASC' | 'DESC' = 'ASC'): Query {\n        this._sortBy = field;\n        this._sortDirection = direction;\n        return this;\n    }\n\n    public withTotalRowCount(): Query {\n        this._foundRows = true;\n        return this;\n    }\n\n    public toString(final: boolean = false): string {\n        let query = '';\n\n        let fields = this.fields?.join(',');\n\n        let where = '';\n        if (this._where.length > 0) {\n            where = `WHERE ${this._where[0]}`;\n            for (let i = 1; i < this._where.length; i++) {\n                where += this._where[i].toString(false);\n            }\n        }\n\n        let limit = '';\n        if (typeof this._limit === 'number') {\n            limit = `LIMIT ${this._limit}`;\n            if (typeof this._offset === 'number' && this._offset !== 0) {\n                limit += ` OFFSET ${this._offset}`;\n            }\n        }\n\n        let orderBy = '';\n        if (typeof this._sortBy === 'string') {\n            orderBy = `ORDER BY ${this._sortBy} ${this._sortDirection}`;\n        }\n\n        switch (this.type) {\n            case QueryType.SELECT:\n                query = `SELECT ${this._foundRows ? 'SQL_CALC_FOUND_ROWS' : ''} ${fields} FROM ${this.table} ${where} ${orderBy} ${limit}`;\n                break;\n            case QueryType.UPDATE:\n                query = `UPDATE ${this.table} SET ${fields} ${where} ${orderBy} ${limit}`;\n                break;\n            case QueryType.DELETE:\n                query = `DELETE FROM ${this.table} ${where} ${orderBy} ${limit}`;\n                break;\n\n        }\n\n        return final ? query : `(${query})`;\n    }\n\n    public build(): string {\n        return this.toString(true);\n    }\n\n    public get variables(): any[] {\n        const variables: any[] = [];\n        this.fields?.filter(v => v instanceof FieldValue)\n            .flatMap(v => (<FieldValue>v).variables)\n            .forEach(v => variables.push(v));\n        this._where.flatMap(v => v.variables)\n            .forEach(v => variables.push(v));\n        return variables;\n    }\n\n    public isCacheable(): boolean {\n        return this.type === QueryType.SELECT && this.fields.length === 1 && this.fields[0] === '*';\n    }\n\n    public async execute(connection?: Connection): Promise<QueryResult> {\n        const queryResult = await query(this.build(), this.variables, connection);\n        if (this._foundRows) {\n            const foundRows = await query('SELECT FOUND_ROWS() as r', undefined, connection);\n            queryResult.foundRows = foundRows.results[0].r;\n        }\n        return queryResult;\n    }\n}\n\nexport enum QueryType {\n    SELECT,\n    UPDATE,\n    DELETE,\n}\n\nenum WhereOperator {\n    AND = 'AND',\n    OR = 'OR',\n}\n\nenum WhereTest {\n    EQUALS = '=',\n    DIFFERENT = '!=',\n    IN = ' IN ',\n}\n\nclass FieldValue {\n    protected readonly field: string;\n    protected value: any;\n\n    constructor(field: string, value: any) {\n        this.field = field;\n        this.value = value;\n    }\n\n    public toString(first: boolean = true): string {\n        return `${!first ? ',' : ''}${this.field}${this.test}${this.value instanceof Query ? this.value : (Array.isArray(this.value) ? '(?)' : '?')}`;\n    }\n\n    protected get test(): string {\n        return '=';\n    }\n\n    public get variables(): any[] {\n        return this.value instanceof Query ? this.value.variables : [this.value];\n    }\n}\n\nclass SelectFieldValue extends FieldValue {\n    public toString(first: boolean = true): string {\n        return `(${this.value instanceof Query ? this.value : '?'}) AS ${this.field}`;\n    }\n}\n\nclass UpdateFieldValue extends FieldValue {\n}\n\nclass WhereFieldValue extends FieldValue {\n    private readonly operator: WhereOperator;\n    private readonly _test: WhereTest;\n\n    constructor(field: string, value: any, operator: WhereOperator, test: WhereTest) {\n        super(field, value);\n        this.operator = operator;\n        this._test = test;\n    }\n\n    public toString(first: boolean = true): string {\n        return (!first ? ` ${this.operator} ` : '') + super.toString(true);\n    }\n\n    protected get test(): string {\n        return this._test;\n    }\n}"]}