swaf/dist/db/Query.js
2020-04-22 17:49:58 +02:00

176 lines
23 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 { 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}"]}