// (C) Copyright 2015 Moodle Pty Ltd. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. import { SQLiteDB, SQLiteDBRecordValue, SQLiteDBRecordValues } from '@classes/sqlitedb'; /** * Wrapper used to interact with a database table. */ export class CoreDatabaseTable< DBRecord extends SQLiteDBRecordValues = SQLiteDBRecordValues, PrimaryKeyColumn extends keyof DBRecord = 'id', PrimaryKey extends GetDBRecordPrimaryKey = GetDBRecordPrimaryKey > { protected database: SQLiteDB; protected tableName: string; protected primaryKeyColumns: PrimaryKeyColumn[]; constructor(database: SQLiteDB, tableName: string, primaryKeyColumns?: PrimaryKeyColumn[]) { this.database = database; this.tableName = tableName; this.primaryKeyColumns = primaryKeyColumns ?? ['id'] as PrimaryKeyColumn[]; } /** * Get database connection. * * @returns Database connection. */ getDatabase(): SQLiteDB { return this.database; } /** * Get table name. * * @returns Table name. */ getTableName(): string { return this.tableName; } /** * Get primary key columns. * * @returns Primary key columns. */ getPrimaryKeyColumns(): PrimaryKeyColumn[] { return this.primaryKeyColumns.slice(0); } /** * Initialize. */ async initialize(): Promise { // Nothing to initialize by default, override this method if necessary. } /** * Destroy. */ async destroy(): Promise { // Nothing to destroy by default, override this method if necessary. } /** * Get records matching the given conditions. * * @param conditions Matching conditions. If this argument is missing, all records in the table will be returned. * @returns Database records. */ getMany(conditions?: Partial): Promise { return conditions ? this.database.getRecords(this.tableName, conditions) : this.database.getAllRecords(this.tableName); } /** * Find one record matching the given conditions. * * @param conditions Matching conditions. * @returns Database record. */ getOne(conditions: Partial): Promise { return this.database.getRecord(this.tableName, conditions); } /** * Find one record by its primary key. * * @param primaryKey Primary key. * @returns Database record. */ getOneByPrimaryKey(primaryKey: PrimaryKey): Promise { return this.database.getRecord(this.tableName, primaryKey); } /** * Reduce some records into a single value. * * @param reducer Reducer functions in SQL and JavaScript. * @param conditions Matching conditions in SQL and JavaScript. If this argument is missing, all records in the table * will be used. * @returns Reduced value. */ reduce(reducer: CoreDatabaseReducer, conditions?: CoreDatabaseConditions): Promise { return this.database.getFieldSql( `SELECT ${reducer.sql} FROM ${this.tableName} ${conditions?.sql ?? ''}`, conditions?.sqlParams, ) as unknown as Promise; } /** * Insert a new record. * * @param record Database record. */ async insert(record: DBRecord): Promise { await this.database.insertRecord(this.tableName, record); } /** * Update records matching the given conditions. * * @param updates Record updates. * @param conditions Matching conditions. If this argument is missing, all records will be updated. */ async update(updates: Partial, conditions?: Partial): Promise { await this.database.updateRecords(this.tableName, updates, conditions); } /** * Update records matching the given conditions. * * This method should be used when it's necessary to apply complex conditions; the simple `update` * method should be favored otherwise for better performance. * * @param updates Record updates. * @param conditions Matching conditions in SQL and JavaScript. */ async updateWhere(updates: Partial, conditions: CoreDatabaseConditions): Promise { await this.database.updateRecordsWhere(this.tableName, updates, conditions.sql, conditions.sqlParams); } /** * Delete records matching the given conditions. * * @param conditions Matching conditions. If this argument is missing, all records will be deleted. */ async delete(conditions?: Partial): Promise { conditions ? await this.database.deleteRecords(this.tableName, conditions) : await this.database.deleteRecords(this.tableName); } /** * Delete a single record identified by its primary key. * * @param primaryKey Record primary key. */ async deleteByPrimaryKey(primaryKey: PrimaryKey): Promise { await this.database.deleteRecords(this.tableName, primaryKey); } /** * Get the primary key from a database record. * * @param record Database record. * @returns Primary key. */ protected getPrimaryKeyFromRecord(record: DBRecord): PrimaryKey { return this.primaryKeyColumns.reduce((primaryKey, column) => { primaryKey[column] = record[column]; return primaryKey; }, {} as Record) as PrimaryKey; } /** * Serialize a primary key with a string representation. * * @param primaryKey Primary key. * @returns Serialized primary key. */ protected serializePrimaryKey(primaryKey: PrimaryKey): string { return Object.values(primaryKey).map(value => String(value)).join('-'); } /** * Check whether a given record matches the given conditions. * * @param record Database record. * @param conditions Matching conditions. * @returns Whether the record matches the conditions. */ protected recordMatches(record: DBRecord, conditions: Partial): boolean { return !Object.entries(conditions).some(([column, value]) => record[column] !== value); } } /** * Infer primary key type from database record and primary key column types. */ export type GetDBRecordPrimaryKey = { [column in PrimaryKeyColumn]: DBRecord[column]; }; /** * Reducer used to accumulate a value from multiple records both in SQL and JavaScript. * * Both operations should be equivalent. */ export type CoreDatabaseReducer = { sql: string; js: (previousValue: T, record: DBRecord) => T; jsInitialValue: T; }; /** * Conditions to match database records both in SQL and JavaScript. * * Both conditions should be equivalent. */ export type CoreDatabaseConditions = { sql: string; sqlParams?: SQLiteDBRecordValue[]; js: (record: DBRecord) => boolean; };