@simonbackx/simple-database 1.35.0 → 1.36.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.cjs +9 -0
- package/dist/_virtual/_@oxc-project_runtime@0.115.0/helpers/decorate.mjs +9 -0
- package/dist/_virtual/_rolldown/runtime.cjs +23 -0
- package/dist/index.cjs +22 -0
- package/dist/index.d.cts +10 -0
- package/dist/index.d.mts +10 -0
- package/dist/index.mjs +10 -0
- package/dist/src/classes/Column.cjs +134 -0
- package/dist/src/classes/Column.d.cts +34 -0
- package/dist/src/classes/Column.d.cts.map +1 -0
- package/dist/src/classes/Column.d.mts +34 -0
- package/dist/src/classes/Column.d.mts.map +1 -0
- package/dist/src/classes/Column.mjs +135 -0
- package/dist/src/classes/Column.mjs.map +1 -0
- package/dist/src/classes/ColumnType.d.cts +5 -0
- package/dist/src/classes/ColumnType.d.cts.map +1 -0
- package/dist/src/classes/ColumnType.d.mts +5 -0
- package/dist/src/classes/ColumnType.d.mts.map +1 -0
- package/dist/src/classes/Database.cjs +157 -0
- package/dist/src/classes/Database.d.cts +59 -0
- package/dist/src/classes/Database.d.cts.map +1 -0
- package/dist/src/classes/Database.d.mts +59 -0
- package/dist/src/classes/Database.d.mts.map +1 -0
- package/dist/src/classes/Database.mjs +155 -0
- package/dist/src/classes/Database.mjs.map +1 -0
- package/dist/src/classes/DatabaseStoredValue.d.cts +5 -0
- package/dist/src/classes/DatabaseStoredValue.d.cts.map +1 -0
- package/dist/src/classes/DatabaseStoredValue.d.mts +5 -0
- package/dist/src/classes/DatabaseStoredValue.d.mts.map +1 -0
- package/dist/src/classes/Factory.cjs +48 -0
- package/dist/src/classes/Factory.d.cts +17 -0
- package/dist/src/classes/Factory.d.cts.map +1 -0
- package/dist/src/classes/Factory.d.mts +17 -0
- package/dist/src/classes/Factory.d.mts.map +1 -0
- package/dist/src/classes/Factory.mjs +50 -0
- package/dist/src/classes/Factory.mjs.map +1 -0
- package/dist/src/classes/ManyToManyRelation.cjs +193 -0
- package/dist/src/classes/ManyToManyRelation.d.cts +83 -0
- package/dist/src/classes/ManyToManyRelation.d.cts.map +1 -0
- package/dist/src/classes/ManyToManyRelation.d.mts +83 -0
- package/dist/src/classes/ManyToManyRelation.d.mts.map +1 -0
- package/dist/src/classes/ManyToManyRelation.mjs +195 -0
- package/dist/src/classes/ManyToManyRelation.mjs.map +1 -0
- package/dist/src/classes/ManyToOneRelation.cjs +41 -0
- package/dist/src/classes/ManyToOneRelation.d.cts +26 -0
- package/dist/src/classes/ManyToOneRelation.d.cts.map +1 -0
- package/dist/src/classes/ManyToOneRelation.d.mts +26 -0
- package/dist/src/classes/ManyToOneRelation.d.mts.map +1 -0
- package/dist/src/classes/ManyToOneRelation.mjs +43 -0
- package/dist/src/classes/ManyToOneRelation.mjs.map +1 -0
- package/dist/src/classes/Migration.cjs +123 -0
- package/dist/src/classes/Migration.d.cts +16 -0
- package/dist/src/classes/Migration.d.cts.map +1 -0
- package/dist/src/classes/Migration.d.mts +16 -0
- package/dist/src/classes/Migration.d.mts.map +1 -0
- package/dist/src/classes/Migration.mjs +123 -0
- package/dist/src/classes/Migration.mjs.map +1 -0
- package/dist/src/classes/Model.cjs +459 -0
- package/dist/src/classes/Model.d.cts +162 -0
- package/dist/src/classes/Model.d.cts.map +1 -0
- package/dist/src/classes/Model.d.mts +162 -0
- package/dist/src/classes/Model.d.mts.map +1 -0
- package/dist/src/classes/Model.mjs +460 -0
- package/dist/src/classes/Model.mjs.map +1 -0
- package/dist/src/classes/OneToManyRelation.cjs +66 -0
- package/dist/src/classes/OneToManyRelation.d.cts +42 -0
- package/dist/src/classes/OneToManyRelation.d.cts.map +1 -0
- package/dist/src/classes/OneToManyRelation.d.mts +42 -0
- package/dist/src/classes/OneToManyRelation.d.mts.map +1 -0
- package/dist/src/classes/OneToManyRelation.mjs +68 -0
- package/dist/src/classes/OneToManyRelation.mjs.map +1 -0
- package/dist/src/classes/data/boys.cjs +1005 -0
- package/dist/src/classes/data/boys.mjs +1007 -0
- package/dist/src/classes/data/boys.mjs.map +1 -0
- package/dist/src/classes/data/family-names.cjs +1005 -0
- package/dist/src/classes/data/family-names.mjs +1007 -0
- package/dist/src/classes/data/family-names.mjs.map +1 -0
- package/dist/src/classes/data/girls.cjs +1004 -0
- package/dist/src/classes/data/girls.mjs +1006 -0
- package/dist/src/classes/data/girls.mjs.map +1 -0
- package/dist/src/decorators/Column.cjs +26 -0
- package/dist/src/decorators/Column.d.cts +22 -0
- package/dist/src/decorators/Column.d.cts.map +1 -0
- package/dist/src/decorators/Column.d.mts +22 -0
- package/dist/src/decorators/Column.d.mts.map +1 -0
- package/dist/src/decorators/Column.mjs +28 -0
- package/dist/src/decorators/Column.mjs.map +1 -0
- package/dist/src/models/Migration.cjs +37 -0
- package/dist/src/models/Migration.mjs +39 -0
- package/dist/src/models/Migration.mjs.map +1 -0
- package/package.json +16 -13
- package/dist/cjs/index.d.ts +0 -10
- package/dist/cjs/index.d.ts.map +0 -1
- package/dist/cjs/index.js +0 -13
- package/dist/cjs/index.js.map +0 -1
- package/dist/cjs/package.json +0 -1
- package/dist/cjs/src/classes/Column.d.ts +0 -30
- package/dist/cjs/src/classes/Column.d.ts.map +0 -1
- package/dist/cjs/src/classes/Column.js +0 -183
- package/dist/cjs/src/classes/Column.js.map +0 -1
- package/dist/cjs/src/classes/ColumnType.d.ts +0 -2
- package/dist/cjs/src/classes/ColumnType.d.ts.map +0 -1
- package/dist/cjs/src/classes/ColumnType.js +0 -3
- package/dist/cjs/src/classes/ColumnType.js.map +0 -1
- package/dist/cjs/src/classes/Database.d.ts +0 -59
- package/dist/cjs/src/classes/Database.d.ts.map +0 -1
- package/dist/cjs/src/classes/Database.js +0 -176
- package/dist/cjs/src/classes/Database.js.map +0 -1
- package/dist/cjs/src/classes/DatabaseStoredValue.d.ts +0 -2
- package/dist/cjs/src/classes/DatabaseStoredValue.d.ts.map +0 -1
- package/dist/cjs/src/classes/DatabaseStoredValue.js +0 -3
- package/dist/cjs/src/classes/DatabaseStoredValue.js.map +0 -1
- package/dist/cjs/src/classes/Factory.d.ts +0 -14
- package/dist/cjs/src/classes/Factory.d.ts.map +0 -1
- package/dist/cjs/src/classes/Factory.js +0 -56
- package/dist/cjs/src/classes/Factory.js.map +0 -1
- package/dist/cjs/src/classes/ManyToManyRelation.d.ts +0 -79
- package/dist/cjs/src/classes/ManyToManyRelation.d.ts.map +0 -1
- package/dist/cjs/src/classes/ManyToManyRelation.js +0 -258
- package/dist/cjs/src/classes/ManyToManyRelation.js.map +0 -1
- package/dist/cjs/src/classes/ManyToOneRelation.d.ts +0 -22
- package/dist/cjs/src/classes/ManyToOneRelation.d.ts.map +0 -1
- package/dist/cjs/src/classes/ManyToOneRelation.js +0 -51
- package/dist/cjs/src/classes/ManyToOneRelation.js.map +0 -1
- package/dist/cjs/src/classes/Migration.d.ts +0 -14
- package/dist/cjs/src/classes/Migration.d.ts.map +0 -1
- package/dist/cjs/src/classes/Migration.js +0 -206
- package/dist/cjs/src/classes/Migration.js.map +0 -1
- package/dist/cjs/src/classes/Model.d.ts +0 -159
- package/dist/cjs/src/classes/Model.d.ts.map +0 -1
- package/dist/cjs/src/classes/Model.js +0 -640
- package/dist/cjs/src/classes/Model.js.map +0 -1
- package/dist/cjs/src/classes/OneToManyRelation.d.ts +0 -38
- package/dist/cjs/src/classes/OneToManyRelation.d.ts.map +0 -1
- package/dist/cjs/src/classes/OneToManyRelation.js +0 -79
- package/dist/cjs/src/classes/OneToManyRelation.js.map +0 -1
- package/dist/cjs/src/classes/data/boys.d.ts +0 -3
- package/dist/cjs/src/classes/data/boys.d.ts.map +0 -1
- package/dist/cjs/src/classes/data/boys.js +0 -1005
- package/dist/cjs/src/classes/data/boys.js.map +0 -1
- package/dist/cjs/src/classes/data/family-names.d.ts +0 -3
- package/dist/cjs/src/classes/data/family-names.d.ts.map +0 -1
- package/dist/cjs/src/classes/data/family-names.js +0 -1005
- package/dist/cjs/src/classes/data/family-names.js.map +0 -1
- package/dist/cjs/src/classes/data/girls.d.ts +0 -3
- package/dist/cjs/src/classes/data/girls.d.ts.map +0 -1
- package/dist/cjs/src/classes/data/girls.js +0 -1004
- package/dist/cjs/src/classes/data/girls.js.map +0 -1
- package/dist/cjs/src/classes/data/streets.d.ts +0 -3
- package/dist/cjs/src/classes/data/streets.d.ts.map +0 -1
- package/dist/cjs/src/classes/data/streets.js +0 -296
- package/dist/cjs/src/classes/data/streets.js.map +0 -1
- package/dist/cjs/src/decorators/Column.d.ts +0 -18
- package/dist/cjs/src/decorators/Column.d.ts.map +0 -1
- package/dist/cjs/src/decorators/Column.js +0 -39
- package/dist/cjs/src/decorators/Column.js.map +0 -1
- package/dist/cjs/src/models/Migration.d.ts +0 -11
- package/dist/cjs/src/models/Migration.d.ts.map +0 -1
- package/dist/cjs/src/models/Migration.js +0 -52
- package/dist/cjs/src/models/Migration.js.map +0 -1
- package/dist/cjs/tsconfig.cjs.tsbuildinfo +0 -1
- package/dist/esm/index.d.ts +0 -10
- package/dist/esm/index.d.ts.map +0 -1
- package/dist/esm/index.js +0 -10
- package/dist/esm/index.js.map +0 -1
- package/dist/esm/src/classes/Column.d.ts +0 -30
- package/dist/esm/src/classes/Column.d.ts.map +0 -1
- package/dist/esm/src/classes/Column.js +0 -179
- package/dist/esm/src/classes/Column.js.map +0 -1
- package/dist/esm/src/classes/ColumnType.d.ts +0 -2
- package/dist/esm/src/classes/ColumnType.d.ts.map +0 -1
- package/dist/esm/src/classes/ColumnType.js +0 -2
- package/dist/esm/src/classes/ColumnType.js.map +0 -1
- package/dist/esm/src/classes/Database.d.ts +0 -59
- package/dist/esm/src/classes/Database.d.ts.map +0 -1
- package/dist/esm/src/classes/Database.js +0 -171
- package/dist/esm/src/classes/Database.js.map +0 -1
- package/dist/esm/src/classes/DatabaseStoredValue.d.ts +0 -2
- package/dist/esm/src/classes/DatabaseStoredValue.d.ts.map +0 -1
- package/dist/esm/src/classes/DatabaseStoredValue.js +0 -2
- package/dist/esm/src/classes/DatabaseStoredValue.js.map +0 -1
- package/dist/esm/src/classes/Factory.d.ts +0 -14
- package/dist/esm/src/classes/Factory.d.ts.map +0 -1
- package/dist/esm/src/classes/Factory.js +0 -51
- package/dist/esm/src/classes/Factory.js.map +0 -1
- package/dist/esm/src/classes/ManyToManyRelation.d.ts +0 -79
- package/dist/esm/src/classes/ManyToManyRelation.d.ts.map +0 -1
- package/dist/esm/src/classes/ManyToManyRelation.js +0 -254
- package/dist/esm/src/classes/ManyToManyRelation.js.map +0 -1
- package/dist/esm/src/classes/ManyToOneRelation.d.ts +0 -22
- package/dist/esm/src/classes/ManyToOneRelation.d.ts.map +0 -1
- package/dist/esm/src/classes/ManyToOneRelation.js +0 -47
- package/dist/esm/src/classes/ManyToOneRelation.js.map +0 -1
- package/dist/esm/src/classes/Migration.d.ts +0 -14
- package/dist/esm/src/classes/Migration.d.ts.map +0 -1
- package/dist/esm/src/classes/Migration.js +0 -168
- package/dist/esm/src/classes/Migration.js.map +0 -1
- package/dist/esm/src/classes/Model.d.ts +0 -159
- package/dist/esm/src/classes/Model.d.ts.map +0 -1
- package/dist/esm/src/classes/Model.js +0 -635
- package/dist/esm/src/classes/Model.js.map +0 -1
- package/dist/esm/src/classes/OneToManyRelation.d.ts +0 -38
- package/dist/esm/src/classes/OneToManyRelation.d.ts.map +0 -1
- package/dist/esm/src/classes/OneToManyRelation.js +0 -75
- package/dist/esm/src/classes/OneToManyRelation.js.map +0 -1
- package/dist/esm/src/classes/data/boys.d.ts +0 -3
- package/dist/esm/src/classes/data/boys.d.ts.map +0 -1
- package/dist/esm/src/classes/data/boys.js +0 -1003
- package/dist/esm/src/classes/data/boys.js.map +0 -1
- package/dist/esm/src/classes/data/family-names.d.ts +0 -3
- package/dist/esm/src/classes/data/family-names.d.ts.map +0 -1
- package/dist/esm/src/classes/data/family-names.js +0 -1003
- package/dist/esm/src/classes/data/family-names.js.map +0 -1
- package/dist/esm/src/classes/data/girls.d.ts +0 -3
- package/dist/esm/src/classes/data/girls.d.ts.map +0 -1
- package/dist/esm/src/classes/data/girls.js +0 -1002
- package/dist/esm/src/classes/data/girls.js.map +0 -1
- package/dist/esm/src/classes/data/streets.d.ts +0 -3
- package/dist/esm/src/classes/data/streets.d.ts.map +0 -1
- package/dist/esm/src/classes/data/streets.js +0 -294
- package/dist/esm/src/classes/data/streets.js.map +0 -1
- package/dist/esm/src/decorators/Column.d.ts +0 -18
- package/dist/esm/src/decorators/Column.d.ts.map +0 -1
- package/dist/esm/src/decorators/Column.js +0 -36
- package/dist/esm/src/decorators/Column.js.map +0 -1
- package/dist/esm/src/models/Migration.d.ts +0 -11
- package/dist/esm/src/models/Migration.d.ts.map +0 -1
- package/dist/esm/src/models/Migration.js +0 -48
- package/dist/esm/src/models/Migration.js.map +0 -1
- package/dist/esm/tsconfig.esm.tsbuildinfo +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Model.mjs","names":[],"sources":["../../../src/classes/Model.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-argument */\nimport { Column } from './Column.js';\nimport { Database } from './Database.js';\nimport { type DatabaseStoredValue } from './DatabaseStoredValue.js';\nimport { type ManyToManyRelation } from './ManyToManyRelation.js';\nimport { type ManyToOneRelation } from './ManyToOneRelation.js';\nimport { type OneToManyRelation } from './OneToManyRelation.js';\n\ntype SQLWhere = { sign: string; value: string | Date | number | null | (string | null)[] | (number | null)[]; mode?: string };\ntype SQLWhereQuery = { [key: string]: string | Date | number | null | SQLWhere | SQLWhere[] };\n\ntype Listener<Value> = (value: Value) => Promise<void> | void;\n\n/**\n * Controls the fetching and decrypting of members\n */\nexport class ModelEventBus<Value> {\n protected listeners: Map<any, { listener: Listener<Value> }[]> = new Map();\n\n addListener(owner: any, listener: Listener<Value>) {\n const existing = this.listeners.get(owner);\n if (existing) {\n existing.push({ listener });\n }\n else {\n this.listeners.set(owner, [{ listener }]);\n }\n }\n\n removeListener(owner: any) {\n this.listeners.delete(owner);\n }\n\n async sendEvent(value: Value) {\n const values: (Promise<void>)[] = [];\n for (const owner of this.listeners.values()) {\n for (const listener of owner) {\n const v = listener.listener(value);\n if (v) {\n values.push(v);\n }\n }\n }\n return await Promise.all(values);\n }\n}\n\nexport type ModelEventType = 'created' | 'updated' | 'deleted';\n\nexport type ModelEvent<M extends Model = Model> = {\n type: 'created';\n model: M;\n} | {\n type: 'updated';\n model: M;\n changedFields: Record<string, DatabaseStoredValue>;\n originalFields: Record<string, DatabaseStoredValue>;\n\n /**\n * Use this method to compare changes\n */\n getOldModel(): M;\n} | {\n type: 'deleted';\n model: M;\n};\n\nexport class Model /* static implements RowInitiable<Model> */ {\n static primary: Column;\n static modelEventBus = new ModelEventBus<ModelEvent>();\n\n /**\n * Properties that are stored in the table (including foreign keys, but without mapped relations!)\n */\n static columns: Map<string, Column>;\n static debug = false;\n static showWarnings = false;\n static table: string; // override this!\n static relations: ManyToOneRelation<string, Model>[];\n\n existsInDatabase = false;\n\n savedProperties = new Map<string, DatabaseStoredValue | ({\n to: () => DatabaseStoredValue;\n from: () => unknown;\n })>();\n\n /**\n * Sometimes we have skipUpdate properties that still should get saved on specific occasions.\n * E.g. update updatedAt field manually if is the only changed field.\n */\n forceSaveProperties = new Set<string>();\n\n constructor() {\n // Read values\n if (!this.static.relations) {\n this.static.relations = [];\n }\n\n if (!this.static.columns) {\n this.static.columns = new Map<string, Column>();\n }\n }\n\n /**\n * Delete the value of a key from memory\n */\n eraseProperty(key: string) {\n const column = this.static.columns.get(key);\n if (!column) {\n throw new Error('Unknown property ' + key);\n }\n delete this[key];\n this.savedProperties.delete(key);\n }\n\n /**\n * Make sure this key will get saved on the next save, even when it is not changed or when it is skipUpdate\n */\n forceSaveProperty(key: string) {\n this.forceSaveProperties.add(key);\n }\n\n /**\n * Mark all properties as changed, so they will get updated on the next save\n */\n markAllChanged() {\n this.savedProperties.clear();\n }\n\n /**\n * Returns the default select to select the needed properties of this table\n * @param namespace: optional namespace of this select\n */\n static getDefaultSelect(namespace: string = this.table): string {\n return '`' + namespace + '`.*';\n }\n\n static selectColumnsWithout(namespace: string = this.table, ...exclude: string[]): string {\n const properties = Array.from(this.columns.keys()).flatMap(name => (exclude.includes(name) ? [] : [name]));\n\n if (properties.length === 0) {\n // todo: check what to do in this case.\n throw new Error('Not implemented yet');\n }\n return '`' + namespace + '`.`' + properties.join('`, `' + namespace + '`.`') + '`';\n }\n\n /**\n * Set a relation to undefined, marking it as not loaded (so it won't get saved in the next save)\n * @param relation\n */\n unloadRelation<Key extends keyof any, Value extends Model>(\n this: this & Record<Key, Value>,\n relation: ManyToOneRelation<Key, any>,\n ): this & Record<Key, undefined> {\n // Todo: check if relation is nullable?\n const t = this as any;\n delete t[relation.modelKey];\n return t;\n }\n\n /**\n * Set a relation to null, deleting it on the next save (unless unloadRelation is called)\n * @param relation\n */\n unsetRelation<Key extends keyof any, Value extends Model>(\n this: this & Record<Key, Value>,\n relation: ManyToOneRelation<Key, any>,\n ): this & Record<Key, null> {\n // Todo: check if relation is nullable?\n return this.setOptionalRelation(relation, null);\n }\n\n setOptionalRelation<Key extends keyof any, Value extends Model>(\n relation: ManyToOneRelation<Key, Value>,\n value: Value | null,\n ): this & Record<Key, Value | null> {\n // Todo: check if relation is nullable?\n\n if (value !== null && !value.existsInDatabase) {\n throw new Error('You cannot set a relation to a model that are not yet saved in the database.');\n }\n const t = this as any;\n t[relation.modelKey] = value;\n t[relation.foreignKey] = value ? value.getPrimaryKey() : null;\n return t;\n }\n\n setRelation<Key extends keyof any, Value extends Model, V extends Value>(relation: ManyToOneRelation<Key, Value>, value: V): this & Record<Key, V> {\n if (!value.existsInDatabase) {\n throw new Error('You cannot set a relation to a model that are not yet saved in the database.');\n }\n const t = this as any;\n t[relation.modelKey] = value;\n t[relation.foreignKey] = value.getPrimaryKey();\n return t;\n }\n\n /**\n * Set a many relation. Note that this doesn't save the relation! You'll need to use the methods of the relation instead\n */\n setManyRelation<Key extends keyof any, Value extends Model>(\n relation: ManyToManyRelation<Key, any, Value, any> | OneToManyRelation<Key, any, Value>,\n value: Value[],\n ): this & Record<Key, Value[]> {\n value.forEach((v) => {\n if (!v.existsInDatabase) {\n throw new Error('You cannot set a relation to models that are not yet saved in the database.');\n }\n });\n const t = this as any;\n t[relation.modelKey] = value;\n return t;\n }\n\n /**\n * Set a many relation. Note that this doesn't save the relation! You'll need to use the methods of the relation instead\n */\n getManyRelation<Key extends keyof any, Value extends Model>(\n relation: ManyToManyRelation<Key, any, Value, any> | OneToManyRelation<Key, any, Value>,\n ): Value[] | null {\n const t = this as any;\n return t[relation.modelKey] ?? null;\n }\n\n /**\n * Load the returned properties from a DB response row into the model\n * If the row's primary key is null, undefined is returned\n */\n static fromRow<T extends typeof Model>(this: T, row: Record<string, DatabaseStoredValue>): InstanceType<T> | undefined {\n if (row === undefined || (this.primary && (row[this.primary.name] === null || row[this.primary.name] === undefined))) {\n return undefined;\n }\n\n const model = new this() as InstanceType<T>;\n for (const column of this.columns.values()) {\n if (row[column.name] !== undefined) {\n const value = column.from(row[column.name]);\n model[column.name] = value;\n }\n else {\n // Override default value to prevent any saving!\n model[column.name] = undefined;\n }\n }\n\n model.markSaved(row, { fromMySQL: true }); // note: we don't pass row because that value is unreliable for json values (mysql performs sorting and other opertions on keys)\n return model;\n }\n\n static fromRows<T extends typeof Model>(this: T, rows: Record<string, Record<string, DatabaseStoredValue>>[], namespace: string): InstanceType<T>[] {\n return rows.flatMap((row) => {\n const model = this.fromRow(row[namespace]);\n if (model) {\n return [model];\n }\n return [];\n });\n }\n\n markSaved(fields?: Record<string, DatabaseStoredValue>, options?: { fromMySQL?: boolean }) {\n this.existsInDatabase = true;\n this.forceSaveProperties.clear();\n\n for (const column of this.static.columns.values()) {\n if ((fields ? fields[column.name] : this[column.name]) !== undefined) {\n // If undefined: do not update, since we didn't save the value\n this.savedProperties.set(column.name, fields\n ? (column.type === 'json' && options?.fromMySQL\n ? {\n // We can never use the original value we received from MySQL because the ordering of keys is different in MySQL.\n // We use this object to avoid doing expensive CPU operations if we don't need to check if a value has changed\n // so we only evaluate this once we actually try to save a model to the database\n to() {\n if (this._to) {\n return this._to;\n }\n const value = this._from ?? column.from(fields[column.name]);\n this._from = value;\n\n const found = column.to(value);\n // Store permanently\n this._to = found;\n return found;\n },\n from() {\n if (this._from) {\n return this._from;\n }\n const from = column.from(fields[column.name]);\n this._from = from;\n return from;\n },\n }\n : fields[column.name]\n )\n : column.to(this[column.name]),\n );\n }\n }\n\n /// Save relation foreign keys (so we can check if the id has changed)\n for (const relation of this.static.relations) {\n if (relation.isLoaded(this)) {\n if (relation.isSet(this)) {\n const model = this[relation.modelKey];\n this[relation.foreignKey] = model.getPrimaryKey();\n }\n else {\n this[relation.foreignKey] = null;\n }\n }\n }\n }\n\n copyFrom<T extends Model>(this: T, from: T) {\n for (const column of this.static.columns.values()) {\n this[column.name] = from[column.name];\n }\n\n // Unload all relations\n for (const relation of this.static.relations) {\n if (relation.isLoaded(this)) {\n this[relation.modelKey] = undefined;\n }\n }\n\n this.savedProperties = from.savedProperties;\n this.forceSaveProperties = from.forceSaveProperties;\n this.existsInDatabase = from.existsInDatabase;\n }\n\n get static(): typeof Model {\n return this.constructor as typeof Model;\n }\n\n getPrimaryKey(): number | string | null {\n return this[this.static.primary.name];\n }\n\n /**\n * Get a model by its primary key\n * @param id primary key\n */\n static async getByID<T extends typeof Model>(this: T, id: number | string): Promise<InstanceType<T> | undefined> {\n const [rows] = await Database.select(`SELECT ${this.getDefaultSelect()} FROM \\`${this.table}\\` WHERE \\`${this.primary.name}\\` = ? LIMIT 1`, [id]);\n\n if (rows.length === 0) {\n return undefined;\n }\n\n // Read member + address from first row\n const row = rows[0][this.table];\n return this.fromRow(row);\n }\n\n /**\n * Get multiple models by their ID\n * @param ids primary key of the models you want to fetch\n */\n static async getByIDs<T extends typeof Model>(this: T, ...ids: (number | string)[]): Promise<InstanceType<T>[]> {\n if (ids.length === 0) {\n return [];\n }\n const [rows] = await Database.select(`SELECT ${this.getDefaultSelect()} FROM \\`${this.table}\\` WHERE \\`${this.primary.name}\\` IN (?) LIMIT ?`, [\n ids,\n ids.length,\n ]);\n\n // Read member + address from first row\n return this.fromRows(rows, this.table);\n }\n\n static buildWhereOperator(key: string, value: SQLWhere): [string, any[]] {\n let whereQuery: string;\n const params: any[] = [];\n\n switch (value.sign) {\n case 'MATCH': {\n let suffix = '';\n if (value.mode) {\n switch (value.mode) {\n case 'BOOLEAN':\n suffix = ' IN BOOLEAN MODE';\n break;\n default:\n throw new Error('Unknown match mode ' + value.mode);\n }\n }\n whereQuery = (`MATCH(\\`${this.table}\\`.\\`${key}\\`) AGAINST(?${suffix})`);\n params.push(value.value);\n break;\n }\n\n case 'LIKE':\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` LIKE ?`);\n params.push(value.value);\n break;\n\n case '!=':\n if (value.value === null) {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` IS NOT NULL`);\n }\n else {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` != ?`);\n params.push(value.value);\n }\n break;\n\n case '<':\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` < ?`);\n params.push(value.value);\n break;\n\n case '>':\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` > ?`);\n params.push(value.value);\n break;\n\n case '<=':\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` <= ?`);\n params.push(value.value);\n break;\n\n case '>=':\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` >= ?`);\n params.push(value.value);\n break;\n\n case '=':\n if (value.value === null) {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` IS NULL`);\n }\n else {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` = ?`);\n params.push(value.value);\n }\n break;\n\n case 'IN':\n if (!Array.isArray(value.value)) {\n throw new Error('Expected an array for IN where query');\n }\n if (value.value.includes(null)) {\n const filtered = (value.value as (string | number | null)[]).filter(v => v !== null);\n if (filtered.length === 0) {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` IS NULL`);\n }\n else if (filtered.length === 1) {\n // Special query\n whereQuery = (`(\\`${this.table}\\`.\\`${key}\\` = ? OR \\`${this.table}\\`.\\`${key}\\` IS NULL)`);\n params.push(filtered[0]);\n }\n else {\n // Special query\n whereQuery = (`(\\`${this.table}\\`.\\`${key}\\` IN (?) OR \\`${this.table}\\`.\\`${key}\\` IS NULL)`);\n params.push(filtered);\n }\n }\n else {\n whereQuery = (`\\`${this.table}\\`.\\`${key}\\` IN (?)`);\n params.push(value.value);\n }\n\n break;\n\n default:\n throw new Error('Sign not supported.');\n }\n\n return [whereQuery, params];\n }\n\n static buildWhereQuery(where: SQLWhereQuery): [string, any[]] {\n const whereQuery: string[] = [];\n const params: any[] = [];\n\n for (const key in where) {\n if (Object.hasOwnProperty.call(where, key)) {\n const value = where[key];\n if (Array.isArray(value)) {\n for (const v of value) {\n const [w, p] = this.buildWhereOperator(key, v);\n whereQuery.push(w);\n params.push(...p);\n }\n }\n else if (typeof value === 'object' && value !== null && !(value instanceof Date)) {\n const [w, p] = this.buildWhereOperator(key, value);\n whereQuery.push(w);\n params.push(...p);\n }\n else {\n if (value === null) {\n whereQuery.push(`\\`${this.table}\\`.\\`${key}\\` IS NULL`);\n }\n else {\n whereQuery.push(`\\`${this.table}\\`.\\`${key}\\` = ?`);\n params.push(value);\n }\n }\n }\n }\n\n return [whereQuery.join(' AND '), params];\n }\n\n /**\n * @deprecated Use the new SQL package instead\n * Get multiple models by a simple where\n */\n static async where<T extends typeof Model>(this: T, where: SQLWhereQuery, extra?: {\n limit?: number;\n sort?: (string | { column: string | SQLWhereQuery; direction?: 'ASC' | 'DESC' })[];\n select?: string;\n }): Promise<InstanceType<T>[]> {\n const params: any[] = [];\n\n let query = `SELECT ${extra?.select ?? this.getDefaultSelect()} FROM \\`${this.table}\\``;\n if (Object.keys(where).length !== 0) {\n const [whereQuery, p] = this.buildWhereQuery(where);\n query += `WHERE ` + whereQuery;\n params.push(...p);\n }\n\n if (extra && extra.sort !== undefined) {\n const sortQuery: string[] = [];\n\n for (const item of extra.sort) {\n if (typeof item === 'string') {\n sortQuery.push(Database.escapeId(item) + ' ASC');\n }\n else if (typeof item.column === 'string') {\n sortQuery.push(Database.escapeId(item.column) + ' ' + (item.direction ?? 'ASC'));\n }\n else {\n // Repeat use of a where query in the sorting (needed when fulltext searching and need to sort on the score, because limit won't work without this for an unknown reason)\n const [w, p] = this.buildWhereQuery(item.column);\n\n sortQuery.push('(' + w + ') ' + (item.direction ?? 'ASC'));\n params.push(...p);\n }\n }\n\n query += ` ORDER BY ` + sortQuery.join(', ');\n }\n\n if (extra && extra.limit !== undefined) {\n query += ` LIMIT ?`;\n params.push(extra.limit);\n }\n\n const [rows] = await Database.select(query, params);\n\n return this.fromRows(rows, this.table);\n }\n\n /**\n * Get multiple models by a simple where\n */\n static async all<T extends typeof Model>(this: T, limit?: number): Promise<InstanceType<T>[]> {\n const params: any[] = [];\n let query = `SELECT ${this.getDefaultSelect()} FROM \\`${this.table}\\``;\n\n if (limit) {\n query += ` LIMIT ?`;\n params.push(limit);\n }\n\n const [rows] = await Database.select(query, params);\n return this.fromRows(rows, this.table);\n }\n\n /**\n * Return an object of all the properties that are changed and their database representation\n */\n getChangedDatabaseProperties(): { fields: Record<string, DatabaseStoredValue>; skipUpdate: number } {\n const set: Record<string, DatabaseStoredValue> = {};\n let skipUpdate = 0;\n\n for (const column of this.static.columns.values()) {\n if (column.primary && column.type === 'integer' && this.static.primary.name === 'id') {\n // Auto increment: not allowed to set\n continue;\n }\n\n if (!this.existsInDatabase && this[column.name] === undefined) {\n // In the future we might make some columns optional because they have a default value in the database.\n // But that could cause inconsitent state, so it would be better to generate default values in code.\n throw new Error('Tried to create model ' + this.constructor.name + ' with undefined property ' + column.name);\n }\n\n if (this[column.name] !== undefined) {\n const forceSave = this.forceSaveProperties.has(column.name);\n const oldSavedValue = forceSave ? undefined : this.savedProperties.get(column.name);\n const saveValue = forceSave ? null : column.to(this[column.name]); // .to is expensive, so avoid calling it when not needed\n if (forceSave || oldSavedValue === undefined || column.isChanged(typeof oldSavedValue === 'object' && oldSavedValue !== null && 'to' in oldSavedValue ? oldSavedValue.to() : oldSavedValue, saveValue)) {\n set[column.name] = forceSave ? column.to(this[column.name]) : saveValue;\n\n if (column.skipUpdate && !forceSave) {\n skipUpdate++;\n }\n }\n }\n }\n\n return { fields: set, skipUpdate };\n }\n\n beforeSave() {\n for (const column of this.static.columns.values()) {\n // Run beforeSave\n if (column.beforeSave) {\n this[column.name] = column.beforeSave.call(this, this[column.name]);\n }\n }\n }\n\n /**\n * Return original value from the database or undefined if not known\n */\n getOriginalValue(key: string): unknown | undefined {\n const column = this.static.columns.get(key);\n if (!column) {\n throw new Error('Unknown property ' + key);\n }\n const c = this.savedProperties.get(key);\n if (c === undefined) {\n return undefined;\n }\n if (typeof c === 'object' && c !== null && 'from' in c) {\n return c.from();\n }\n return column.from(c);\n }\n\n async save(): Promise<boolean> {\n if (!this.static.table) {\n throw new Error('Table name not set');\n }\n\n if (!this.static.primary) {\n throw new Error('Primary key not set for model ' + this.constructor.name + ' ' + this.static);\n }\n\n // Check if relation models were modified\n for (const relation of this.static.relations) {\n // If a relation is loaded, we can check if it changed the value of the foreign key\n if (relation.isLoaded(this)) {\n // The foreign key is modified (set, changed or cleared)\n if (relation.isSet(this)) {\n const model = this[relation.modelKey] as Model;\n\n if (!model.existsInDatabase) {\n throw new Error('You cannot set a relation that is not yet saved in the database.');\n }\n\n if (this[relation.foreignKey] !== model.getPrimaryKey()) {\n // Should always match because setRelation will make it match on setting it.\n throw new Error(\n 'You cannot modify the value of a foreign key when the relation is loaded. Unload the relation first or modify the relation with setRelation',\n );\n }\n }\n else {\n if (this[relation.foreignKey] !== null) {\n // Should always match because setRelation will make it match on setting it.\n throw new Error(\n 'You cannot set a foreign key when the relation is loaded. Unload the relation first or modify the relation with setRelation',\n );\n }\n }\n }\n }\n\n const id = this.getPrimaryKey();\n if (!id) {\n if (this.existsInDatabase) {\n throw new Error('Model ' + this.constructor.name + \" was loaded from the Database, but didn't select the ID. Saving not possible.\");\n }\n }\n else {\n if (!this.existsInDatabase && this.static.primary.type === 'integer' && this.static.primary.name === 'id') {\n throw new Error(\n `PrimaryKey was set programmatically without fetching the model ${this.constructor.name} from the database. This is not allowed for integer primary keys.`,\n );\n }\n }\n\n this.beforeSave();\n const { fields, skipUpdate } = this.getChangedDatabaseProperties();\n\n if (Object.keys(fields).length === 0) {\n if (Model.showWarnings) console.warn('Tried to update model without any properties modified');\n return false;\n }\n\n if (this.existsInDatabase && skipUpdate === Object.keys(fields).length) {\n if (Model.showWarnings) console.warn('Tried to update model without any properties modified');\n return false;\n }\n\n if (Model.debug) console.log('Saving ' + this.constructor.name + ' to...', fields);\n\n // todo: save here\n if (!this.existsInDatabase) {\n if (Model.debug) console.log(`Creating new ${this.constructor.name}`);\n\n const [result] = await Database.insert('INSERT INTO `' + this.static.table + '` SET ?', [fields]);\n\n if (this.static.primary.type === 'integer' && this.static.primary.name === 'id') {\n // Auto increment value\n this[this.static.primary.name] = result.insertId;\n fields[this.static.primary.name] = result.insertId; // Also mark saved\n if (Model.debug) console.log(`New id = ${this[this.static.primary.name]}`);\n }\n }\n else {\n if (Model.debug) console.log(`Updating ${this.constructor.name} where ${this.static.primary.name} = ${id}`);\n\n const [result] = await Database.update('UPDATE `' + this.static.table + '` SET ? WHERE `' + this.static.primary.name + '` = ?', [fields, id]);\n if (result.changedRows !== 1 && Model.showWarnings) {\n console.warn(`Updated ${this.constructor.name}, but it didn't change a row. Check if ID exists.`);\n }\n }\n\n // Emit event\n if (this.existsInDatabase) {\n // Keep reference to old properties before marking saved\n const originalProperties = Object.fromEntries(\n Array.from(this.savedProperties.entries()).map(([key, value]) => {\n if (typeof value === 'object' && value !== null && 'to' in value) {\n return [key, value.to()];\n }\n return [key, value];\n }),\n );\n\n // Mark saved before sending the event\n this.markSaved(fields);\n\n await this.static.modelEventBus.sendEvent({\n type: 'updated',\n model: this,\n changedFields: fields,\n originalFields: originalProperties,\n getOldModel: () => {\n // Build a new model as if it was loaded from the database using the same original properties\n const v = this.static.fromRow(originalProperties);\n if (v === undefined) {\n throw new Error('Failed to create old model: primary key might be missing');\n }\n return v;\n },\n });\n }\n else {\n this.markSaved(fields);\n await this.static.modelEventBus.sendEvent({ type: 'created', model: this });\n }\n\n return true;\n }\n\n async delete() {\n const id = this.getPrimaryKey();\n\n if (!id && this.existsInDatabase) {\n throw new Error('Model ' + this.constructor.name + \" was loaded from the Database, but didn't select the ID. Deleting not possible.\");\n }\n\n if (!id || !this.existsInDatabase) {\n throw new Error('Model ' + this.constructor.name + \" can't be deleted if it doesn't exist in the database already\");\n }\n\n if (Model.debug) console.log(`Deleting ${this.constructor.name} where ${this.static.primary.name} = ${id}`);\n\n const [result] = await Database.delete('DELETE FROM `' + this.static.table + '` WHERE `' + this.static.primary.name + '` = ?', [id]);\n if (result.affectedRows !== 1 && Model.showWarnings) {\n console.warn(`Deleted ${this.constructor.name}, but it didn't change a row. Check if ID exists.`);\n }\n\n this.existsInDatabase = false;\n if (this.static.primary.type === 'integer' && this.static.primary.name === 'id') {\n this.eraseProperty(this.static.primary.name);\n }\n this.savedProperties.clear();\n\n await this.static.modelEventBus.sendEvent({ type: 'deleted', model: this });\n }\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
const require_Database = require("./Database.cjs");
|
|
2
|
+
//#region src/classes/OneToManyRelation.ts
|
|
3
|
+
var OneToManyRelation = class {
|
|
4
|
+
modelA;
|
|
5
|
+
modelB;
|
|
6
|
+
/**
|
|
7
|
+
* Foreign key of the other model that references the current model
|
|
8
|
+
*/
|
|
9
|
+
foreignKey;
|
|
10
|
+
/**
|
|
11
|
+
* E.g. categories
|
|
12
|
+
*/
|
|
13
|
+
modelKey;
|
|
14
|
+
/**
|
|
15
|
+
* Sort the loading of this relation
|
|
16
|
+
*/
|
|
17
|
+
sortKey;
|
|
18
|
+
sortOrder = "ASC";
|
|
19
|
+
constructor(modelA, modelB, modelKey, foreignKey) {
|
|
20
|
+
this.modelA = modelA;
|
|
21
|
+
this.modelB = modelB;
|
|
22
|
+
this.modelKey = modelKey;
|
|
23
|
+
this.foreignKey = foreignKey;
|
|
24
|
+
}
|
|
25
|
+
setSort(key, order = "ASC") {
|
|
26
|
+
this.sortKey = key;
|
|
27
|
+
this.sortOrder = order;
|
|
28
|
+
return this;
|
|
29
|
+
}
|
|
30
|
+
isLoaded(model) {
|
|
31
|
+
return Array.isArray(model[this.modelKey]);
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Generate a join query
|
|
35
|
+
* @param namespaceA namespace in the SQL query of modelA
|
|
36
|
+
* @param namespaceB namespace in the SQL query of modelB
|
|
37
|
+
*/
|
|
38
|
+
joinQuery(namespaceA, namespaceB) {
|
|
39
|
+
return `LEFT JOIN ${this.modelB.table} as ${namespaceB} on ${namespaceB}.${this.foreignKey.toString()} = ${namespaceA}.${this.modelA.primary.name}\n`;
|
|
40
|
+
}
|
|
41
|
+
orderByQuery(namespaceB) {
|
|
42
|
+
if (this.sortKey === void 0) return "";
|
|
43
|
+
let str = `\nORDER BY ${namespaceB}.${this.sortKey.toString()}`;
|
|
44
|
+
if (this.sortOrder === "DESC") str += " DESC";
|
|
45
|
+
return str + "\n";
|
|
46
|
+
}
|
|
47
|
+
async load(modelA, sorted = true, where) {
|
|
48
|
+
const namespaceB = "B";
|
|
49
|
+
let str = `SELECT ${this.modelB.getDefaultSelect(namespaceB)} FROM ${this.modelB.table} as ${namespaceB}\n`;
|
|
50
|
+
str += `WHERE ${namespaceB}.${this.foreignKey.toString()} = ?`;
|
|
51
|
+
const params = [modelA.getPrimaryKey()];
|
|
52
|
+
if (where) {
|
|
53
|
+
for (const key in where) if (Object.hasOwnProperty.call(where, key)) {
|
|
54
|
+
str += ` AND ${namespaceB}.\`${key}\` = ?`;
|
|
55
|
+
params.push(where[key]);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
if (sorted) str += this.orderByQuery("B");
|
|
59
|
+
const [rows] = await require_Database.Database.select(str, params);
|
|
60
|
+
const modelsB = this.modelB.fromRows(rows, namespaceB);
|
|
61
|
+
modelA.setManyRelation(this, modelsB);
|
|
62
|
+
return modelsB;
|
|
63
|
+
}
|
|
64
|
+
};
|
|
65
|
+
//#endregion
|
|
66
|
+
exports.OneToManyRelation = OneToManyRelation;
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Model } from "./Model.cjs";
|
|
2
|
+
|
|
3
|
+
//#region src/classes/OneToManyRelation.d.ts
|
|
4
|
+
declare class OneToManyRelation<Key extends keyof any, A extends Model, B extends Model> {
|
|
5
|
+
modelA: {
|
|
6
|
+
new (): A;
|
|
7
|
+
} & typeof Model;
|
|
8
|
+
modelB: {
|
|
9
|
+
new (): B;
|
|
10
|
+
} & typeof Model;
|
|
11
|
+
/**
|
|
12
|
+
* Foreign key of the other model that references the current model
|
|
13
|
+
*/
|
|
14
|
+
foreignKey: keyof B;
|
|
15
|
+
/**
|
|
16
|
+
* E.g. categories
|
|
17
|
+
*/
|
|
18
|
+
modelKey: Key;
|
|
19
|
+
/**
|
|
20
|
+
* Sort the loading of this relation
|
|
21
|
+
*/
|
|
22
|
+
sortKey: keyof B | undefined;
|
|
23
|
+
sortOrder: 'ASC' | 'DESC';
|
|
24
|
+
constructor(modelA: {
|
|
25
|
+
new (): A;
|
|
26
|
+
} & typeof Model, modelB: {
|
|
27
|
+
new (): B;
|
|
28
|
+
} & typeof Model, modelKey: Key, foreignKey: keyof B);
|
|
29
|
+
setSort(key: keyof B, order?: 'ASC' | 'DESC'): this;
|
|
30
|
+
isLoaded<T extends A>(model: T): model is T & Record<Key, B[]>;
|
|
31
|
+
/**
|
|
32
|
+
* Generate a join query
|
|
33
|
+
* @param namespaceA namespace in the SQL query of modelA
|
|
34
|
+
* @param namespaceB namespace in the SQL query of modelB
|
|
35
|
+
*/
|
|
36
|
+
joinQuery(namespaceA: string, namespaceB: string): string;
|
|
37
|
+
orderByQuery(namespaceB: string): string;
|
|
38
|
+
load(modelA: A, sorted?: boolean, where?: object): Promise<B[]>;
|
|
39
|
+
}
|
|
40
|
+
//#endregion
|
|
41
|
+
export { OneToManyRelation };
|
|
42
|
+
//# sourceMappingURL=OneToManyRelation.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OneToManyRelation.d.cts","names":[],"sources":["../../../src/classes/OneToManyRelation.ts"],"mappings":";;;cAGa,iBAAA,kCAAmD,KAAA,YAAiB,KAAA;EAC7E,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA;EAC/B,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA;EADb;;;EAMlB,UAAA,QAAkB,CAAA;EAAA;;;EAKlB,QAAA,EAAU,GAAA;EAQiC;;;EAH3C,OAAA,QAAe,CAAA;EACf,SAAA;cAEY,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA,EAAO,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA,EAAO,QAAA,EAAU,GAAA,EAAK,UAAA,QAAkB,CAAA;EAOzH,OAAA,CAAQ,GAAA,QAAW,CAAA,EAAG,KAAA;EAOtB,QAAA,WAAmB,CAAA,CAAA,CAAG,KAAA,EAAO,CAAA,GAAI,KAAA,IAAS,CAAA,GAAI,MAAA,CAAO,GAAA,EAAK,CAAA;EA0BvC;;;;;EAhBnB,SAAA,CAAU,UAAA,UAAoB,UAAA;EAI9B,YAAA,CAAa,UAAA;EAYP,IAAA,CAAK,MAAA,EAAQ,CAAA,EAAG,MAAA,YAAe,KAAA,YAAiB,OAAA,CAAQ,CAAA;AAAA"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Model } from "./Model.mjs";
|
|
2
|
+
|
|
3
|
+
//#region src/classes/OneToManyRelation.d.ts
|
|
4
|
+
declare class OneToManyRelation<Key extends keyof any, A extends Model, B extends Model> {
|
|
5
|
+
modelA: {
|
|
6
|
+
new (): A;
|
|
7
|
+
} & typeof Model;
|
|
8
|
+
modelB: {
|
|
9
|
+
new (): B;
|
|
10
|
+
} & typeof Model;
|
|
11
|
+
/**
|
|
12
|
+
* Foreign key of the other model that references the current model
|
|
13
|
+
*/
|
|
14
|
+
foreignKey: keyof B;
|
|
15
|
+
/**
|
|
16
|
+
* E.g. categories
|
|
17
|
+
*/
|
|
18
|
+
modelKey: Key;
|
|
19
|
+
/**
|
|
20
|
+
* Sort the loading of this relation
|
|
21
|
+
*/
|
|
22
|
+
sortKey: keyof B | undefined;
|
|
23
|
+
sortOrder: 'ASC' | 'DESC';
|
|
24
|
+
constructor(modelA: {
|
|
25
|
+
new (): A;
|
|
26
|
+
} & typeof Model, modelB: {
|
|
27
|
+
new (): B;
|
|
28
|
+
} & typeof Model, modelKey: Key, foreignKey: keyof B);
|
|
29
|
+
setSort(key: keyof B, order?: 'ASC' | 'DESC'): this;
|
|
30
|
+
isLoaded<T extends A>(model: T): model is T & Record<Key, B[]>;
|
|
31
|
+
/**
|
|
32
|
+
* Generate a join query
|
|
33
|
+
* @param namespaceA namespace in the SQL query of modelA
|
|
34
|
+
* @param namespaceB namespace in the SQL query of modelB
|
|
35
|
+
*/
|
|
36
|
+
joinQuery(namespaceA: string, namespaceB: string): string;
|
|
37
|
+
orderByQuery(namespaceB: string): string;
|
|
38
|
+
load(modelA: A, sorted?: boolean, where?: object): Promise<B[]>;
|
|
39
|
+
}
|
|
40
|
+
//#endregion
|
|
41
|
+
export { OneToManyRelation };
|
|
42
|
+
//# sourceMappingURL=OneToManyRelation.d.mts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OneToManyRelation.d.mts","names":[],"sources":["../../../src/classes/OneToManyRelation.ts"],"mappings":";;;cAGa,iBAAA,kCAAmD,KAAA,YAAiB,KAAA;EAC7E,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA;EAC/B,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA;EADb;;;EAMlB,UAAA,QAAkB,CAAA;EAAA;;;EAKlB,QAAA,EAAU,GAAA;EAQiC;;;EAH3C,OAAA,QAAe,CAAA;EACf,SAAA;cAEY,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA,EAAO,MAAA;IAAA,QAAkB,CAAA;EAAA,WAAa,KAAA,EAAO,QAAA,EAAU,GAAA,EAAK,UAAA,QAAkB,CAAA;EAOzH,OAAA,CAAQ,GAAA,QAAW,CAAA,EAAG,KAAA;EAOtB,QAAA,WAAmB,CAAA,CAAA,CAAG,KAAA,EAAO,CAAA,GAAI,KAAA,IAAS,CAAA,GAAI,MAAA,CAAO,GAAA,EAAK,CAAA;EA0BvC;;;;;EAhBnB,SAAA,CAAU,UAAA,UAAoB,UAAA;EAI9B,YAAA,CAAa,UAAA;EAYP,IAAA,CAAK,MAAA,EAAQ,CAAA,EAAG,MAAA,YAAe,KAAA,YAAiB,OAAA,CAAQ,CAAA;AAAA"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { Database } from "./Database.mjs";
|
|
2
|
+
//#region src/classes/OneToManyRelation.ts
|
|
3
|
+
var OneToManyRelation = class {
|
|
4
|
+
modelA;
|
|
5
|
+
modelB;
|
|
6
|
+
/**
|
|
7
|
+
* Foreign key of the other model that references the current model
|
|
8
|
+
*/
|
|
9
|
+
foreignKey;
|
|
10
|
+
/**
|
|
11
|
+
* E.g. categories
|
|
12
|
+
*/
|
|
13
|
+
modelKey;
|
|
14
|
+
/**
|
|
15
|
+
* Sort the loading of this relation
|
|
16
|
+
*/
|
|
17
|
+
sortKey;
|
|
18
|
+
sortOrder = "ASC";
|
|
19
|
+
constructor(modelA, modelB, modelKey, foreignKey) {
|
|
20
|
+
this.modelA = modelA;
|
|
21
|
+
this.modelB = modelB;
|
|
22
|
+
this.modelKey = modelKey;
|
|
23
|
+
this.foreignKey = foreignKey;
|
|
24
|
+
}
|
|
25
|
+
setSort(key, order = "ASC") {
|
|
26
|
+
this.sortKey = key;
|
|
27
|
+
this.sortOrder = order;
|
|
28
|
+
return this;
|
|
29
|
+
}
|
|
30
|
+
isLoaded(model) {
|
|
31
|
+
return Array.isArray(model[this.modelKey]);
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Generate a join query
|
|
35
|
+
* @param namespaceA namespace in the SQL query of modelA
|
|
36
|
+
* @param namespaceB namespace in the SQL query of modelB
|
|
37
|
+
*/
|
|
38
|
+
joinQuery(namespaceA, namespaceB) {
|
|
39
|
+
return `LEFT JOIN ${this.modelB.table} as ${namespaceB} on ${namespaceB}.${this.foreignKey.toString()} = ${namespaceA}.${this.modelA.primary.name}\n`;
|
|
40
|
+
}
|
|
41
|
+
orderByQuery(namespaceB) {
|
|
42
|
+
if (this.sortKey === void 0) return "";
|
|
43
|
+
let str = `\nORDER BY ${namespaceB}.${this.sortKey.toString()}`;
|
|
44
|
+
if (this.sortOrder === "DESC") str += " DESC";
|
|
45
|
+
return str + "\n";
|
|
46
|
+
}
|
|
47
|
+
async load(modelA, sorted = true, where) {
|
|
48
|
+
const namespaceB = "B";
|
|
49
|
+
let str = `SELECT ${this.modelB.getDefaultSelect(namespaceB)} FROM ${this.modelB.table} as ${namespaceB}\n`;
|
|
50
|
+
str += `WHERE ${namespaceB}.${this.foreignKey.toString()} = ?`;
|
|
51
|
+
const params = [modelA.getPrimaryKey()];
|
|
52
|
+
if (where) {
|
|
53
|
+
for (const key in where) if (Object.hasOwnProperty.call(where, key)) {
|
|
54
|
+
str += ` AND ${namespaceB}.\`${key}\` = ?`;
|
|
55
|
+
params.push(where[key]);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
if (sorted) str += this.orderByQuery("B");
|
|
59
|
+
const [rows] = await Database.select(str, params);
|
|
60
|
+
const modelsB = this.modelB.fromRows(rows, namespaceB);
|
|
61
|
+
modelA.setManyRelation(this, modelsB);
|
|
62
|
+
return modelsB;
|
|
63
|
+
}
|
|
64
|
+
};
|
|
65
|
+
//#endregion
|
|
66
|
+
export { OneToManyRelation };
|
|
67
|
+
|
|
68
|
+
//# sourceMappingURL=OneToManyRelation.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OneToManyRelation.mjs","names":[],"sources":["../../../src/classes/OneToManyRelation.ts"],"sourcesContent":["import { Database } from './Database.js';\nimport { Model } from './Model.js';\n\nexport class OneToManyRelation<Key extends keyof any, A extends Model, B extends Model> {\n modelA: { new (): A } & typeof Model;\n modelB: { new (): B } & typeof Model;\n\n /**\n * Foreign key of the other model that references the current model\n */\n foreignKey: keyof B;\n\n /**\n * E.g. categories\n */\n modelKey: Key;\n\n /**\n * Sort the loading of this relation\n */\n sortKey: keyof B | undefined;\n sortOrder: 'ASC' | 'DESC' = 'ASC';\n\n constructor(modelA: { new (): A } & typeof Model, modelB: { new (): B } & typeof Model, modelKey: Key, foreignKey: keyof B) {\n this.modelA = modelA;\n this.modelB = modelB;\n this.modelKey = modelKey;\n this.foreignKey = foreignKey;\n }\n\n setSort(key: keyof B, order: 'ASC' | 'DESC' = 'ASC'): this {\n this.sortKey = key;\n this.sortOrder = order;\n return this;\n }\n\n /// Whether this relation is loaded\n isLoaded<T extends A>(model: T): model is T & Record<Key, B[]> {\n // Also not loaded if null, since it should be an empty array or an array if it is loaded\n return Array.isArray((model as any)[this.modelKey]);\n }\n\n /**\n * Generate a join query\n * @param namespaceA namespace in the SQL query of modelA\n * @param namespaceB namespace in the SQL query of modelB\n */\n joinQuery(namespaceA: string, namespaceB: string): string {\n return `LEFT JOIN ${this.modelB.table} as ${namespaceB} on ${namespaceB}.${this.foreignKey.toString()} = ${namespaceA}.${this.modelA.primary.name}\\n`;\n }\n\n orderByQuery(namespaceB: string): string {\n if (this.sortKey === undefined) {\n return '';\n }\n let str = `\\nORDER BY ${namespaceB}.${this.sortKey.toString()}`;\n if (this.sortOrder === 'DESC') {\n str += ' DESC';\n }\n return str + '\\n';\n }\n\n /// Load the relation of a model and return the loaded models\n async load(modelA: A, sorted = true, where?: object): Promise<B[]> {\n const namespaceB = 'B';\n let str = `SELECT ${this.modelB.getDefaultSelect(namespaceB)} FROM ${this.modelB.table} as ${namespaceB}\\n`;\n str += `WHERE ${namespaceB}.${this.foreignKey.toString()} = ?`;\n\n const params = [modelA.getPrimaryKey()];\n if (where) {\n for (const key in where) {\n if (Object.hasOwnProperty.call(where, key)) {\n str += ` AND ${namespaceB}.\\`${key}\\` = ?`;\n params.push(where[key] as string);\n }\n }\n }\n if (sorted) {\n str += this.orderByQuery('B');\n }\n\n const [rows] = await Database.select(str, params);\n const modelsB = this.modelB.fromRows(rows, namespaceB) as B[];\n modelA.setManyRelation(this, modelsB);\n return modelsB;\n }\n}\n"],"mappings":";;AAGA,IAAa,oBAAb,MAAwF;CACpF;CACA;;;;CAKA;;;;CAKA;;;;CAKA;CACA,YAA4B;CAE5B,YAAY,QAAsC,QAAsC,UAAe,YAAqB;AACxH,OAAK,SAAS;AACd,OAAK,SAAS;AACd,OAAK,WAAW;AAChB,OAAK,aAAa;;CAGtB,QAAQ,KAAc,QAAwB,OAAa;AACvD,OAAK,UAAU;AACf,OAAK,YAAY;AACjB,SAAO;;CAIX,SAAsB,OAAyC;AAE3D,SAAO,MAAM,QAAS,MAAc,KAAK,UAAU;;;;;;;CAQvD,UAAU,YAAoB,YAA4B;AACtD,SAAO,aAAa,KAAK,OAAO,MAAM,MAAM,WAAW,MAAM,WAAW,GAAG,KAAK,WAAW,UAAU,CAAC,KAAK,WAAW,GAAG,KAAK,OAAO,QAAQ,KAAK;;CAGtJ,aAAa,YAA4B;AACrC,MAAI,KAAK,YAAY,KAAA,EACjB,QAAO;EAEX,IAAI,MAAM,cAAc,WAAW,GAAG,KAAK,QAAQ,UAAU;AAC7D,MAAI,KAAK,cAAc,OACnB,QAAO;AAEX,SAAO,MAAM;;CAIjB,MAAM,KAAK,QAAW,SAAS,MAAM,OAA8B;EAC/D,MAAM,aAAa;EACnB,IAAI,MAAM,UAAU,KAAK,OAAO,iBAAiB,WAAW,CAAC,QAAQ,KAAK,OAAO,MAAM,MAAM,WAAW;AACxG,SAAO,SAAS,WAAW,GAAG,KAAK,WAAW,UAAU,CAAC;EAEzD,MAAM,SAAS,CAAC,OAAO,eAAe,CAAC;AACvC,MAAI;QACK,MAAM,OAAO,MACd,KAAI,OAAO,eAAe,KAAK,OAAO,IAAI,EAAE;AACxC,WAAO,QAAQ,WAAW,KAAK,IAAI;AACnC,WAAO,KAAK,MAAM,KAAe;;;AAI7C,MAAI,OACA,QAAO,KAAK,aAAa,IAAI;EAGjC,MAAM,CAAC,QAAQ,MAAM,SAAS,OAAO,KAAK,OAAO;EACjD,MAAM,UAAU,KAAK,OAAO,SAAS,MAAM,WAAW;AACtD,SAAO,gBAAgB,MAAM,QAAQ;AACrC,SAAO"}
|