velocious 1.0.331 → 1.0.332
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/README.md +43 -0
- package/build/src/configuration-types.d.ts +5 -0
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +2 -1
- package/build/src/configuration.d.ts +9 -1
- package/build/src/configuration.d.ts.map +1 -1
- package/build/src/configuration.js +10 -2
- package/build/src/database/query/model-class-query.d.ts.map +1 -1
- package/build/src/database/query/model-class-query.js +6 -1
- package/build/src/database/record/index.d.ts +7 -0
- package/build/src/database/record/index.d.ts.map +1 -1
- package/build/src/database/record/index.js +8 -6
- package/build/src/database/record/instance-relationships/base.d.ts +18 -0
- package/build/src/database/record/instance-relationships/base.d.ts.map +1 -1
- package/build/src/database/record/instance-relationships/base.js +77 -1
- package/build/src/database/record/instance-relationships/belongs-to.d.ts.map +1 -1
- package/build/src/database/record/instance-relationships/belongs-to.js +9 -1
- package/build/src/database/record/instance-relationships/has-many.d.ts.map +1 -1
- package/build/src/database/record/instance-relationships/has-many.js +16 -6
- package/build/src/database/record/instance-relationships/has-one.d.ts.map +1 -1
- package/build/src/database/record/instance-relationships/has-one.js +9 -1
- package/build/src/database/record/relationships/base.d.ts +9 -1
- package/build/src/database/record/relationships/base.d.ts.map +1 -1
- package/build/src/database/record/relationships/base.js +6 -2
- package/package.json +1 -1
|
@@ -55,6 +55,24 @@ export default class VelociousDatabaseRecordBaseInstanceRelationship<MC extends
|
|
|
55
55
|
* @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.
|
|
56
56
|
*/
|
|
57
57
|
load(): Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>;
|
|
58
|
+
/**
|
|
59
|
+
* Loads the relationship if not already loaded. When the parent record was
|
|
60
|
+
* loaded as part of a batch (cohort) and autoload is enabled, siblings in
|
|
61
|
+
* the cohort that share this relationship and have not preloaded it yet
|
|
62
|
+
* are batched into a single query via the existing preloader path.
|
|
63
|
+
* @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.
|
|
64
|
+
*/
|
|
65
|
+
autoloadOrLoad(): Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>;
|
|
66
|
+
/**
|
|
67
|
+
* Attempts to batch-load this relationship across cohort siblings via the
|
|
68
|
+
* existing preloader path. Returns true when a batch ran (self is always
|
|
69
|
+
* included because callers reset their own `_preloaded` state before
|
|
70
|
+
* calling), false when autoload is off, there is no cohort, or no batch
|
|
71
|
+
* candidates remain. Siblings that have already preloaded this relationship
|
|
72
|
+
* are skipped so their cached value is preserved.
|
|
73
|
+
* @returns {Promise<boolean>} - Whether a cohort batch preload ran.
|
|
74
|
+
*/
|
|
75
|
+
_tryCohortPreload(): Promise<boolean>;
|
|
58
76
|
/** @returns {boolean} Whether the relationship has been preloaded */
|
|
59
77
|
isLoaded(): boolean;
|
|
60
78
|
/** @returns {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} The loaded model or models (depending on relationship type) */
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/base.js"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/base.js"],"names":[],"mappings":"AAMA;;;;;;GAMG;AAEH;;;;GAIG;AACH,qEAHqD,EAAE,SAA1C,cAAe,aAAa,EAAE,OAAQ,yCACE,GAAG,SAA3C,cAAe,aAAa,EAAE,OAAQ;IAUjD;;OAEG;IACH,qCAFW,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAMhD;IAdD,kCAAkC;IAClC,WADW,OAAO,GAAG,SAAS,CACT;IACrB,kCAAkC;IAClC,YADW,OAAO,GAAG,SAAS,CACR;IACtB,uEAAuE;IACvE,SADW,YAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAChD;IAMjB,gBAAmB;IACnB,wBAAkB;IAClB,yDAAgC;IAGlC;;;;OAIG;IACH,oBAHW,YAAY,CAAC,GAAG,CAAC,EAAE,GAAG,YAAY,CAAC,GAAG,CAAC,GACrC,IAAI,CAIhB;IAED;;;;OAIG;IACH,kBAHW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,YAAY,CAAC,GAAG,CAAC,CAI7B;IAED,kHAAkH;IAClH,eADc,OAAO,GAAG,SAAS,CACM;IAEvC;;;OAGG;IACH,8BAHW,OAAO,GACL,IAAI,CAEkD;IAEnE;;;OAGG;IACH,mBAHW,OAAO,GACL,IAAI,CAE4B;IAE7C,+EAA+E;IAC/E,YADc,OAAO,CACY;IAEjC;;;OAGG;IACH,QAFa,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAI7E;IAED;;;;;;OAMG;IACH,kBAFa,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAU7E;IAED;;;;;;;;OAQG;IACH,qBAFa,OAAO,CAAC,OAAO,CAAC,CAwD5B;IAED,sEAAsE;IACtE,YADe,OAAO,CACqB;IAE3C,sIAAsI;IACtI,UADc,YAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAOrE;IAED,oHAAoH;IACpH,iBADY,YAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,QAC3B;IAEzC,iGAAiG;IACjG,wBADc,YAAY,CAAC,GAAG,CAAC,GAAG,YAAY,CAAC,GAAG,CAAC,EAAE,GAAG,SAAS,CACnB;IAE9C,qFAAqF;IACrF,gBADc,OAAO,CAC6B;IAElD,4EAA4E;IAC5E,0BADY,OAAO,QACwC;IAE3D,8DAA8D;IAC9D,iBADc,MAAM,CAC6C;IAEjE,gDAAgD;IAChD,YADc,YAAY,CAAC,EAAE,CAAC,CACE;IAEhC,sEAAsE;IACtE,iBADc,MAAM,CAC6C;IAEjE,gIAAgI;IAChI,mBADc,OAAO,0BAA0B,EAAE,OAAO,CACV;IAE9C;;;;OAIG;IACH,WAJa,CAAC,SACH,CAAC,GACC,CAAC,CAIb;IAED,iFAAiF;IACjF,uBADc,GAAG,GAAG,SAAS,CAK5B;IAED,uFAAuF;IACvF,WADc,MAAM,CACiC;CACtD;0CAjNoD,EAAE,SAA1C,cAAe,aAAa,EAAE,OAAQ,yCACE,GAAG,SAA3C,cAAe,aAAa,EAAE,OAAQ;;;;WAErC,YAAY,CAAC,EAAE,CAAC;;;;kBAChB,OAAO,0BAA0B,EAAE,OAAO"}
|
|
@@ -1,4 +1,7 @@
|
|
|
1
1
|
// @ts-check
|
|
2
|
+
import BelongsToPreloader from "../../query/preloader/belongs-to.js";
|
|
3
|
+
import HasManyPreloader from "../../query/preloader/has-many.js";
|
|
4
|
+
import HasOnePreloader from "../../query/preloader/has-one.js";
|
|
2
5
|
/**
|
|
3
6
|
* @template {typeof import("../index.js").default} [MC=typeof import("../index.js").default]
|
|
4
7
|
* @template {typeof import("../index.js").default} [TMC=typeof import("../index.js").default]
|
|
@@ -63,6 +66,79 @@ export default class VelociousDatabaseRecordBaseInstanceRelationship {
|
|
|
63
66
|
load() {
|
|
64
67
|
throw new Error("'load' not implemented");
|
|
65
68
|
}
|
|
69
|
+
/**
|
|
70
|
+
* Loads the relationship if not already loaded. When the parent record was
|
|
71
|
+
* loaded as part of a batch (cohort) and autoload is enabled, siblings in
|
|
72
|
+
* the cohort that share this relationship and have not preloaded it yet
|
|
73
|
+
* are batched into a single query via the existing preloader path.
|
|
74
|
+
* @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.
|
|
75
|
+
*/
|
|
76
|
+
async autoloadOrLoad() {
|
|
77
|
+
if (this._loaded !== undefined)
|
|
78
|
+
return this._loaded;
|
|
79
|
+
const batched = await this._tryCohortPreload();
|
|
80
|
+
if (!batched)
|
|
81
|
+
await this.load();
|
|
82
|
+
return this._loaded;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Attempts to batch-load this relationship across cohort siblings via the
|
|
86
|
+
* existing preloader path. Returns true when a batch ran (self is always
|
|
87
|
+
* included because callers reset their own `_preloaded` state before
|
|
88
|
+
* calling), false when autoload is off, there is no cohort, or no batch
|
|
89
|
+
* candidates remain. Siblings that have already preloaded this relationship
|
|
90
|
+
* are skipped so their cached value is preserved.
|
|
91
|
+
* @returns {Promise<boolean>} - Whether a cohort batch preload ran.
|
|
92
|
+
*/
|
|
93
|
+
async _tryCohortPreload() {
|
|
94
|
+
const relationshipDef = this.getRelationship();
|
|
95
|
+
const configuration = relationshipDef.getConfiguration();
|
|
96
|
+
const cohort = /** @type {Array<import("../index.js").default> | undefined} */ ( /** @type {any} */(this.model)._loadCohort);
|
|
97
|
+
if (!configuration.getAutoload() || !relationshipDef.getAutoload() || !cohort || cohort.length <= 1) {
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
const relationshipName = relationshipDef.getRelationshipName();
|
|
101
|
+
const OwnerModelClass = /** @type {any} */ (this.model).constructor;
|
|
102
|
+
/** @type {Array<import("../index.js").default>} */
|
|
103
|
+
const batch = [];
|
|
104
|
+
// Exact same class, persisted, no existing in-memory relationship state.
|
|
105
|
+
// Skip siblings where `_loaded` is already set — they may have been
|
|
106
|
+
// preloaded (preserve cache) OR locally manipulated via `build...` /
|
|
107
|
+
// `set...` (preserve unsaved edits). Either way the preloader would
|
|
108
|
+
// overwrite that state.
|
|
109
|
+
for (const sibling of cohort) {
|
|
110
|
+
if (sibling.constructor !== OwnerModelClass)
|
|
111
|
+
continue;
|
|
112
|
+
if (!sibling.isPersisted())
|
|
113
|
+
continue;
|
|
114
|
+
const siblingInstanceRelationship = sibling.getRelationshipByName(relationshipName);
|
|
115
|
+
if (siblingInstanceRelationship.getLoadedOrUndefined() !== undefined)
|
|
116
|
+
continue;
|
|
117
|
+
batch.push(sibling);
|
|
118
|
+
}
|
|
119
|
+
if (batch.length === 0)
|
|
120
|
+
return false;
|
|
121
|
+
const type = relationshipDef.getType();
|
|
122
|
+
if (type == "belongsTo") {
|
|
123
|
+
const belongsToRelationship = /** @type {import("../relationships/belongs-to.js").default} */ (relationshipDef);
|
|
124
|
+
const preloader = new BelongsToPreloader({ models: batch, relationship: belongsToRelationship });
|
|
125
|
+
await preloader.run();
|
|
126
|
+
}
|
|
127
|
+
else if (type == "hasMany") {
|
|
128
|
+
const hasManyRelationship = /** @type {import("../relationships/has-many.js").default} */ (relationshipDef);
|
|
129
|
+
const preloader = new HasManyPreloader({ models: batch, relationship: hasManyRelationship });
|
|
130
|
+
await preloader.run();
|
|
131
|
+
}
|
|
132
|
+
else if (type == "hasOne") {
|
|
133
|
+
const hasOneRelationship = /** @type {import("../relationships/has-one.js").default} */ (relationshipDef);
|
|
134
|
+
const preloader = new HasOnePreloader({ models: batch, relationship: hasOneRelationship });
|
|
135
|
+
await preloader.run();
|
|
136
|
+
}
|
|
137
|
+
else {
|
|
138
|
+
throw new Error(`Unknown relationship type: ${type}`);
|
|
139
|
+
}
|
|
140
|
+
return true;
|
|
141
|
+
}
|
|
66
142
|
/** @returns {boolean} Whether the relationship has been preloaded */
|
|
67
143
|
isLoaded() { return Boolean(this._loaded); }
|
|
68
144
|
/** @returns {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} The loaded model or models (depending on relationship type) */
|
|
@@ -104,4 +180,4 @@ export default class VelociousDatabaseRecordBaseInstanceRelationship {
|
|
|
104
180
|
/** @returns {string} The type of relationship (e.g. "has_many", "belongs_to", etc.) */
|
|
105
181
|
getType() { return this.getRelationship().getType(); }
|
|
106
182
|
}
|
|
107
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\n/**\n * @template {typeof import(\"../index.js\").default} [MC=typeof import(\"../index.js\").default]\n * @template {typeof import(\"../index.js\").default} [TMC=typeof import(\"../index.js\").default]\n * @typedef {object} InstanceRelationshipsBaseArgs\n * @property {InstanceType<MC>} model - Parent model instance.\n * @property {import(\"../relationships/base.js\").default} relationship - Relationship metadata definition.\n */\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../index.js\").default} [MC=typeof import(\"../index.js\").default]\n * @template {typeof import(\"../index.js\").default} [TMC=typeof import(\"../index.js\").default]\n */\nexport default class VelociousDatabaseRecordBaseInstanceRelationship {\n  /** @type {boolean | undefined} */\n  _autoSave = undefined\n  /** @type {boolean | undefined} */\n  _preloaded = undefined\n  /** @type {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} */\n  _loaded = undefined\n\n  /**\n   * @param {InstanceRelationshipsBaseArgs<MC, TMC>} args - Options object.\n   */\n  constructor({model, relationship}) {\n    this._dirty = false\n    this.model = model\n    this.relationship = relationship\n  }\n\n  /**\n   * @abstract\n   * @param {InstanceType<TMC>[] | InstanceType<TMC>} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  addToLoaded(models) { // eslint-disable-line no-unused-vars\n    throw new Error(\"addToLoaded not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @param {Record<string, any>} attributes - Attributes.\n   * @returns {InstanceType<TMC>} - The build.\n   */\n  build(attributes) { // eslint-disable-line no-unused-vars\n    throw new Error(\"'build' not implemented\")\n  }\n\n  /** @returns {boolean | undefined} Whether the relationship should be auto-saved before saving the parent model */\n  getAutoSave() { return this._autoSave }\n\n  /**\n   * @param {boolean} newAutoSaveValue Whether the relationship should be auto-saved before saving the parent model\n   * @returns {void} - No return value.\n   */\n  setAutoSave(newAutoSaveValue) { this._autoSave = newAutoSaveValue }\n\n  /**\n   * @param {boolean} newValue Whether the relationship is dirty (has been modified)\n   * @returns {void} - No return value.\n   */\n  setDirty(newValue) { this._dirty = newValue }\n\n  /** @returns {boolean} Whether the relationship is dirty (has been modified) */\n  getDirty() { return this._dirty }\n\n  /**\n   * @abstract\n   * @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.\n   */\n  load() {\n    throw new Error(\"'load' not implemented\")\n  }\n\n  /**  @returns {boolean} Whether the relationship has been preloaded */\n  isLoaded() { return Boolean(this._loaded) }\n\n  /** @returns {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} The loaded model or models (depending on relationship type) */\n  loaded() {\n    if (!this._preloaded && this.model.isPersisted()) {\n      throw new Error(`${this.model.constructor.name}#${this.relationship.getRelationshipName()} hasn't been preloaded`)\n    }\n\n    return this._loaded\n  }\n\n  /** @param {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} model - Related model(s) to mark as loaded. */\n  setLoaded(model) { this._loaded = model }\n\n  /** @returns {InstanceType<TMC> | InstanceType<TMC>[] | undefined} - The loaded or undefined.  */\n  getLoadedOrUndefined() { return this._loaded }\n\n  /** @returns {boolean} The loaded model or models (depending on relationship type) */\n  getPreloaded() { return this._preloaded || false }\n\n  /** @param {boolean} isPreloaded - Whether the relationship is preloaded. */\n  setPreloaded(isPreloaded) { this._preloaded = isPreloaded }\n\n  /** @returns {string} The foreign key for this relationship */\n  getForeignKey() { return this.getRelationship().getForeignKey() }\n\n  /** @returns {InstanceType<MC>} - The model.  */\n  getModel() { return this.model }\n\n  /** @returns {string} The primary key for this relationship's model */\n  getPrimaryKey() { return this.getRelationship().getPrimaryKey() }\n\n  /** @returns {import(\"../relationships/base.js\").default} The relationship object that this instance relationship is based on */\n  getRelationship() { return this.relationship }\n\n  /**\n   * @template T\n   * @param {T} query - Query instance.\n   * @returns {T} - Scoped query.\n   */\n  applyScope(query) {\n    return this.getRelationship().applyScope(query)\n  }\n\n  /** @returns {TMC | undefined} The model class that this instance relationship */\n  getTargetModelClass() {\n    const TargetModelClass = /** @type {TMC} */ (this.getRelationship().getTargetModelClass())\n\n    return TargetModelClass\n  }\n\n  /** @returns {string} The type of relationship (e.g. \"has_many\", \"belongs_to\", etc.) */\n  getType() { return this.getRelationship().getType() }\n}\n"]}
|
|
183
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BelongsToPreloader from \"../../query/preloader/belongs-to.js\"\nimport HasManyPreloader from \"../../query/preloader/has-many.js\"\nimport HasOnePreloader from \"../../query/preloader/has-one.js\"\n\n/**\n * @template {typeof import(\"../index.js\").default} [MC=typeof import(\"../index.js\").default]\n * @template {typeof import(\"../index.js\").default} [TMC=typeof import(\"../index.js\").default]\n * @typedef {object} InstanceRelationshipsBaseArgs\n * @property {InstanceType<MC>} model - Parent model instance.\n * @property {import(\"../relationships/base.js\").default} relationship - Relationship metadata definition.\n */\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../index.js\").default} [MC=typeof import(\"../index.js\").default]\n * @template {typeof import(\"../index.js\").default} [TMC=typeof import(\"../index.js\").default]\n */\nexport default class VelociousDatabaseRecordBaseInstanceRelationship {\n  /** @type {boolean | undefined} */\n  _autoSave = undefined\n  /** @type {boolean | undefined} */\n  _preloaded = undefined\n  /** @type {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} */\n  _loaded = undefined\n\n  /**\n   * @param {InstanceRelationshipsBaseArgs<MC, TMC>} args - Options object.\n   */\n  constructor({model, relationship}) {\n    this._dirty = false\n    this.model = model\n    this.relationship = relationship\n  }\n\n  /**\n   * @abstract\n   * @param {InstanceType<TMC>[] | InstanceType<TMC>} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  addToLoaded(models) { // eslint-disable-line no-unused-vars\n    throw new Error(\"addToLoaded not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @param {Record<string, any>} attributes - Attributes.\n   * @returns {InstanceType<TMC>} - The build.\n   */\n  build(attributes) { // eslint-disable-line no-unused-vars\n    throw new Error(\"'build' not implemented\")\n  }\n\n  /** @returns {boolean | undefined} Whether the relationship should be auto-saved before saving the parent model */\n  getAutoSave() { return this._autoSave }\n\n  /**\n   * @param {boolean} newAutoSaveValue Whether the relationship should be auto-saved before saving the parent model\n   * @returns {void} - No return value.\n   */\n  setAutoSave(newAutoSaveValue) { this._autoSave = newAutoSaveValue }\n\n  /**\n   * @param {boolean} newValue Whether the relationship is dirty (has been modified)\n   * @returns {void} - No return value.\n   */\n  setDirty(newValue) { this._dirty = newValue }\n\n  /** @returns {boolean} Whether the relationship is dirty (has been modified) */\n  getDirty() { return this._dirty }\n\n  /**\n   * @abstract\n   * @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.\n   */\n  load() {\n    throw new Error(\"'load' not implemented\")\n  }\n\n  /**\n   * Loads the relationship if not already loaded. When the parent record was\n   * loaded as part of a batch (cohort) and autoload is enabled, siblings in\n   * the cohort that share this relationship and have not preloaded it yet\n   * are batched into a single query via the existing preloader path.\n   * @returns {Promise<InstanceType<TMC> | Array<InstanceType<TMC>> | undefined>} - Resolves with loaded relationship value.\n   */\n  async autoloadOrLoad() {\n    if (this._loaded !== undefined) return this._loaded\n\n    const batched = await this._tryCohortPreload()\n\n    if (!batched) await this.load()\n\n    return this._loaded\n  }\n\n  /**\n   * Attempts to batch-load this relationship across cohort siblings via the\n   * existing preloader path. Returns true when a batch ran (self is always\n   * included because callers reset their own `_preloaded` state before\n   * calling), false when autoload is off, there is no cohort, or no batch\n   * candidates remain. Siblings that have already preloaded this relationship\n   * are skipped so their cached value is preserved.\n   * @returns {Promise<boolean>} - Whether a cohort batch preload ran.\n   */\n  async _tryCohortPreload() {\n    const relationshipDef = this.getRelationship()\n    const configuration = relationshipDef.getConfiguration()\n    const cohort = /** @type {Array<import(\"../index.js\").default> | undefined} */ (/** @type {any} */ (this.model)._loadCohort)\n\n    if (!configuration.getAutoload() || !relationshipDef.getAutoload() || !cohort || cohort.length <= 1) {\n      return false\n    }\n\n    const relationshipName = relationshipDef.getRelationshipName()\n    const OwnerModelClass = /** @type {any} */ (this.model).constructor\n    /** @type {Array<import(\"../index.js\").default>} */\n    const batch = []\n\n    // Exact same class, persisted, no existing in-memory relationship state.\n    // Skip siblings where `_loaded` is already set — they may have been\n    // preloaded (preserve cache) OR locally manipulated via `build...` /\n    // `set...` (preserve unsaved edits). Either way the preloader would\n    // overwrite that state.\n    for (const sibling of cohort) {\n      if (sibling.constructor !== OwnerModelClass) continue\n      if (!sibling.isPersisted()) continue\n\n      const siblingInstanceRelationship = sibling.getRelationshipByName(relationshipName)\n\n      if (siblingInstanceRelationship.getLoadedOrUndefined() !== undefined) continue\n\n      batch.push(sibling)\n    }\n\n    if (batch.length === 0) return false\n\n    const type = relationshipDef.getType()\n\n    if (type == \"belongsTo\") {\n      const belongsToRelationship = /** @type {import(\"../relationships/belongs-to.js\").default} */ (relationshipDef)\n      const preloader = new BelongsToPreloader({models: batch, relationship: belongsToRelationship})\n\n      await preloader.run()\n    } else if (type == \"hasMany\") {\n      const hasManyRelationship = /** @type {import(\"../relationships/has-many.js\").default} */ (relationshipDef)\n      const preloader = new HasManyPreloader({models: batch, relationship: hasManyRelationship})\n\n      await preloader.run()\n    } else if (type == \"hasOne\") {\n      const hasOneRelationship = /** @type {import(\"../relationships/has-one.js\").default} */ (relationshipDef)\n      const preloader = new HasOnePreloader({models: batch, relationship: hasOneRelationship})\n\n      await preloader.run()\n    } else {\n      throw new Error(`Unknown relationship type: ${type}`)\n    }\n\n    return true\n  }\n\n  /**  @returns {boolean} Whether the relationship has been preloaded */\n  isLoaded() { return Boolean(this._loaded) }\n\n  /** @returns {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} The loaded model or models (depending on relationship type) */\n  loaded() {\n    if (!this._preloaded && this.model.isPersisted()) {\n      throw new Error(`${this.model.constructor.name}#${this.relationship.getRelationshipName()} hasn't been preloaded`)\n    }\n\n    return this._loaded\n  }\n\n  /** @param {InstanceType<TMC> | Array<InstanceType<TMC>> | undefined} model - Related model(s) to mark as loaded. */\n  setLoaded(model) { this._loaded = model }\n\n  /** @returns {InstanceType<TMC> | InstanceType<TMC>[] | undefined} - The loaded or undefined.  */\n  getLoadedOrUndefined() { return this._loaded }\n\n  /** @returns {boolean} The loaded model or models (depending on relationship type) */\n  getPreloaded() { return this._preloaded || false }\n\n  /** @param {boolean} isPreloaded - Whether the relationship is preloaded. */\n  setPreloaded(isPreloaded) { this._preloaded = isPreloaded }\n\n  /** @returns {string} The foreign key for this relationship */\n  getForeignKey() { return this.getRelationship().getForeignKey() }\n\n  /** @returns {InstanceType<MC>} - The model.  */\n  getModel() { return this.model }\n\n  /** @returns {string} The primary key for this relationship's model */\n  getPrimaryKey() { return this.getRelationship().getPrimaryKey() }\n\n  /** @returns {import(\"../relationships/base.js\").default} The relationship object that this instance relationship is based on */\n  getRelationship() { return this.relationship }\n\n  /**\n   * @template T\n   * @param {T} query - Query instance.\n   * @returns {T} - Scoped query.\n   */\n  applyScope(query) {\n    return this.getRelationship().applyScope(query)\n  }\n\n  /** @returns {TMC | undefined} The model class that this instance relationship */\n  getTargetModelClass() {\n    const TargetModelClass = /** @type {TMC} */ (this.getRelationship().getTargetModelClass())\n\n    return TargetModelClass\n  }\n\n  /** @returns {string} The type of relationship (e.g. \"has_many\", \"belongs_to\", etc.) */\n  getType() { return this.getRelationship().getType() }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"belongs-to.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/belongs-to.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,0EAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;
|
|
1
|
+
{"version":3,"file":"belongs-to.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/belongs-to.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,0EAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;CA2DF;qCAzEoC,WAAW"}
|
|
@@ -27,6 +27,14 @@ export default class VelociousDatabaseRecordBelongsToInstanceRelationship extend
|
|
|
27
27
|
}
|
|
28
28
|
getLoadedOrUndefined() { return this._loaded; }
|
|
29
29
|
async load() {
|
|
30
|
+
// Force-reload: discard the cached value and fetch fresh. When the parent
|
|
31
|
+
// record was loaded as part of a batch, batch the belongs-to lookup across
|
|
32
|
+
// cohort siblings that have not preloaded this relationship yet.
|
|
33
|
+
this._preloaded = false;
|
|
34
|
+
this._loaded = undefined;
|
|
35
|
+
const batched = await this._tryCohortPreload();
|
|
36
|
+
if (batched)
|
|
37
|
+
return this.loaded();
|
|
30
38
|
const foreignKey = this.getForeignKey();
|
|
31
39
|
const foreignModelID = this.getModel().readColumn(foreignKey);
|
|
32
40
|
const TargetModelClass = this.getTargetModelClass();
|
|
@@ -50,4 +58,4 @@ export default class VelociousDatabaseRecordBelongsToInstanceRelationship extend
|
|
|
50
58
|
return this.loaded();
|
|
51
59
|
}
|
|
52
60
|
}
|
|
53
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVsb25ncy10by5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9kYXRhYmFzZS9yZWNvcmQvaW5zdGFuY2UtcmVsYXRpb25zaGlwcy9iZWxvbmdzLXRvLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVk7QUFFWixPQUFPLHdCQUF3QixNQUFNLFdBQVcsQ0FBQTtBQUVoRDs7Ozs7R0FLRztBQUNILE1BQU0sQ0FBQyxPQUFPLE9BQU8sb0RBQXFELFNBQVEsd0JBQXdCO0lBQ3hHOztPQUVHO0lBQ0gsWUFBWSxJQUFJO1FBQ2QsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFBO0lBQ2IsQ0FBQztJQUVEOzs7T0FHRztJQUNILEtBQUssQ0FBQyxJQUFJO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLENBQUE7UUFFeEUsSUFBSSxDQUFDLGdCQUFnQjtZQUFFLE1BQU0sSUFBSSxLQUFLLENBQUMsaURBQWlELENBQUMsQ0FBQTtRQUV6RixNQUFNLFdBQVcsR0FBRyxnQ0FBZ0MsQ0FBQyxDQUFDLElBQUksZ0JBQWdCLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQTtRQUVqRixJQUFJLENBQUMsT0FBTyxHQUFHLFdBQVcsQ0FBQTtRQUUxQixPQUFPLFdBQVcsQ0FBQTtJQUNwQixDQUFDO0lBRUQsb0JBQW9CLEtBQUssT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFBLENBQUMsQ0FBQztJQUU5QyxLQUFLLENBQUMsSUFBSTtRQUNSLDBFQUEwRTtRQUMxRSwyRUFBMkU7UUFDM0UsaUVBQWlFO1FBQ2pFLElBQUksQ0FBQyxVQUFVLEdBQUcsS0FBSyxDQUFBO1FBQ3ZCLElBQUksQ0FBQyxPQUFPLEdBQUcsU0FBUyxDQUFBO1FBRXhCLE1BQU0sT0FBTyxHQUFHLE1BQU0sSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUE7UUFFOUMsSUFBSSxPQUFPO1lBQUUsT0FBTyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUE7UUFFakMsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFBO1FBQ3ZDLE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLENBQUE7UUFDN0QsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsQ0FBQTtRQUVuRCxJQUFJLENBQUMsZ0JBQWdCO1lBQUUsTUFBTSxJQUFJLEtBQUssQ0FBQyxtQ0FBbUMsQ0FBQyxDQUFBO1FBRTNFLE1BQU0sVUFBVSxHQUFHLGdCQUFnQixDQUFDLFVBQVUsRUFBRSxDQUFBO1FBQ2hELGlFQUFpRTtRQUNqRSxNQUFNLFNBQVMsR0FBRyxFQUFFLENBQUE7UUFFcEIsU0FBUyxDQUFDLFVBQVUsQ0FBQyxHQUFHLGNBQWMsQ0FBQTtRQUV0QyxJQUFJLEtBQUssR0FBRyxnQkFBZ0IsQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUE7UUFFN0MsS0FBSyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsS0FBSyxDQUFDLENBQUE7UUFFOUIsTUFBTSxZQUFZLEdBQUcsTUFBTSxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUE7UUFFeEMsSUFBSSxZQUFZLEVBQUUsQ0FBQztZQUNqQixJQUFJLENBQUMsU0FBUyxDQUFDLFlBQVksQ0FBQyxDQUFBO1FBQzlCLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQTtRQUMzQixDQUFDO1FBQ0QsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQTtRQUNwQixJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFBO1FBRXZCLE9BQU8sSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFBO0lBQ3RCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQgQmFzZUluc3RhbmNlUmVsYXRpb25zaGlwIGZyb20gXCIuL2Jhc2UuanNcIlxuXG4vKipcbiAqIEEgZ2VuZXJpYyBxdWVyeSBvdmVyIHNvbWUgbW9kZWwgdHlwZS5cbiAqIEB0ZW1wbGF0ZSB7dHlwZW9mIGltcG9ydChcIi4uL2luZGV4LmpzXCIpLmRlZmF1bHR9IE1DXG4gKiBAdGVtcGxhdGUge3R5cGVvZiBpbXBvcnQoXCIuLi9pbmRleC5qc1wiKS5kZWZhdWx0fSBUTUNcbiAqIEBhdWdtZW50cyB7QmFzZUluc3RhbmNlUmVsYXRpb25zaGlwPE1DLCBUTUM+fVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBWZWxvY2lvdXNEYXRhYmFzZVJlY29yZEJlbG9uZ3NUb0luc3RhbmNlUmVsYXRpb25zaGlwIGV4dGVuZHMgQmFzZUluc3RhbmNlUmVsYXRpb25zaGlwIHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi9iYXNlLmpzXCIpLkluc3RhbmNlUmVsYXRpb25zaGlwc0Jhc2VBcmdzPE1DLCBUTUM+fSBhcmdzIC0gT3B0aW9ucyBvYmplY3QuXG4gICAqL1xuICBjb25zdHJ1Y3RvcihhcmdzKSB7XG4gICAgc3VwZXIoYXJncylcbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IGRhdGEgLSBEYXRhIHBheWxvYWQuXG4gICAqIEByZXR1cm5zIHtJbnN0YW5jZVR5cGU8VE1DPn0gLSBUaGUgYnVpbGQuXG4gICAqL1xuICBidWlsZChkYXRhKSB7XG4gICAgY29uc3QgVGFyZ2V0TW9kZWxDbGFzcyA9IC8qKiBAdHlwZSB7VE1DfSAqLyAodGhpcy5nZXRUYXJnZXRNb2RlbENsYXNzKCkpXG5cbiAgICBpZiAoIVRhcmdldE1vZGVsQ2xhc3MpIHRocm93IG5ldyBFcnJvcihcIkNhbid0IGJ1aWxkIGEgbmV3IHJlY29yZCB3aXRob3V0IGEgdGFyZ2V0IG1vZGVsXCIpXG5cbiAgICBjb25zdCBuZXdJbnN0YW5jZSA9IC8qKiBAdHlwZSB7SW5zdGFuY2VUeXBlPFRNQz59ICovIChuZXcgVGFyZ2V0TW9kZWxDbGFzcyhkYXRhKSlcblxuICAgIHRoaXMuX2xvYWRlZCA9IG5ld0luc3RhbmNlXG5cbiAgICByZXR1cm4gbmV3SW5zdGFuY2VcbiAgfVxuXG4gIGdldExvYWRlZE9yVW5kZWZpbmVkKCkgeyByZXR1cm4gdGhpcy5fbG9hZGVkIH1cblxuICBhc3luYyBsb2FkKCkge1xuICAgIC8vIEZvcmNlLXJlbG9hZDogZGlzY2FyZCB0aGUgY2FjaGVkIHZhbHVlIGFuZCBmZXRjaCBmcmVzaC4gV2hlbiB0aGUgcGFyZW50XG4gICAgLy8gcmVjb3JkIHdhcyBsb2FkZWQgYXMgcGFydCBvZiBhIGJhdGNoLCBiYXRjaCB0aGUgYmVsb25ncy10byBsb29rdXAgYWNyb3NzXG4gICAgLy8gY29ob3J0IHNpYmxpbmdzIHRoYXQgaGF2ZSBub3QgcHJlbG9hZGVkIHRoaXMgcmVsYXRpb25zaGlwIHlldC5cbiAgICB0aGlzLl9wcmVsb2FkZWQgPSBmYWxzZVxuICAgIHRoaXMuX2xvYWRlZCA9IHVuZGVmaW5lZFxuXG4gICAgY29uc3QgYmF0Y2hlZCA9IGF3YWl0IHRoaXMuX3RyeUNvaG9ydFByZWxvYWQoKVxuXG4gICAgaWYgKGJhdGNoZWQpIHJldHVybiB0aGlzLmxvYWRlZCgpXG5cbiAgICBjb25zdCBmb3JlaWduS2V5ID0gdGhpcy5nZXRGb3JlaWduS2V5KClcbiAgICBjb25zdCBmb3JlaWduTW9kZWxJRCA9IHRoaXMuZ2V0TW9kZWwoKS5yZWFkQ29sdW1uKGZvcmVpZ25LZXkpXG4gICAgY29uc3QgVGFyZ2V0TW9kZWxDbGFzcyA9IHRoaXMuZ2V0VGFyZ2V0TW9kZWxDbGFzcygpXG5cbiAgICBpZiAoIVRhcmdldE1vZGVsQ2xhc3MpIHRocm93IG5ldyBFcnJvcihcIkNhbid0IGxvYWQgd2l0aG91dCBhIHRhcmdldCBtb2RlbFwiKVxuXG4gICAgY29uc3QgcHJpbWFyeUtleSA9IFRhcmdldE1vZGVsQ2xhc3MucHJpbWFyeUtleSgpXG4gICAgLyoqIEB0eXBlIHtSZWNvcmQ8c3RyaW5nLCBzdHJpbmcgfCBudW1iZXIgfCBudWxsIHwgdW5kZWZpbmVkPn0gKi9cbiAgICBjb25zdCB3aGVyZUFyZ3MgPSB7fVxuXG4gICAgd2hlcmVBcmdzW3ByaW1hcnlLZXldID0gZm9yZWlnbk1vZGVsSURcblxuICAgIGxldCBxdWVyeSA9IFRhcmdldE1vZGVsQ2xhc3Mud2hlcmUod2hlcmVBcmdzKVxuXG4gICAgcXVlcnkgPSB0aGlzLmFwcGx5U2NvcGUocXVlcnkpXG5cbiAgICBjb25zdCBmb3JlaWduTW9kZWwgPSBhd2FpdCBxdWVyeS5maXJzdCgpXG5cbiAgICBpZiAoZm9yZWlnbk1vZGVsKSB7XG4gICAgICB0aGlzLnNldExvYWRlZChmb3JlaWduTW9kZWwpXG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMuc2V0TG9hZGVkKHVuZGVmaW5lZClcbiAgICB9XG4gICAgdGhpcy5zZXREaXJ0eShmYWxzZSlcbiAgICB0aGlzLnNldFByZWxvYWRlZCh0cnVlKVxuXG4gICAgcmV0dXJuIHRoaXMubG9hZGVkKClcbiAgfVxufVxuIl19
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"has-many.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-many.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,wEAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;IA2DD;;;OAGG;IACH,aAHW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAQtC;IAED,6EAA6E;IAC7E,QADc,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,
|
|
1
|
+
{"version":3,"file":"has-many.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-many.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,wEAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;IA2DD;;;OAGG;IACH,aAHW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAQtC;IAED,6EAA6E;IAC7E,QADc,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAuBzC;IAED,yEAAyE;IACzE,WADc,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAOzC;IAED;;;OAGG;IACH,kBAHW,OAAO,sBAAsB,EAAE,mBAAmB,GAChD,OAAO,kCAAkC,EAAE,OAAO,CAAC,GAAG,CAAC,CAInE;IAED;;;OAGG;IACH,cAHW,MAAM,GAAG,MAAM,GACb,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAItC;IAED,uFAAuF;IACvF,SADc,OAAO,kCAAkC,EAAE,OAAO,CAAC,GAAG,CAAC,CA6CpE;IAED,sGAAsG;IACtG,UADc,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAWrC;IA8BD;;;OAGG;IACH,kBAHW,YAAY,CAAC,GAAG,CAAC,EAAE,GACjB,IAAI,CAMhB;CACF;qCAxOoC,WAAW"}
|
|
@@ -69,6 +69,17 @@ export default class VelociousDatabaseRecordHasManyInstanceRelationship extends
|
|
|
69
69
|
}
|
|
70
70
|
/** @returns {Promise<InstanceType<TMC>[]>} - Resolves with loaded models. */
|
|
71
71
|
async load() {
|
|
72
|
+
// Force-reload: discard the cached value and fetch fresh. When the parent
|
|
73
|
+
// record was loaded as part of a batch, route through the cohort preloader
|
|
74
|
+
// so siblings that have not preloaded this relationship get batched too.
|
|
75
|
+
// The scoped query path (`instance.query().where(...).load()`) bypasses
|
|
76
|
+
// this and stays a single-record load by design.
|
|
77
|
+
this._preloaded = false;
|
|
78
|
+
this._loaded = undefined;
|
|
79
|
+
const batched = await this._tryCohortPreload();
|
|
80
|
+
if (batched) {
|
|
81
|
+
return /** @type {InstanceType<TMC>[]} */ (Array.isArray(this._loaded) ? this._loaded : []);
|
|
82
|
+
}
|
|
72
83
|
const foreignModels = /** @type {InstanceType<TMC>[]} */ (await this.query().load());
|
|
73
84
|
this.setLoaded(foreignModels);
|
|
74
85
|
this.setDirty(false);
|
|
@@ -77,11 +88,10 @@ export default class VelociousDatabaseRecordHasManyInstanceRelationship extends
|
|
|
77
88
|
}
|
|
78
89
|
/** @returns {Promise<InstanceType<TMC>[]>} - Resolves with the array. */
|
|
79
90
|
async toArray() {
|
|
80
|
-
const loadedValue = this.
|
|
81
|
-
if (loadedValue
|
|
82
|
-
return
|
|
83
|
-
|
|
84
|
-
return await this.load();
|
|
91
|
+
const loadedValue = await this.autoloadOrLoad();
|
|
92
|
+
if (loadedValue === undefined)
|
|
93
|
+
return [];
|
|
94
|
+
return Array.isArray(loadedValue) ? loadedValue : [loadedValue];
|
|
85
95
|
}
|
|
86
96
|
/**
|
|
87
97
|
* @param {import("../../query/index.js").NestedPreloadRecord} preloads - Preload map for related records.
|
|
@@ -186,4 +196,4 @@ export default class VelociousDatabaseRecordHasManyInstanceRelationship extends
|
|
|
186
196
|
this._loaded = models;
|
|
187
197
|
}
|
|
188
198
|
}
|
|
189
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"has-many.js","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-many.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseInstanceRelationship from \"./base.js\"\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../index.js\").default} MC\n * @template {typeof import(\"../index.js\").default} TMC\n * @augments {BaseInstanceRelationship<MC, TMC>}\n */\nexport default class VelociousDatabaseRecordHasManyInstanceRelationship extends BaseInstanceRelationship {\n  /**\n   * @param {import(\"./base.js\").InstanceRelationshipsBaseArgs<MC, TMC>} args - Options object.\n   */\n  constructor(args) {\n    super(args)\n  }\n\n  /**\n   * @param {Record<string, any>} data - Data payload.\n   * @returns {InstanceType<TMC>} - The build.\n   */\n  build(data) {\n    // Spawn new model of the targeted class\n    const targetModelClass = this.getTargetModelClass()\n\n    if (!targetModelClass) throw new Error(\"Can't build a new record without a taget model class\")\n\n    const newInstance = /** @type {InstanceType<TMC>} */ (new targetModelClass(data))\n\n\n    // Add it to the loaded models of this relationship\n    if (this._loaded === undefined) {\n      this._loaded = [newInstance]\n    } else if (Array.isArray(this._loaded)) {\n      this._loaded.push(newInstance)\n    } else {\n      throw new Error(`Loaded had an unexpected type: ${typeof this._loaded}`)\n    }\n\n\n    // Set loaded on the models inversed relationship\n    const inverseOf = this.getRelationship().getInverseOf()\n\n    if (inverseOf) {\n      const inverseInstanceRelationship = newInstance.getRelationshipByName(inverseOf)\n\n      inverseInstanceRelationship.setAutoSave(false)\n      inverseInstanceRelationship.setLoaded(this.getModel())\n    }\n\n\n    // Assign the foreign key to the new model\n    const parentModel = this.getModel()\n\n    if (parentModel.isPersisted()) {\n      const foreignKeyName = this.getForeignKey()\n      const columnNameMap = targetModelClass.getColumnNameToAttributeNameMap()\n      const foreignKeyAttributeName = columnNameMap[foreignKeyName]\n      if (!foreignKeyAttributeName) throw new Error(`Unknown foreign key attribute name for ${foreignKeyName}`)\n      const primaryKeyName = this.getPrimaryKey()\n      const foreignKeyValue = parentModel.readColumn(primaryKeyName)\n      /** @type {Record<string, any>} */\n      const assignData = {}\n\n      assignData[foreignKeyAttributeName] = foreignKeyValue\n\n      newInstance.assign(assignData)\n    }\n\n\n    // Return the new contructed model\n    return newInstance\n  }\n\n  /**\n   * @param {Record<string, any>} data - Data payload.\n   * @returns {Promise<InstanceType<TMC>>} - Resolves with the create.\n   */\n  async create(data) {\n    const model = this.build(data)\n\n    await model.save()\n\n    return model\n  }\n\n  /** @returns {Promise<InstanceType<TMC>[]>} - Resolves with loaded models. */\n  async load() {\n    const foreignModels = /** @type {InstanceType<TMC>[]} */ (await this.query().load())\n\n    this.setLoaded(foreignModels)\n    this.setDirty(false)\n    this.setPreloaded(true)\n\n    return foreignModels\n  }\n\n  /** @returns {Promise<InstanceType<TMC>[]>} - Resolves with the array. */\n  async toArray() {\n    const loadedValue = this.getLoadedOrUndefined()\n\n    if (loadedValue !== undefined) {\n      return Array.isArray(loadedValue) ? loadedValue : []\n    }\n\n    return await this.load()\n  }\n\n  /**\n   * @param {import(\"../../query/index.js\").NestedPreloadRecord} preloads - Preload map for related records.\n   * @returns {import(\"../../query/model-class-query.js\").default<TMC>} - The preload.\n   */\n  preload(preloads) {\n    return this.query().clone().preload(preloads)\n  }\n\n  /**\n   * @param {string | number} modelID - Related model identifier.\n   * @returns {Promise<InstanceType<TMC>>} - Resolves with the find.\n   */\n  async find(modelID) {\n    return /** @type {Promise<InstanceType<TMC>>} */ (this.query().find(modelID))\n  }\n\n  /** @returns {import(\"../../query/model-class-query.js\").default<TMC>} - The query.  */\n  query() {\n    if (!this.getModel().isPersisted()) throw new Error(\"Cannot build a query for an unpersisted parent model\")\n\n    const TargetModelClass = this.getTargetModelClass()\n\n    if (!TargetModelClass) throw new Error(\"Cannot load without a target model class\")\n\n    const throughRelationshipName = this.getRelationship().through\n\n    if (throughRelationshipName) {\n      const parentModelClass = this.getModel().getModelClass()\n      const throughRelationship = parentModelClass.getRelationshipByName(throughRelationshipName)\n      const throughModelClass = throughRelationship.getTargetModelClass()\n\n      if (!throughModelClass) throw new Error(`Through relationship ${throughRelationshipName} has no target model class`)\n\n      const throughForeignKey = throughRelationship.getForeignKey()\n      const throughPrimaryKey = throughRelationship.getPrimaryKey()\n      const targetForeignKey = this.getForeignKey()\n      const targetTable = TargetModelClass.tableName()\n      const throughTable = throughModelClass.tableName()\n      const driver = TargetModelClass.connection()\n      const parentPrimaryKey = this.getPrimaryKey()\n      const parentId = /** @type {string | number} */ (this.getModel().readColumn(parentPrimaryKey))\n      const joinSql = `LEFT JOIN ${driver.quoteTable(throughTable)} ON ${driver.quoteTable(throughTable)}.${driver.quoteColumn(throughPrimaryKey)} = ${driver.quoteTable(targetTable)}.${driver.quoteColumn(targetForeignKey)}`\n      const whereSql = `${driver.quoteTable(throughTable)}.${driver.quoteColumn(throughForeignKey)} = ${driver.options().quote(parentId)}`\n\n      const query = TargetModelClass.joins(joinSql).where(whereSql)\n\n      return this.applyScope(query)\n    }\n\n    const foreignKey = this.getForeignKey()\n    const primaryKey = this.getPrimaryKey()\n    const primaryModelID = /** @type {string | number} */ (this.getModel().readColumn(primaryKey))\n\n    /** @type {Record<string, string | number>} */\n    const whereArgs = {}\n\n    whereArgs[foreignKey] = primaryModelID\n\n    const query = TargetModelClass.where(whereArgs)\n\n    return this.applyScope(query)\n  }\n\n  /** @returns {Array<InstanceType<TMC>>} The loaded model or models (depending on relationship type) */\n  loaded() {\n    if (!this._preloaded && this.model.isPersisted()) {\n      throw new Error(`${this.model.constructor.name}#${this.relationship.getRelationshipName()} hasn't been preloaded`)\n    }\n\n    if (this._loaded === undefined && this.model.isNewRecord()) {\n      return []\n    }\n\n    return Array.isArray(this._loaded) ? this._loaded : []\n  }\n\n  /**\n   * @param {InstanceType<TMC>[] | InstanceType<TMC>} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  addToLoaded(models) {\n    if (!models) {\n      throw new Error(\"Need to give something\")\n    } else if (Array.isArray(models)) {\n      for (const model of models) {\n        if (this._loaded === undefined) {\n          this._loaded = [model]\n        } else if (Array.isArray(this._loaded)) {\n          this._loaded.push(model)\n        } else {\n          throw new Error(`Unexpected loaded type: ${typeof this._loaded}`)\n        }\n      }\n    } else {\n      if (this._loaded === undefined) {\n        this._loaded = [models]\n      } else if (Array.isArray(this._loaded)) {\n        this._loaded.push(models)\n      } else {\n        throw new Error(`Unexpected loaded type: ${typeof this._loaded}`)\n      }\n    }\n  }\n\n  /**\n   * @param {InstanceType<TMC>[]} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  setLoaded(models) {\n    if (!Array.isArray(models)) throw new Error(`Argument given to setLoaded wasn't an array: ${typeof models}`)\n\n    this._loaded = models\n  }\n}\n"]}
|
|
199
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"has-many.js","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-many.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseInstanceRelationship from \"./base.js\"\n\n/**\n * A generic query over some model type.\n * @template {typeof import(\"../index.js\").default} MC\n * @template {typeof import(\"../index.js\").default} TMC\n * @augments {BaseInstanceRelationship<MC, TMC>}\n */\nexport default class VelociousDatabaseRecordHasManyInstanceRelationship extends BaseInstanceRelationship {\n  /**\n   * @param {import(\"./base.js\").InstanceRelationshipsBaseArgs<MC, TMC>} args - Options object.\n   */\n  constructor(args) {\n    super(args)\n  }\n\n  /**\n   * @param {Record<string, any>} data - Data payload.\n   * @returns {InstanceType<TMC>} - The build.\n   */\n  build(data) {\n    // Spawn new model of the targeted class\n    const targetModelClass = this.getTargetModelClass()\n\n    if (!targetModelClass) throw new Error(\"Can't build a new record without a taget model class\")\n\n    const newInstance = /** @type {InstanceType<TMC>} */ (new targetModelClass(data))\n\n\n    // Add it to the loaded models of this relationship\n    if (this._loaded === undefined) {\n      this._loaded = [newInstance]\n    } else if (Array.isArray(this._loaded)) {\n      this._loaded.push(newInstance)\n    } else {\n      throw new Error(`Loaded had an unexpected type: ${typeof this._loaded}`)\n    }\n\n\n    // Set loaded on the models inversed relationship\n    const inverseOf = this.getRelationship().getInverseOf()\n\n    if (inverseOf) {\n      const inverseInstanceRelationship = newInstance.getRelationshipByName(inverseOf)\n\n      inverseInstanceRelationship.setAutoSave(false)\n      inverseInstanceRelationship.setLoaded(this.getModel())\n    }\n\n\n    // Assign the foreign key to the new model\n    const parentModel = this.getModel()\n\n    if (parentModel.isPersisted()) {\n      const foreignKeyName = this.getForeignKey()\n      const columnNameMap = targetModelClass.getColumnNameToAttributeNameMap()\n      const foreignKeyAttributeName = columnNameMap[foreignKeyName]\n      if (!foreignKeyAttributeName) throw new Error(`Unknown foreign key attribute name for ${foreignKeyName}`)\n      const primaryKeyName = this.getPrimaryKey()\n      const foreignKeyValue = parentModel.readColumn(primaryKeyName)\n      /** @type {Record<string, any>} */\n      const assignData = {}\n\n      assignData[foreignKeyAttributeName] = foreignKeyValue\n\n      newInstance.assign(assignData)\n    }\n\n\n    // Return the new contructed model\n    return newInstance\n  }\n\n  /**\n   * @param {Record<string, any>} data - Data payload.\n   * @returns {Promise<InstanceType<TMC>>} - Resolves with the create.\n   */\n  async create(data) {\n    const model = this.build(data)\n\n    await model.save()\n\n    return model\n  }\n\n  /** @returns {Promise<InstanceType<TMC>[]>} - Resolves with loaded models. */\n  async load() {\n    // Force-reload: discard the cached value and fetch fresh. When the parent\n    // record was loaded as part of a batch, route through the cohort preloader\n    // so siblings that have not preloaded this relationship get batched too.\n    // The scoped query path (`instance.query().where(...).load()`) bypasses\n    // this and stays a single-record load by design.\n    this._preloaded = false\n    this._loaded = undefined\n\n    const batched = await this._tryCohortPreload()\n\n    if (batched) {\n      return /** @type {InstanceType<TMC>[]} */ (Array.isArray(this._loaded) ? this._loaded : [])\n    }\n\n    const foreignModels = /** @type {InstanceType<TMC>[]} */ (await this.query().load())\n\n    this.setLoaded(foreignModels)\n    this.setDirty(false)\n    this.setPreloaded(true)\n\n    return foreignModels\n  }\n\n  /** @returns {Promise<InstanceType<TMC>[]>} - Resolves with the array. */\n  async toArray() {\n    const loadedValue = await this.autoloadOrLoad()\n\n    if (loadedValue === undefined) return []\n\n    return Array.isArray(loadedValue) ? loadedValue : [loadedValue]\n  }\n\n  /**\n   * @param {import(\"../../query/index.js\").NestedPreloadRecord} preloads - Preload map for related records.\n   * @returns {import(\"../../query/model-class-query.js\").default<TMC>} - The preload.\n   */\n  preload(preloads) {\n    return this.query().clone().preload(preloads)\n  }\n\n  /**\n   * @param {string | number} modelID - Related model identifier.\n   * @returns {Promise<InstanceType<TMC>>} - Resolves with the find.\n   */\n  async find(modelID) {\n    return /** @type {Promise<InstanceType<TMC>>} */ (this.query().find(modelID))\n  }\n\n  /** @returns {import(\"../../query/model-class-query.js\").default<TMC>} - The query.  */\n  query() {\n    if (!this.getModel().isPersisted()) throw new Error(\"Cannot build a query for an unpersisted parent model\")\n\n    const TargetModelClass = this.getTargetModelClass()\n\n    if (!TargetModelClass) throw new Error(\"Cannot load without a target model class\")\n\n    const throughRelationshipName = this.getRelationship().through\n\n    if (throughRelationshipName) {\n      const parentModelClass = this.getModel().getModelClass()\n      const throughRelationship = parentModelClass.getRelationshipByName(throughRelationshipName)\n      const throughModelClass = throughRelationship.getTargetModelClass()\n\n      if (!throughModelClass) throw new Error(`Through relationship ${throughRelationshipName} has no target model class`)\n\n      const throughForeignKey = throughRelationship.getForeignKey()\n      const throughPrimaryKey = throughRelationship.getPrimaryKey()\n      const targetForeignKey = this.getForeignKey()\n      const targetTable = TargetModelClass.tableName()\n      const throughTable = throughModelClass.tableName()\n      const driver = TargetModelClass.connection()\n      const parentPrimaryKey = this.getPrimaryKey()\n      const parentId = /** @type {string | number} */ (this.getModel().readColumn(parentPrimaryKey))\n      const joinSql = `LEFT JOIN ${driver.quoteTable(throughTable)} ON ${driver.quoteTable(throughTable)}.${driver.quoteColumn(throughPrimaryKey)} = ${driver.quoteTable(targetTable)}.${driver.quoteColumn(targetForeignKey)}`\n      const whereSql = `${driver.quoteTable(throughTable)}.${driver.quoteColumn(throughForeignKey)} = ${driver.options().quote(parentId)}`\n\n      const query = TargetModelClass.joins(joinSql).where(whereSql)\n\n      return this.applyScope(query)\n    }\n\n    const foreignKey = this.getForeignKey()\n    const primaryKey = this.getPrimaryKey()\n    const primaryModelID = /** @type {string | number} */ (this.getModel().readColumn(primaryKey))\n\n    /** @type {Record<string, string | number>} */\n    const whereArgs = {}\n\n    whereArgs[foreignKey] = primaryModelID\n\n    const query = TargetModelClass.where(whereArgs)\n\n    return this.applyScope(query)\n  }\n\n  /** @returns {Array<InstanceType<TMC>>} The loaded model or models (depending on relationship type) */\n  loaded() {\n    if (!this._preloaded && this.model.isPersisted()) {\n      throw new Error(`${this.model.constructor.name}#${this.relationship.getRelationshipName()} hasn't been preloaded`)\n    }\n\n    if (this._loaded === undefined && this.model.isNewRecord()) {\n      return []\n    }\n\n    return Array.isArray(this._loaded) ? this._loaded : []\n  }\n\n  /**\n   * @param {InstanceType<TMC>[] | InstanceType<TMC>} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  addToLoaded(models) {\n    if (!models) {\n      throw new Error(\"Need to give something\")\n    } else if (Array.isArray(models)) {\n      for (const model of models) {\n        if (this._loaded === undefined) {\n          this._loaded = [model]\n        } else if (Array.isArray(this._loaded)) {\n          this._loaded.push(model)\n        } else {\n          throw new Error(`Unexpected loaded type: ${typeof this._loaded}`)\n        }\n      }\n    } else {\n      if (this._loaded === undefined) {\n        this._loaded = [models]\n      } else if (Array.isArray(this._loaded)) {\n        this._loaded.push(models)\n      } else {\n        throw new Error(`Unexpected loaded type: ${typeof this._loaded}`)\n      }\n    }\n  }\n\n  /**\n   * @param {InstanceType<TMC>[]} models - Model instances.\n   * @returns {void} - No return value.\n   */\n  setLoaded(models) {\n    if (!Array.isArray(models)) throw new Error(`Argument given to setLoaded wasn't an array: ${typeof models}`)\n\n    this._loaded = models\n  }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"has-one.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-one.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,uEAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;IAED,4CAA4C;IAC5C,SADW,YAAY,CAAC,GAAG,CAAC,GAAG,SAAS,CACrB;
|
|
1
|
+
{"version":3,"file":"has-one.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/instance-relationships/has-one.js"],"names":[],"mappings":"AAIA;;;;;GAKG;AACH,uEAJoD,EAAE,SAAzC,cAAe,aAAa,EAAE,OAAQ,EACC,GAAG,SAA1C,cAAe,aAAa,EAAE,OAAQ;IAIjD;;OAEG;IACH,kBAFW,OAAO,WAAW,EAAE,6BAA6B,CAAC,EAAE,EAAE,GAAG,CAAC,EAIpE;IAED,4CAA4C;IAC5C,SADW,YAAY,CAAC,GAAG,CAAC,GAAG,SAAS,CACrB;IAqEnB,sDAA8C;CAU/C;qCAhGoC,WAAW"}
|
|
@@ -28,6 +28,14 @@ export default class VelociousDatabaseRecordHasOneInstanceRelationship extends B
|
|
|
28
28
|
return newInstance;
|
|
29
29
|
}
|
|
30
30
|
async load() {
|
|
31
|
+
// Force-reload: discard the cached value and fetch fresh. When the parent
|
|
32
|
+
// record was loaded as part of a batch, batch the has-one lookup across
|
|
33
|
+
// cohort siblings that have not preloaded this relationship yet.
|
|
34
|
+
this._preloaded = false;
|
|
35
|
+
this._loaded = undefined;
|
|
36
|
+
const batched = await this._tryCohortPreload();
|
|
37
|
+
if (batched)
|
|
38
|
+
return this.loaded();
|
|
31
39
|
const foreignKey = this.getForeignKey();
|
|
32
40
|
const primaryKey = this.getPrimaryKey();
|
|
33
41
|
const primaryModelID = /** @type {string | number} */ (this.getModel().readColumn(primaryKey));
|
|
@@ -68,4 +76,4 @@ export default class VelociousDatabaseRecordHasOneInstanceRelationship extends B
|
|
|
68
76
|
}
|
|
69
77
|
getTargetModelClass() { return /** @type {TMC | undefined} */ (this.relationship.getTargetModelClass()); }
|
|
70
78
|
}
|
|
71
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFzLW9uZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9kYXRhYmFzZS9yZWNvcmQvaW5zdGFuY2UtcmVsYXRpb25zaGlwcy9oYXMtb25lLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFlBQVk7QUFFWixPQUFPLHdCQUF3QixNQUFNLFdBQVcsQ0FBQTtBQUVoRDs7Ozs7R0FLRztBQUNILE1BQU0sQ0FBQyxPQUFPLE9BQU8saURBQWtELFNBQVEsd0JBQXdCO0lBQ3JHOztPQUVHO0lBQ0gsWUFBWSxJQUFJO1FBQ2QsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFBO0lBQ2IsQ0FBQztJQUVELDRDQUE0QztJQUM1QyxPQUFPLEdBQUcsU0FBUyxDQUFBO0lBRW5COzs7T0FHRztJQUNILEtBQUssQ0FBQyxJQUFJO1FBQ1IsTUFBTSxnQkFBZ0IsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLENBQUE7UUFFeEUsSUFBSSxDQUFDLGdCQUFnQjtZQUFFLE1BQU0sSUFBSSxLQUFLLENBQUMsdURBQXVELENBQUMsQ0FBQTtRQUUvRixNQUFNLFdBQVcsR0FBRyxnQ0FBZ0MsQ0FBQyxDQUFDLElBQUksZ0JBQWdCLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQTtRQUVqRixJQUFJLENBQUMsT0FBTyxHQUFHLFdBQVcsQ0FBQTtRQUUxQixPQUFPLFdBQVcsQ0FBQTtJQUNwQixDQUFDO0lBRUQsS0FBSyxDQUFDLElBQUk7UUFDUiwwRUFBMEU7UUFDMUUsd0VBQXdFO1FBQ3hFLGlFQUFpRTtRQUNqRSxJQUFJLENBQUMsVUFBVSxHQUFHLEtBQUssQ0FBQTtRQUN2QixJQUFJLENBQUMsT0FBTyxHQUFHLFNBQVMsQ0FBQTtRQUV4QixNQUFNLE9BQU8sR0FBRyxNQUFNLElBQUksQ0FBQyxpQkFBaUIsRUFBRSxDQUFBO1FBRTlDLElBQUksT0FBTztZQUFFLE9BQU8sSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFBO1FBRWpDLE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQTtRQUN2QyxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUE7UUFDdkMsTUFBTSxjQUFjLEdBQUcsOEJBQThCLENBQUMsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsVUFBVSxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUE7UUFDOUYsTUFBTSxnQkFBZ0IsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDLENBQUE7UUFFeEUsSUFBSSxDQUFDLGdCQUFnQjtZQUFFLE1BQU0sSUFBSSxLQUFLLENBQUMseUNBQXlDLENBQUMsQ0FBQTtRQUVqRiw4Q0FBOEM7UUFDOUMsTUFBTSxTQUFTLEdBQUcsRUFBRSxDQUFBO1FBRXBCLFNBQVMsQ0FBQyxVQUFVLENBQUMsR0FBRyxjQUFjLENBQUE7UUFFdEMsSUFBSSxLQUFLLEdBQUcsZ0JBQWdCLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFBO1FBRTdDLEtBQUssR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBRTlCLE1BQU0sWUFBWSxHQUFHLE1BQU0sS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFBO1FBRXhDLElBQUksWUFBWSxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxZQUFZLENBQUMsQ0FBQTtRQUM5QixDQUFDO2FBQU0sQ0FBQztZQUNOLElBQUksQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUE7UUFDM0IsQ0FBQztRQUNELElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUE7UUFDcEIsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQTtRQUV2QixPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQTtJQUN0QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNO1FBQ0osSUFBSSxDQUFDLElBQUksQ0FBQyxVQUFVLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsRUFBRSxDQUFDO1lBQ2pELE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxJQUFJLElBQUksSUFBSSxDQUFDLFlBQVksQ0FBQyxtQkFBbUIsRUFBRSx3QkFBd0IsQ0FBQyxDQUFBO1FBQ3BILENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUE7SUFDckIsQ0FBQztJQUVELG9CQUFvQixLQUFLLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQSxDQUFDLENBQUM7SUFFOUMsa0dBQWtHO0lBQ2xHLFNBQVMsQ0FBQyxLQUFLO1FBQ2IsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQztZQUFFLE1BQU0sSUFBSSxLQUFLLENBQUMsNkNBQTZDLE9BQU8sS0FBSyxFQUFFLENBQUMsQ0FBQTtRQUV0RyxJQUFJLENBQUMsT0FBTyxHQUFHLEtBQUssQ0FBQTtJQUN0QixDQUFDO0lBRUQsbUJBQW1CLEtBQUssT0FBTyw4QkFBOEIsQ0FBQyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsbUJBQW1CLEVBQUUsQ0FBQyxDQUFBLENBQUMsQ0FBQztDQUMxRyIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQgQmFzZUluc3RhbmNlUmVsYXRpb25zaGlwIGZyb20gXCIuL2Jhc2UuanNcIlxuXG4vKipcbiAqIEEgZ2VuZXJpYyBxdWVyeSBvdmVyIHNvbWUgbW9kZWwgdHlwZS5cbiAqIEB0ZW1wbGF0ZSB7dHlwZW9mIGltcG9ydChcIi4uL2luZGV4LmpzXCIpLmRlZmF1bHR9IE1DXG4gKiBAdGVtcGxhdGUge3R5cGVvZiBpbXBvcnQoXCIuLi9pbmRleC5qc1wiKS5kZWZhdWx0fSBUTUNcbiAqIEBhdWdtZW50cyB7QmFzZUluc3RhbmNlUmVsYXRpb25zaGlwPE1DLCBUTUM+fVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBWZWxvY2lvdXNEYXRhYmFzZVJlY29yZEhhc09uZUluc3RhbmNlUmVsYXRpb25zaGlwIGV4dGVuZHMgQmFzZUluc3RhbmNlUmVsYXRpb25zaGlwIHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi9iYXNlLmpzXCIpLkluc3RhbmNlUmVsYXRpb25zaGlwc0Jhc2VBcmdzPE1DLCBUTUM+fSBhcmdzIC0gT3B0aW9ucyBvYmplY3QuXG4gICAqL1xuICBjb25zdHJ1Y3RvcihhcmdzKSB7XG4gICAgc3VwZXIoYXJncylcbiAgfVxuXG4gIC8qKiBAdHlwZSB7SW5zdGFuY2VUeXBlPFRNQz4gfCB1bmRlZmluZWR9ICovXG4gIF9sb2FkZWQgPSB1bmRlZmluZWRcblxuICAvKipcbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBkYXRhIC0gRGF0YSBwYXlsb2FkLlxuICAgKiBAcmV0dXJucyB7SW5zdGFuY2VUeXBlPFRNQz59IC0gVGhlIGJ1aWxkLlxuICAgKi9cbiAgYnVpbGQoZGF0YSkge1xuICAgIGNvbnN0IFRhcmdldE1vZGVsQ2xhc3MgPSAvKiogQHR5cGUge1RNQ30gKi8gKHRoaXMuZ2V0VGFyZ2V0TW9kZWxDbGFzcygpKVxuXG4gICAgaWYgKCFUYXJnZXRNb2RlbENsYXNzKSB0aHJvdyBuZXcgRXJyb3IoXCJDYW4ndCBidWlsZCBhIG5ldyByZWNvcmQgd2l0aG91dCBhIHRhcmdldCBtb2RlbCBjbGFzc1wiKVxuXG4gICAgY29uc3QgbmV3SW5zdGFuY2UgPSAvKiogQHR5cGUge0luc3RhbmNlVHlwZTxUTUM+fSAqLyAobmV3IFRhcmdldE1vZGVsQ2xhc3MoZGF0YSkpXG5cbiAgICB0aGlzLl9sb2FkZWQgPSBuZXdJbnN0YW5jZVxuXG4gICAgcmV0dXJuIG5ld0luc3RhbmNlXG4gIH1cblxuICBhc3luYyBsb2FkKCkge1xuICAgIC8vIEZvcmNlLXJlbG9hZDogZGlzY2FyZCB0aGUgY2FjaGVkIHZhbHVlIGFuZCBmZXRjaCBmcmVzaC4gV2hlbiB0aGUgcGFyZW50XG4gICAgLy8gcmVjb3JkIHdhcyBsb2FkZWQgYXMgcGFydCBvZiBhIGJhdGNoLCBiYXRjaCB0aGUgaGFzLW9uZSBsb29rdXAgYWNyb3NzXG4gICAgLy8gY29ob3J0IHNpYmxpbmdzIHRoYXQgaGF2ZSBub3QgcHJlbG9hZGVkIHRoaXMgcmVsYXRpb25zaGlwIHlldC5cbiAgICB0aGlzLl9wcmVsb2FkZWQgPSBmYWxzZVxuICAgIHRoaXMuX2xvYWRlZCA9IHVuZGVmaW5lZFxuXG4gICAgY29uc3QgYmF0Y2hlZCA9IGF3YWl0IHRoaXMuX3RyeUNvaG9ydFByZWxvYWQoKVxuXG4gICAgaWYgKGJhdGNoZWQpIHJldHVybiB0aGlzLmxvYWRlZCgpXG5cbiAgICBjb25zdCBmb3JlaWduS2V5ID0gdGhpcy5nZXRGb3JlaWduS2V5KClcbiAgICBjb25zdCBwcmltYXJ5S2V5ID0gdGhpcy5nZXRQcmltYXJ5S2V5KClcbiAgICBjb25zdCBwcmltYXJ5TW9kZWxJRCA9IC8qKiBAdHlwZSB7c3RyaW5nIHwgbnVtYmVyfSAqLyAodGhpcy5nZXRNb2RlbCgpLnJlYWRDb2x1bW4ocHJpbWFyeUtleSkpXG4gICAgY29uc3QgVGFyZ2V0TW9kZWxDbGFzcyA9IC8qKiBAdHlwZSB7VE1DfSAqLyAodGhpcy5nZXRUYXJnZXRNb2RlbENsYXNzKCkpXG5cbiAgICBpZiAoIVRhcmdldE1vZGVsQ2xhc3MpIHRocm93IG5ldyBFcnJvcihcIkNhbid0IGxvYWQgd2l0aG91dCBhIHRhcmdldCBtb2RlbCBjbGFzc1wiKVxuXG4gICAgLyoqIEB0eXBlIHtSZWNvcmQ8c3RyaW5nLCBzdHJpbmcgfCBudW1iZXI+fSAqL1xuICAgIGNvbnN0IHdoZXJlQXJncyA9IHt9XG5cbiAgICB3aGVyZUFyZ3NbZm9yZWlnbktleV0gPSBwcmltYXJ5TW9kZWxJRFxuXG4gICAgbGV0IHF1ZXJ5ID0gVGFyZ2V0TW9kZWxDbGFzcy53aGVyZSh3aGVyZUFyZ3MpXG5cbiAgICBxdWVyeSA9IHRoaXMuYXBwbHlTY29wZShxdWVyeSlcblxuICAgIGNvbnN0IGZvcmVpZ25Nb2RlbCA9IGF3YWl0IHF1ZXJ5LmZpcnN0KClcblxuICAgIGlmIChmb3JlaWduTW9kZWwpIHtcbiAgICAgIHRoaXMuc2V0TG9hZGVkKGZvcmVpZ25Nb2RlbClcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5zZXRMb2FkZWQodW5kZWZpbmVkKVxuICAgIH1cbiAgICB0aGlzLnNldERpcnR5KGZhbHNlKVxuICAgIHRoaXMuc2V0UHJlbG9hZGVkKHRydWUpXG5cbiAgICByZXR1cm4gdGhpcy5sb2FkZWQoKVxuICB9XG5cbiAgLyoqXG4gICAqIEByZXR1cm5zIHtJbnN0YW5jZVR5cGU8VE1DPiB8IEFycmF5PEluc3RhbmNlVHlwZTxUTUM+PiB8IHVuZGVmaW5lZH0gVGhlIGxvYWRlZCBtb2RlbCBvciBtb2RlbHMgKGRlcGVuZGluZyBvbiByZWxhdGlvbnNoaXAgdHlwZSlcbiAgICovXG4gIGxvYWRlZCgpIHtcbiAgICBpZiAoIXRoaXMuX3ByZWxvYWRlZCAmJiB0aGlzLm1vZGVsLmlzUGVyc2lzdGVkKCkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgJHt0aGlzLm1vZGVsLmNvbnN0cnVjdG9yLm5hbWV9IyR7dGhpcy5yZWxhdGlvbnNoaXAuZ2V0UmVsYXRpb25zaGlwTmFtZSgpfSBoYXNuJ3QgYmVlbiBwcmVsb2FkZWRgKVxuICAgIH1cblxuICAgIHJldHVybiB0aGlzLl9sb2FkZWRcbiAgfVxuXG4gIGdldExvYWRlZE9yVW5kZWZpbmVkKCkgeyByZXR1cm4gdGhpcy5fbG9hZGVkIH1cblxuICAvKiogQHBhcmFtIHtJbnN0YW5jZVR5cGU8VE1DPiB8IEFycmF5PEluc3RhbmNlVHlwZTxUTUM+PiB8IHVuZGVmaW5lZH0gbW9kZWwgLSBSZWxhdGVkIG1vZGVsKHMpLiAqL1xuICBzZXRMb2FkZWQobW9kZWwpIHtcbiAgICBpZiAoQXJyYXkuaXNBcnJheShtb2RlbCkpIHRocm93IG5ldyBFcnJvcihgQXJndW1lbnQgZ2l2ZW4gdG8gc2V0TG9hZGVkIHdhcyBhbiBhcnJheTogJHt0eXBlb2YgbW9kZWx9YClcblxuICAgIHRoaXMuX2xvYWRlZCA9IG1vZGVsXG4gIH1cblxuICBnZXRUYXJnZXRNb2RlbENsYXNzKCkgeyByZXR1cm4gLyoqIEB0eXBlIHtUTUMgfCB1bmRlZmluZWR9ICovICh0aGlzLnJlbGF0aW9uc2hpcC5nZXRUYXJnZXRNb2RlbENsYXNzKCkpIH1cbn1cbiJdfQ==
|
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
*/
|
|
4
4
|
/**
|
|
5
5
|
* @typedef {object} RelationshipBaseArgsType
|
|
6
|
+
* @property {boolean} [autoload] - Whether to auto-batch-preload siblings when this relationship is lazy-loaded. Default true.
|
|
6
7
|
* @property {string} [className] - Name of the related model class.
|
|
7
8
|
* @property {boolean} [counterCache] - Auto-sync parent count column on create/update/destroy.
|
|
8
9
|
* @property {string} [dependent] - Dependent action when parent is destroyed.
|
|
@@ -19,7 +20,8 @@
|
|
|
19
20
|
*/
|
|
20
21
|
export default class VelociousDatabaseRecordBaseRelationship {
|
|
21
22
|
/** @param {RelationshipBaseArgsType} args - Relationship definition arguments. */
|
|
22
|
-
constructor({ className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey, polymorphic, relationshipName, scope, through, type, ...restArgs }: RelationshipBaseArgsType);
|
|
23
|
+
constructor({ autoload, className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey, polymorphic, relationshipName, scope, through, type, ...restArgs }: RelationshipBaseArgsType);
|
|
24
|
+
_autoload: boolean;
|
|
23
25
|
className: string | undefined;
|
|
24
26
|
_counterCache: boolean;
|
|
25
27
|
_dependent: string | undefined;
|
|
@@ -33,6 +35,8 @@ export default class VelociousDatabaseRecordBaseRelationship {
|
|
|
33
35
|
_scope: RelationshipScopeCallback | undefined;
|
|
34
36
|
through: string | undefined;
|
|
35
37
|
type: string;
|
|
38
|
+
/** @returns {boolean} Whether this relationship auto-batch-preloads siblings on lazy access. */
|
|
39
|
+
getAutoload(): boolean;
|
|
36
40
|
getConfiguration(): import("../../../configuration.js").default;
|
|
37
41
|
/** @returns {boolean} Whether a counter cache column is synced on the parent. */
|
|
38
42
|
getCounterCache(): boolean;
|
|
@@ -74,6 +78,10 @@ export default class VelociousDatabaseRecordBaseRelationship {
|
|
|
74
78
|
}
|
|
75
79
|
export type RelationshipScopeCallback = (query: import("../../query/model-class-query.js").default<any>) => (import("../../query/model-class-query.js").default<any> | void);
|
|
76
80
|
export type RelationshipBaseArgsType = {
|
|
81
|
+
/**
|
|
82
|
+
* - Whether to auto-batch-preload siblings when this relationship is lazy-loaded. Default true.
|
|
83
|
+
*/
|
|
84
|
+
autoload?: boolean | undefined;
|
|
77
85
|
/**
|
|
78
86
|
* - Name of the related model class.
|
|
79
87
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/relationships/base.js"],"names":[],"mappings":"AAKA;;GAEG;AACH
|
|
1
|
+
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../../../src/database/record/relationships/base.js"],"names":[],"mappings":"AAKA;;GAEG;AACH;;;;;;;;;;;;;;;;GAgBG;AAEH;IACE,kFAAkF;IAClF,sLADY,wBAAwB,EAyBnC;IAdC,mBAAmC;IACnC,8BAA0B;IAC1B,uBAA0C;IAC1C,+BAA2B;IAC3B,+BAA4B;IAC5B,+BAA2B;IAC3B,wDAAkB;IAClB,iDAA4B;IAC5B,kCAA+B;IAC/B,oBAA6B;IAC7B,yBAAwC;IACxC,8CAAmB;IACnB,4BAAsB;IACtB,aAAgB;IAGlB,gGAAgG;IAChG,eADc,OAAO,CACkB;IAEvC,gEAAiE;IAEjE,iFAAiF;IACjF,mBADc,OAAO,CAC0B;IAE/C,sIAAsI;IACtI,gBADc,MAAM,GAAG,SAAS,CACS;IAEzC;;;OAGG;IACH,iBAFa,MAAM,CAIlB;IAED;;;OAGG;IACH,gBAFa,MAAM,GAAG,SAAS,CAI9B;IAED,0EAA0E;IAC1E,iBADc,cAAc,aAAa,EAAE,OAAO,CACR;IAE1C,4FAA4F;IAC5F,uBADc,MAAM,CACkC;IAEtD,6EAA6E;IAC7E,YADc,yBAAyB,GAAG,SAAS,CAClB;IAEjC;;;;OAIG;IACH,WAJa,CAAC,SACH,CAAC,GACC,CAAC,CAUb;IAED,iDAAiD;IACjD,kBADc,OAAO,CAGpB;IAED,wDAAwD;IACxD,4BADc,MAAM,CAmBnB;IATK,2CAAiE;IAWvE,oEAAoE;IACpE,iBADc,MAAM,CACuB;IAE3C,+HAA+H;IAC/H,WADc,MAAM,CACU;IAE9B,mMAAmM;IACnM,uBADc,cAAc,aAAa,EAAE,OAAO,GAAG,SAAS,CAW7D;CACF;wCA/IY,CAAC,KAAK,EAAE,OAAO,kCAAkC,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,kCAAkC,EAAE,OAAO,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC;;;;;;;;;;;;;;;;;;;;;iBAQnI,MAAM,GAAG,SAAS;;;;;;;;;;;;gBAGlB,cAAc,aAAa,EAAE,OAAO;;;;;;;;;;;;sBAGpC,MAAM;;;;;;;;;;;;UAGN,MAAM"}
|
|
@@ -6,6 +6,7 @@ import * as inflection from "inflection";
|
|
|
6
6
|
*/
|
|
7
7
|
/**
|
|
8
8
|
* @typedef {object} RelationshipBaseArgsType
|
|
9
|
+
* @property {boolean} [autoload] - Whether to auto-batch-preload siblings when this relationship is lazy-loaded. Default true.
|
|
9
10
|
* @property {string} [className] - Name of the related model class.
|
|
10
11
|
* @property {boolean} [counterCache] - Auto-sync parent count column on create/update/destroy.
|
|
11
12
|
* @property {string} [dependent] - Dependent action when parent is destroyed.
|
|
@@ -22,7 +23,7 @@ import * as inflection from "inflection";
|
|
|
22
23
|
*/
|
|
23
24
|
export default class VelociousDatabaseRecordBaseRelationship {
|
|
24
25
|
/** @param {RelationshipBaseArgsType} args - Relationship definition arguments. */
|
|
25
|
-
constructor({ className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey = "id", polymorphic, relationshipName, scope, through, type, ...restArgs }) {
|
|
26
|
+
constructor({ autoload, className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey = "id", polymorphic, relationshipName, scope, through, type, ...restArgs }) {
|
|
26
27
|
restArgsError(restArgs);
|
|
27
28
|
if (!modelClass)
|
|
28
29
|
throw new Error(`'modelClass' wasn't given for ${relationshipName}`);
|
|
@@ -31,6 +32,7 @@ export default class VelociousDatabaseRecordBaseRelationship {
|
|
|
31
32
|
if (className == "EventSery") {
|
|
32
33
|
throw new Error(`Invalid model name: ${className}`);
|
|
33
34
|
}
|
|
35
|
+
this._autoload = autoload !== false;
|
|
34
36
|
this.className = className;
|
|
35
37
|
this._counterCache = counterCache || false;
|
|
36
38
|
this._dependent = dependent;
|
|
@@ -45,6 +47,8 @@ export default class VelociousDatabaseRecordBaseRelationship {
|
|
|
45
47
|
this.through = through;
|
|
46
48
|
this.type = type;
|
|
47
49
|
}
|
|
50
|
+
/** @returns {boolean} Whether this relationship auto-batch-preloads siblings on lazy access. */
|
|
51
|
+
getAutoload() { return this._autoload; }
|
|
48
52
|
getConfiguration() { return this.modelClass._getConfiguration(); }
|
|
49
53
|
/** @returns {boolean} Whether a counter cache column is synced on the parent. */
|
|
50
54
|
getCounterCache() { return this._counterCache; }
|
|
@@ -121,4 +125,4 @@ export default class VelociousDatabaseRecordBaseRelationship {
|
|
|
121
125
|
throw new Error("Couldn't figure out the target model class");
|
|
122
126
|
}
|
|
123
127
|
}
|
|
124
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../../../src/database/record/relationships/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport restArgsError from \"../../../utils/rest-args-error.js\"\nimport * as inflection from \"inflection\"\n\n/**\n * @typedef {(query: import(\"../../query/model-class-query.js\").default<any>) => (import(\"../../query/model-class-query.js\").default<any> | void)} RelationshipScopeCallback\n */\n/**\n * @typedef {object} RelationshipBaseArgsType\n * @property {string} [className] - Name of the related model class.\n * @property {boolean} [counterCache] - Auto-sync parent count column on create/update/destroy.\n * @property {string} [dependent] - Dependent action when parent is destroyed.\n * @property {string | undefined} [foreignKey] - Explicit foreign key column name.\n * @property {string} [inverseOf] - Inverse relationship name on the related model.\n * @property {typeof import(\"../index.js\").default} [klass] - Related model class.\n * @property {typeof import(\"../index.js\").default} modelClass - Owning model class.\n * @property {string} [primaryKey] - Primary key column on the owning model.\n * @property {boolean} [polymorphic] - Whether the relationship is polymorphic.\n * @property {string} relationshipName - Name of the relationship on the model.\n * @property {RelationshipScopeCallback} [scope] - Optional scope callback for the relationship.\n * @property {string} [through] - Name of the through association.\n * @property {string} type - Relationship type (e.g. \"hasMany\").\n */\n\nexport default class VelociousDatabaseRecordBaseRelationship {\n  /** @param {RelationshipBaseArgsType} args - Relationship definition arguments. */\n  constructor({className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey = \"id\", polymorphic, relationshipName, scope, through, type, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!modelClass) throw new Error(`'modelClass' wasn't given for ${relationshipName}`)\n    if (!className && !klass && !polymorphic) throw new Error(`Neither 'className' or 'klass' was given for ${modelClass.name}#${relationshipName}`)\n\n    if (className == \"EventSery\") {\n      throw new Error(`Invalid model name: ${className}`)\n    }\n\n    this.className = className\n    this._counterCache = counterCache || false\n    this._dependent = dependent\n    this.foreignKey = foreignKey\n    this._inverseOf = inverseOf\n    this.klass = klass\n    this.modelClass = modelClass\n    this._polymorphic = polymorphic\n    this._primaryKey = primaryKey\n    this.relationshipName = relationshipName\n    this._scope = scope\n    this.through = through\n    this.type = type\n  }\n\n  getConfiguration() { return this.modelClass._getConfiguration() }\n\n  /** @returns {boolean} Whether a counter cache column is synced on the parent. */\n  getCounterCache() { return this._counterCache }\n\n  /** @returns {string | undefined} What will be done when the parent record is destroyed. E.g. \"destroy\", \"nullify\", \"restrict\" etc. */\n  getDependent() { return this._dependent }\n\n  /**\n   * @abstract\n   * @returns {string} The name of the foreign key, e.g. \"user_id\", \"post_id\" etc.\n   */\n  getForeignKey() {\n    throw new Error(\"getForeignKey not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @returns {string | undefined} The name of the inverse relationship, e.g. \"posts\", \"comments\" etc.\n   */\n  getInverseOf() {\n    throw new Error(\"getInverseOf not implemented\")\n  }\n\n  /** @returns {typeof import(\"../index.js\").default} - The model class.  */\n  getModelClass() { return this.modelClass }\n\n  /** @returns {string} The name of the relationship, e.g. \"posts\", \"user\", \"comments\" etc. */\n  getRelationshipName() { return this.relationshipName }\n\n  /** @returns {RelationshipScopeCallback | undefined} - The scope callback. */\n  getScope() { return this._scope }\n\n  /**\n   * @template T\n   * @param {T} query - Query instance.\n   * @returns {T} - Scoped query.\n   */\n  applyScope(query) {\n    const scope = this.getScope()\n\n    if (!scope) return query\n\n    const scopedQuery = /** @type {T | void} */ (scope.call(query, /** @type {import(\"../../query/model-class-query.js\").default<any>} */ (query)))\n\n    return scopedQuery || query\n  }\n\n  /** @returns {boolean} - Whether polymorphic.  */\n  getPolymorphic() {\n    return this._polymorphic || false\n  }\n\n  /** @returns {string} - The polymorphic type column.  */\n  getPolymorphicTypeColumn() {\n    if (!this.getPolymorphic()) {\n      throw new Error(`${this.modelClass.name}#${this.relationshipName} isn't polymorphic`)\n    }\n\n    if (!this._polymorphicTypeColumn) {\n      const foreignKey = this.getForeignKey()\n\n      if (foreignKey && foreignKey.endsWith(\"_id\")) {\n        this._polymorphicTypeColumn = foreignKey.replace(/_id$/, \"_type\")\n      } else {\n        const underscoredName = inflection.underscore(this.getRelationshipName())\n\n        this._polymorphicTypeColumn = `${underscoredName}_type`\n      }\n    }\n\n    return this._polymorphicTypeColumn\n  }\n\n  /** @returns {string} The name of the foreign key, e.g. \"id\" etc. */\n  getPrimaryKey() { return this._primaryKey }\n\n  /** @returns {string} The type of the relationship, e.g. \"has_many\", \"belongs_to\", \"has_one\", \"has_and_belongs_to_many\" etc. */\n  getType() { return this.type }\n\n  /** @returns {typeof import(\"../index.js\").default | undefined} The target model class for this relationship, e.g. if the relationship is \"posts\" then the target model class is the Post class. */\n  getTargetModelClass() {\n    if (this.getPolymorphic() && this.type == \"belongsTo\") {\n      return undefined\n    } else if (this.className) {\n      return this.getConfiguration().getModelClass(this.className)\n    } else if (this.klass) {\n      return this.klass\n    }\n\n    throw new Error(\"Couldn't figure out the target model class\")\n  }\n}\n"]}
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../../../src/database/record/relationships/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport restArgsError from \"../../../utils/rest-args-error.js\"\nimport * as inflection from \"inflection\"\n\n/**\n * @typedef {(query: import(\"../../query/model-class-query.js\").default<any>) => (import(\"../../query/model-class-query.js\").default<any> | void)} RelationshipScopeCallback\n */\n/**\n * @typedef {object} RelationshipBaseArgsType\n * @property {boolean} [autoload] - Whether to auto-batch-preload siblings when this relationship is lazy-loaded. Default true.\n * @property {string} [className] - Name of the related model class.\n * @property {boolean} [counterCache] - Auto-sync parent count column on create/update/destroy.\n * @property {string} [dependent] - Dependent action when parent is destroyed.\n * @property {string | undefined} [foreignKey] - Explicit foreign key column name.\n * @property {string} [inverseOf] - Inverse relationship name on the related model.\n * @property {typeof import(\"../index.js\").default} [klass] - Related model class.\n * @property {typeof import(\"../index.js\").default} modelClass - Owning model class.\n * @property {string} [primaryKey] - Primary key column on the owning model.\n * @property {boolean} [polymorphic] - Whether the relationship is polymorphic.\n * @property {string} relationshipName - Name of the relationship on the model.\n * @property {RelationshipScopeCallback} [scope] - Optional scope callback for the relationship.\n * @property {string} [through] - Name of the through association.\n * @property {string} type - Relationship type (e.g. \"hasMany\").\n */\n\nexport default class VelociousDatabaseRecordBaseRelationship {\n  /** @param {RelationshipBaseArgsType} args - Relationship definition arguments. */\n  constructor({autoload, className, counterCache, dependent, foreignKey, inverseOf, klass, modelClass, primaryKey = \"id\", polymorphic, relationshipName, scope, through, type, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!modelClass) throw new Error(`'modelClass' wasn't given for ${relationshipName}`)\n    if (!className && !klass && !polymorphic) throw new Error(`Neither 'className' or 'klass' was given for ${modelClass.name}#${relationshipName}`)\n\n    if (className == \"EventSery\") {\n      throw new Error(`Invalid model name: ${className}`)\n    }\n\n    this._autoload = autoload !== false\n    this.className = className\n    this._counterCache = counterCache || false\n    this._dependent = dependent\n    this.foreignKey = foreignKey\n    this._inverseOf = inverseOf\n    this.klass = klass\n    this.modelClass = modelClass\n    this._polymorphic = polymorphic\n    this._primaryKey = primaryKey\n    this.relationshipName = relationshipName\n    this._scope = scope\n    this.through = through\n    this.type = type\n  }\n\n  /** @returns {boolean} Whether this relationship auto-batch-preloads siblings on lazy access. */\n  getAutoload() { return this._autoload }\n\n  getConfiguration() { return this.modelClass._getConfiguration() }\n\n  /** @returns {boolean} Whether a counter cache column is synced on the parent. */\n  getCounterCache() { return this._counterCache }\n\n  /** @returns {string | undefined} What will be done when the parent record is destroyed. E.g. \"destroy\", \"nullify\", \"restrict\" etc. */\n  getDependent() { return this._dependent }\n\n  /**\n   * @abstract\n   * @returns {string} The name of the foreign key, e.g. \"user_id\", \"post_id\" etc.\n   */\n  getForeignKey() {\n    throw new Error(\"getForeignKey not implemented\")\n  }\n\n  /**\n   * @abstract\n   * @returns {string | undefined} The name of the inverse relationship, e.g. \"posts\", \"comments\" etc.\n   */\n  getInverseOf() {\n    throw new Error(\"getInverseOf not implemented\")\n  }\n\n  /** @returns {typeof import(\"../index.js\").default} - The model class.  */\n  getModelClass() { return this.modelClass }\n\n  /** @returns {string} The name of the relationship, e.g. \"posts\", \"user\", \"comments\" etc. */\n  getRelationshipName() { return this.relationshipName }\n\n  /** @returns {RelationshipScopeCallback | undefined} - The scope callback. */\n  getScope() { return this._scope }\n\n  /**\n   * @template T\n   * @param {T} query - Query instance.\n   * @returns {T} - Scoped query.\n   */\n  applyScope(query) {\n    const scope = this.getScope()\n\n    if (!scope) return query\n\n    const scopedQuery = /** @type {T | void} */ (scope.call(query, /** @type {import(\"../../query/model-class-query.js\").default<any>} */ (query)))\n\n    return scopedQuery || query\n  }\n\n  /** @returns {boolean} - Whether polymorphic.  */\n  getPolymorphic() {\n    return this._polymorphic || false\n  }\n\n  /** @returns {string} - The polymorphic type column.  */\n  getPolymorphicTypeColumn() {\n    if (!this.getPolymorphic()) {\n      throw new Error(`${this.modelClass.name}#${this.relationshipName} isn't polymorphic`)\n    }\n\n    if (!this._polymorphicTypeColumn) {\n      const foreignKey = this.getForeignKey()\n\n      if (foreignKey && foreignKey.endsWith(\"_id\")) {\n        this._polymorphicTypeColumn = foreignKey.replace(/_id$/, \"_type\")\n      } else {\n        const underscoredName = inflection.underscore(this.getRelationshipName())\n\n        this._polymorphicTypeColumn = `${underscoredName}_type`\n      }\n    }\n\n    return this._polymorphicTypeColumn\n  }\n\n  /** @returns {string} The name of the foreign key, e.g. \"id\" etc. */\n  getPrimaryKey() { return this._primaryKey }\n\n  /** @returns {string} The type of the relationship, e.g. \"has_many\", \"belongs_to\", \"has_one\", \"has_and_belongs_to_many\" etc. */\n  getType() { return this.type }\n\n  /** @returns {typeof import(\"../index.js\").default | undefined} The target model class for this relationship, e.g. if the relationship is \"posts\" then the target model class is the Post class. */\n  getTargetModelClass() {\n    if (this.getPolymorphic() && this.type == \"belongsTo\") {\n      return undefined\n    } else if (this.className) {\n      return this.getConfiguration().getModelClass(this.className)\n    } else if (this.klass) {\n      return this.klass\n    }\n\n    throw new Error(\"Couldn't figure out the target model class\")\n  }\n}\n"]}
|