velocious 1.0.217 → 1.0.219

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 CHANGED
@@ -331,6 +331,8 @@ Frontend-model command transport preserves `Date` and `undefined` by encoding th
331
331
  - `bigint` -> `{__velocious_type: "bigint", value: "<decimal string>"}`
332
332
  - `NaN` / `Infinity` / `-Infinity` -> `{__velocious_type: "number", value: "NaN" | "Infinity" | "-Infinity"}`
333
333
 
334
+ Frontend-model commands raise an `Error` when the backend responds with `{status: "error"}` (using `errorMessage` when present), so unauthorized or missing-record update/find/destroy responses fail fast in frontend code.
335
+
334
336
  ## Route resolver hooks
335
337
 
336
338
  Libraries can hook unresolved routes and hijack them before Velocious falls back to the built-in 404 controller.
@@ -1,5 +1,10 @@
1
1
  export default class VelociousCliCommandsServer extends BaseCommand {
2
2
  output: string;
3
+ /**
4
+ * @param {string} actionName - Raw route action name.
5
+ * @returns {string} - Normalized method name.
6
+ */
7
+ normalizeActionName(actionName: string): string;
3
8
  execute(): Promise<{
4
9
  output: string;
5
10
  }>;
@@ -1 +1 @@
1
- {"version":3,"file":"routes.d.ts","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/routes.js"],"names":[],"mappings":"AAKA;IACE,eAAW;IAEX;;OAWC;IAED;;;;OAIG;IACH,mBAJW,OAAO,kCAAkC,EAAE,OAAO,UAClD,MAAM,GACJ,IAAI,CAYhB;IAED;;;OAGG;IACH,aAHW,MAAM,GACJ,IAAI,CAQhB;CACF;wBA9CuB,iCAAiC"}
1
+ {"version":3,"file":"routes.d.ts","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/routes.js"],"names":[],"mappings":"AAKA;IACE,eAAW;IAEX;;;OAGG;IACH,gCAHW,MAAM,GACJ,MAAM,CAIlB;IAED;;OAWC;IAED;;;;OAIG;IACH,mBAJW,OAAO,kCAAkC,EAAE,OAAO,UAClD,MAAM,GACJ,IAAI,CAYhB;IAED;;;OAGG;IACH,aAHW,MAAM,GACJ,IAAI,CAQhB;CACF;wBAtDuB,iCAAiC"}
@@ -3,6 +3,13 @@ import Application from "../../../../application.js";
3
3
  import BaseCommand from "../../../../cli/base-command.js";
4
4
  export default class VelociousCliCommandsServer extends BaseCommand {
5
5
  output = "";
6
+ /**
7
+ * @param {string} actionName - Raw route action name.
8
+ * @returns {string} - Normalized method name.
9
+ */
10
+ normalizeActionName(actionName) {
11
+ return inflection.camelize(actionName.replaceAll("-", "_").replaceAll("/", "_"), true);
12
+ }
6
13
  async execute() {
7
14
  const application = new Application({
8
15
  configuration: this.getConfiguration(),
@@ -21,7 +28,7 @@ export default class VelociousCliCommandsServer extends BaseCommand {
21
28
  printRoutes(route, level = 0) {
22
29
  const prefix = " ".repeat(level);
23
30
  for (const routeData of route.getHumanPaths()) {
24
- this.log(`${prefix}${routeData.method} ${routeData.path}${routeData.action ? ` -> ${inflection.camelize(routeData.action.replaceAll("-", "_"), true)}` : ""}`);
31
+ this.log(`${prefix}${routeData.method} ${routeData.path}${routeData.action ? ` -> ${this.normalizeActionName(routeData.action)}` : ""}`);
25
32
  }
26
33
  for (const subRoute of route.getSubRoutes()) {
27
34
  this.printRoutes(subRoute, level + 1);
@@ -40,4 +47,4 @@ export default class VelociousCliCommandsServer extends BaseCommand {
40
47
  }
41
48
  }
42
49
  }
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2Vudmlyb25tZW50LWhhbmRsZXJzL25vZGUvY2xpL2NvbW1hbmRzL3JvdXRlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssVUFBVSxNQUFNLFlBQVksQ0FBQTtBQUV4QyxPQUFPLFdBQVcsTUFBTSw0QkFBNEIsQ0FBQTtBQUNwRCxPQUFPLFdBQVcsTUFBTSxpQ0FBaUMsQ0FBQTtBQUV6RCxNQUFNLENBQUMsT0FBTyxPQUFPLDBCQUEyQixTQUFRLFdBQVc7SUFDakUsTUFBTSxHQUFHLEVBQUUsQ0FBQTtJQUVYLEtBQUssQ0FBQyxPQUFPO1FBQ1gsTUFBTSxXQUFXLEdBQUcsSUFBSSxXQUFXLENBQUM7WUFDbEMsYUFBYSxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRTtZQUN0QyxJQUFJLEVBQUUsUUFBUTtTQUNmLENBQUMsQ0FBQTtRQUNGLE1BQU0sV0FBVyxDQUFDLFVBQVUsRUFBRSxDQUFBO1FBQzlCLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDLFNBQVMsRUFBRSxDQUFBO1FBRWxELElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFBO1FBRWxDLE9BQU8sRUFBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBQyxDQUFBO0lBQzlCLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsV0FBVyxDQUFDLEtBQUssRUFBRSxLQUFLLEdBQUcsQ0FBQztRQUMxQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBRWpDLEtBQUssTUFBTSxTQUFTLElBQUksS0FBSyxDQUFDLGFBQWEsRUFBRSxFQUFFLENBQUM7WUFDOUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLE1BQU0sR0FBRyxTQUFTLENBQUMsTUFBTSxJQUFJLFNBQVMsQ0FBQyxJQUFJLEdBQUcsU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsT0FBTyxVQUFVLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBQ2hLLENBQUM7UUFFRCxLQUFLLE1BQU0sUUFBUSxJQUFJLEtBQUssQ0FBQyxZQUFZLEVBQUUsRUFBRSxDQUFDO1lBQzVDLElBQUksQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFLEtBQUssR0FBRyxDQUFDLENBQUMsQ0FBQTtRQUN2QyxDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7T0FHRztJQUNILEdBQUcsQ0FBQyxPQUFPO1FBQ1QsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsRUFBRSxFQUFFLENBQUM7WUFDMUIsSUFBSSxDQUFDLE1BQU0sSUFBSSxHQUFHLE9BQU8sSUFBSSxDQUFBO1FBQy9CLENBQUM7YUFBTSxDQUFDO1lBQ04sT0FBTyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQTtRQUN0QixDQUFDO0lBQ0gsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgaW5mbGVjdGlvbiBmcm9tIFwiaW5mbGVjdGlvblwiXG5cbmltcG9ydCBBcHBsaWNhdGlvbiBmcm9tIFwiLi4vLi4vLi4vLi4vYXBwbGljYXRpb24uanNcIlxuaW1wb3J0IEJhc2VDb21tYW5kIGZyb20gXCIuLi8uLi8uLi8uLi9jbGkvYmFzZS1jb21tYW5kLmpzXCJcblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgVmVsb2Npb3VzQ2xpQ29tbWFuZHNTZXJ2ZXIgZXh0ZW5kcyBCYXNlQ29tbWFuZHtcbiAgb3V0cHV0ID0gXCJcIlxuXG4gIGFzeW5jIGV4ZWN1dGUoKSB7XG4gICAgY29uc3QgYXBwbGljYXRpb24gPSBuZXcgQXBwbGljYXRpb24oe1xuICAgICAgY29uZmlndXJhdGlvbjogdGhpcy5nZXRDb25maWd1cmF0aW9uKCksXG4gICAgICB0eXBlOiBcInNlcnZlclwiXG4gICAgfSlcbiAgICBhd2FpdCBhcHBsaWNhdGlvbi5pbml0aWFsaXplKClcbiAgICBjb25zdCByb3V0ZXMgPSB0aGlzLmdldENvbmZpZ3VyYXRpb24oKS5nZXRSb3V0ZXMoKVxuXG4gICAgdGhpcy5wcmludFJvdXRlcyhyb3V0ZXMucm9vdFJvdXRlKVxuXG4gICAgcmV0dXJuIHtvdXRwdXQ6IHRoaXMub3V0cHV0fVxuICB9XG5cbiAgLyoqXG4gICAqIEBwYXJhbSB7aW1wb3J0KFwiLi4vLi4vLi4vLi4vcm91dGVzL2Jhc2Utcm91dGUuanNcIikuZGVmYXVsdH0gcm91dGUgLSBSb3V0ZS5cbiAgICogQHBhcmFtIHtudW1iZXJ9IFtsZXZlbF0gLSBMZXZlbC5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgcHJpbnRSb3V0ZXMocm91dGUsIGxldmVsID0gMCkge1xuICAgIGNvbnN0IHByZWZpeCA9IFwiICBcIi5yZXBlYXQobGV2ZWwpXG5cbiAgICBmb3IgKGNvbnN0IHJvdXRlRGF0YSBvZiByb3V0ZS5nZXRIdW1hblBhdGhzKCkpIHtcbiAgICAgIHRoaXMubG9nKGAke3ByZWZpeH0ke3JvdXRlRGF0YS5tZXRob2R9ICR7cm91dGVEYXRhLnBhdGh9JHtyb3V0ZURhdGEuYWN0aW9uID8gYCAtPiAke2luZmxlY3Rpb24uY2FtZWxpemUocm91dGVEYXRhLmFjdGlvbi5yZXBsYWNlQWxsKFwiLVwiLCBcIl9cIiksIHRydWUpfWAgOiBcIlwifWApXG4gICAgfVxuXG4gICAgZm9yIChjb25zdCBzdWJSb3V0ZSBvZiByb3V0ZS5nZXRTdWJSb3V0ZXMoKSkge1xuICAgICAgdGhpcy5wcmludFJvdXRlcyhzdWJSb3V0ZSwgbGV2ZWwgKyAxKVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gY29udGVudCAtIENvbnRlbnQuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIGxvZyhjb250ZW50KSB7XG4gICAgaWYgKHRoaXMuY2xpLmdldFRlc3RpbmcoKSkge1xuICAgICAgdGhpcy5vdXRwdXQgKz0gYCR7Y29udGVudH1cXG5gXG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnNvbGUubG9nKGNvbnRlbnQpXG4gICAgfVxuICB9XG59XG4iXX0=
50
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2Vudmlyb25tZW50LWhhbmRsZXJzL25vZGUvY2xpL2NvbW1hbmRzL3JvdXRlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssVUFBVSxNQUFNLFlBQVksQ0FBQTtBQUV4QyxPQUFPLFdBQVcsTUFBTSw0QkFBNEIsQ0FBQTtBQUNwRCxPQUFPLFdBQVcsTUFBTSxpQ0FBaUMsQ0FBQTtBQUV6RCxNQUFNLENBQUMsT0FBTyxPQUFPLDBCQUEyQixTQUFRLFdBQVc7SUFDakUsTUFBTSxHQUFHLEVBQUUsQ0FBQTtJQUVYOzs7T0FHRztJQUNILG1CQUFtQixDQUFDLFVBQVU7UUFDNUIsT0FBTyxVQUFVLENBQUMsUUFBUSxDQUFDLFVBQVUsQ0FBQyxVQUFVLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsSUFBSSxDQUFDLENBQUE7SUFDeEYsQ0FBQztJQUVELEtBQUssQ0FBQyxPQUFPO1FBQ1gsTUFBTSxXQUFXLEdBQUcsSUFBSSxXQUFXLENBQUM7WUFDbEMsYUFBYSxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRTtZQUN0QyxJQUFJLEVBQUUsUUFBUTtTQUNmLENBQUMsQ0FBQTtRQUNGLE1BQU0sV0FBVyxDQUFDLFVBQVUsRUFBRSxDQUFBO1FBQzlCLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDLFNBQVMsRUFBRSxDQUFBO1FBRWxELElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFBO1FBRWxDLE9BQU8sRUFBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBQyxDQUFBO0lBQzlCLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsV0FBVyxDQUFDLEtBQUssRUFBRSxLQUFLLEdBQUcsQ0FBQztRQUMxQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBRWpDLEtBQUssTUFBTSxTQUFTLElBQUksS0FBSyxDQUFDLGFBQWEsRUFBRSxFQUFFLENBQUM7WUFDOUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLE1BQU0sR0FBRyxTQUFTLENBQUMsTUFBTSxJQUFJLFNBQVMsQ0FBQyxJQUFJLEdBQUcsU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsT0FBTyxJQUFJLENBQUMsbUJBQW1CLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFDMUksQ0FBQztRQUVELEtBQUssTUFBTSxRQUFRLElBQUksS0FBSyxDQUFDLFlBQVksRUFBRSxFQUFFLENBQUM7WUFDNUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUUsS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFBO1FBQ3ZDLENBQUM7SUFDSCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsR0FBRyxDQUFDLE9BQU87UUFDVCxJQUFJLElBQUksQ0FBQyxHQUFHLENBQUMsVUFBVSxFQUFFLEVBQUUsQ0FBQztZQUMxQixJQUFJLENBQUMsTUFBTSxJQUFJLEdBQUcsT0FBTyxJQUFJLENBQUE7UUFDL0IsQ0FBQzthQUFNLENBQUM7WUFDTixPQUFPLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFBO1FBQ3RCLENBQUM7SUFDSCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBpbmZsZWN0aW9uIGZyb20gXCJpbmZsZWN0aW9uXCJcblxuaW1wb3J0IEFwcGxpY2F0aW9uIGZyb20gXCIuLi8uLi8uLi8uLi9hcHBsaWNhdGlvbi5qc1wiXG5pbXBvcnQgQmFzZUNvbW1hbmQgZnJvbSBcIi4uLy4uLy4uLy4uL2NsaS9iYXNlLWNvbW1hbmQuanNcIlxuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBWZWxvY2lvdXNDbGlDb21tYW5kc1NlcnZlciBleHRlbmRzIEJhc2VDb21tYW5ke1xuICBvdXRwdXQgPSBcIlwiXG5cbiAgLyoqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhY3Rpb25OYW1lIC0gUmF3IHJvdXRlIGFjdGlvbiBuYW1lLlxuICAgKiBAcmV0dXJucyB7c3RyaW5nfSAtIE5vcm1hbGl6ZWQgbWV0aG9kIG5hbWUuXG4gICAqL1xuICBub3JtYWxpemVBY3Rpb25OYW1lKGFjdGlvbk5hbWUpIHtcbiAgICByZXR1cm4gaW5mbGVjdGlvbi5jYW1lbGl6ZShhY3Rpb25OYW1lLnJlcGxhY2VBbGwoXCItXCIsIFwiX1wiKS5yZXBsYWNlQWxsKFwiL1wiLCBcIl9cIiksIHRydWUpXG4gIH1cblxuICBhc3luYyBleGVjdXRlKCkge1xuICAgIGNvbnN0IGFwcGxpY2F0aW9uID0gbmV3IEFwcGxpY2F0aW9uKHtcbiAgICAgIGNvbmZpZ3VyYXRpb246IHRoaXMuZ2V0Q29uZmlndXJhdGlvbigpLFxuICAgICAgdHlwZTogXCJzZXJ2ZXJcIlxuICAgIH0pXG4gICAgYXdhaXQgYXBwbGljYXRpb24uaW5pdGlhbGl6ZSgpXG4gICAgY29uc3Qgcm91dGVzID0gdGhpcy5nZXRDb25maWd1cmF0aW9uKCkuZ2V0Um91dGVzKClcblxuICAgIHRoaXMucHJpbnRSb3V0ZXMocm91dGVzLnJvb3RSb3V0ZSlcblxuICAgIHJldHVybiB7b3V0cHV0OiB0aGlzLm91dHB1dH1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uLy4uLy4uLy4uL3JvdXRlcy9iYXNlLXJvdXRlLmpzXCIpLmRlZmF1bHR9IHJvdXRlIC0gUm91dGUuXG4gICAqIEBwYXJhbSB7bnVtYmVyfSBbbGV2ZWxdIC0gTGV2ZWwuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIHByaW50Um91dGVzKHJvdXRlLCBsZXZlbCA9IDApIHtcbiAgICBjb25zdCBwcmVmaXggPSBcIiAgXCIucmVwZWF0KGxldmVsKVxuXG4gICAgZm9yIChjb25zdCByb3V0ZURhdGEgb2Ygcm91dGUuZ2V0SHVtYW5QYXRocygpKSB7XG4gICAgICB0aGlzLmxvZyhgJHtwcmVmaXh9JHtyb3V0ZURhdGEubWV0aG9kfSAke3JvdXRlRGF0YS5wYXRofSR7cm91dGVEYXRhLmFjdGlvbiA/IGAgLT4gJHt0aGlzLm5vcm1hbGl6ZUFjdGlvbk5hbWUocm91dGVEYXRhLmFjdGlvbil9YCA6IFwiXCJ9YClcbiAgICB9XG5cbiAgICBmb3IgKGNvbnN0IHN1YlJvdXRlIG9mIHJvdXRlLmdldFN1YlJvdXRlcygpKSB7XG4gICAgICB0aGlzLnByaW50Um91dGVzKHN1YlJvdXRlLCBsZXZlbCArIDEpXG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBjb250ZW50IC0gQ29udGVudC5cbiAgICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICAgKi9cbiAgbG9nKGNvbnRlbnQpIHtcbiAgICBpZiAodGhpcy5jbGkuZ2V0VGVzdGluZygpKSB7XG4gICAgICB0aGlzLm91dHB1dCArPSBgJHtjb250ZW50fVxcbmBcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc29sZS5sb2coY29udGVudClcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -261,6 +261,22 @@ export default class FrontendModelBase {
261
261
  * @returns {Promise<Record<string, any>>} - Parsed JSON response.
262
262
  */
263
263
  static executeCommand(this: typeof FrontendModelBase, commandType: "find" | "index" | "update" | "destroy", payload: Record<string, any>): Promise<Record<string, any>>;
264
+ /**
265
+ * @this {typeof FrontendModelBase}
266
+ * @param {object} args - Arguments.
267
+ * @param {"find" | "index" | "update" | "destroy"} args.commandType - Command type.
268
+ * @param {Record<string, any>} args.response - Decoded response.
269
+ * @returns {void}
270
+ */
271
+ static throwOnErrorFrontendModelResponse(this: typeof FrontendModelBase, { commandType, response }: {
272
+ commandType: "find" | "index" | "update" | "destroy";
273
+ response: Record<string, any>;
274
+ }): void;
275
+ /**
276
+ * @this {typeof FrontendModelBase}
277
+ * @returns {Set<string>} - Configured frontend model attribute names.
278
+ */
279
+ static configuredFrontendModelAttributeNames(this: typeof FrontendModelBase): Set<string>;
264
280
  /**
265
281
  * @param {Record<string, any>} [attributes] - Initial attributes.
266
282
  */
@@ -1 +1 @@
1
- {"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../src/frontend-models/base.js"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../../src/frontend-models/base.js"],"names":[],"mappings":"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"}
@@ -823,7 +823,12 @@ export default class FrontendModelBase {
823
823
  payload: serializedPayload,
824
824
  url
825
825
  });
826
- return /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(customResponse));
826
+ const decodedResponse = /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(customResponse));
827
+ this.throwOnErrorFrontendModelResponse({
828
+ commandType,
829
+ response: decodedResponse
830
+ });
831
+ return decodedResponse;
827
832
  }
828
833
  const response = await fetch(url, {
829
834
  body: JSON.stringify(serializedPayload),
@@ -838,7 +843,55 @@ export default class FrontendModelBase {
838
843
  }
839
844
  const responseText = await response.text();
840
845
  const json = responseText.length > 0 ? JSON.parse(responseText) : {};
841
- return /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(json));
846
+ const decodedResponse = /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(json));
847
+ this.throwOnErrorFrontendModelResponse({
848
+ commandType,
849
+ response: decodedResponse
850
+ });
851
+ return decodedResponse;
852
+ }
853
+ /**
854
+ * @this {typeof FrontendModelBase}
855
+ * @param {object} args - Arguments.
856
+ * @param {"find" | "index" | "update" | "destroy"} args.commandType - Command type.
857
+ * @param {Record<string, any>} args.response - Decoded response.
858
+ * @returns {void}
859
+ */
860
+ static throwOnErrorFrontendModelResponse({ commandType, response }) {
861
+ if (response?.status !== "error")
862
+ return;
863
+ const responseKeys = Object.keys(response);
864
+ const hasOnlyStatus = responseKeys.length === 1 && responseKeys[0] === "status";
865
+ const hasErrorMessage = typeof response.errorMessage === "string" && response.errorMessage.length > 0;
866
+ const hasErrorEnvelopeKeys = Boolean(response.code !== undefined
867
+ || response.error !== undefined
868
+ || response.errors !== undefined
869
+ || response.message !== undefined);
870
+ const nonStatusKeys = responseKeys.filter((key) => key !== "status");
871
+ const configuredAttributeNames = this.configuredFrontendModelAttributeNames();
872
+ const looksLikeRawModelPayload = nonStatusKeys.length > 0
873
+ && nonStatusKeys.every((key) => configuredAttributeNames.has(key));
874
+ if (!hasErrorMessage && !hasOnlyStatus && !hasErrorEnvelopeKeys && looksLikeRawModelPayload)
875
+ return;
876
+ const errorMessage = hasErrorMessage
877
+ ? response.errorMessage
878
+ : `Request failed for ${this.name}#${commandType}`;
879
+ throw new Error(errorMessage);
880
+ }
881
+ /**
882
+ * @this {typeof FrontendModelBase}
883
+ * @returns {Set<string>} - Configured frontend model attribute names.
884
+ */
885
+ static configuredFrontendModelAttributeNames() {
886
+ const resourceConfig = /** @type {Record<string, any>} */ (this.resourceConfig());
887
+ const attributes = resourceConfig.attributes;
888
+ if (Array.isArray(attributes)) {
889
+ return new Set(attributes.filter((attributeName) => typeof attributeName === "string"));
890
+ }
891
+ if (attributes && typeof attributes === "object") {
892
+ return new Set(Object.keys(attributes));
893
+ }
894
+ return new Set();
842
895
  }
843
896
  }
844
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../src/frontend-models/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport FrontendModelQuery from \"./query.js\"\nimport {deserializeFrontendModelTransportValue, serializeFrontendModelTransportValue} from \"./transport-serialization.js\"\n\n/** @typedef {{commands?: Record<string, string>, path?: string, primaryKey?: string}} FrontendModelResourceConfig */\n/**\n * @typedef {object} FrontendModelTransportConfig\n * @property {string} [baseUrl] - Optional base URL prefixed before resource paths.\n * @property {(() => string | undefined | null)} [baseUrlResolver] - Optional resolver used per request for dynamic base URL.\n * @property {string} [pathPrefix] - Optional path prefix inserted between base URL and resource path.\n * @property {(() => string | undefined | null)} [pathPrefixResolver] - Optional resolver used per request for dynamic path prefix.\n * @property {\"omit\" | \"same-origin\" | \"include\"} [credentials] - Optional credentials mode forwarded to fetch.\n * @property {((args: {commandName: string, commandType: \"find\" | \"index\" | \"update\" | \"destroy\", modelClass: typeof FrontendModelBase, payload: Record<string, any>, url: string}) => Promise<Record<string, any>>)} [request] - Optional custom transport handler.\n */\n\n/** @type {FrontendModelTransportConfig} */\nconst frontendModelTransportConfig = {}\nconst PRELOADED_RELATIONSHIPS_KEY = \"__preloadedRelationships\"\nconst SELECTED_ATTRIBUTES_KEY = \"__selectedAttributes\"\n\n/** Error raised when reading an attribute that was not selected in query payloads. */\nexport class AttributeNotSelectedError extends Error {\n  /**\n   * @param {string} modelName - Model class name.\n   * @param {string} attributeName - Attribute that was requested.\n   */\n  constructor(modelName, attributeName) {\n    super(`${modelName}#${attributeName} was not selected`)\n    this.name = \"AttributeNotSelectedError\"\n  }\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Record<string, any>} - Whether value is a plain object.\n */\nfunction isPlainObject(value) {\n  if (!value || typeof value !== \"object\" || Array.isArray(value)) return false\n\n  const prototype = Object.getPrototypeOf(value)\n\n  return prototype === Object.prototype || prototype === null\n}\n\n/**\n * Lightweight singular relationship state holder for frontend model instances.\n * @template {typeof FrontendModelBase} S\n * @template {typeof FrontendModelBase} T\n */\nexport class FrontendModelSingularRelationship {\n  /**\n   * @param {InstanceType<S>} model - Parent model.\n   * @param {string} relationshipName - Relationship name.\n   * @param {T | null} targetModelClass - Target model class.\n   */\n  constructor(model, relationshipName, targetModelClass) {\n    this.model = model\n    this.relationshipName = relationshipName\n    this.targetModelClass = targetModelClass\n    this._preloaded = false\n    this._loadedValue = null\n  }\n\n  /**\n   * @param {any} loadedValue - Loaded relationship value.\n   * @returns {void}\n   */\n  setLoaded(loadedValue) {\n    this._loadedValue = loadedValue == undefined ? null : loadedValue\n    this._preloaded = true\n  }\n\n  /**\n   * @returns {boolean} - Whether relationship is preloaded.\n   */\n  getPreloaded() {\n    return this._preloaded\n  }\n\n  /**\n   * @returns {any} - Loaded relationship value.\n   */\n  loaded() {\n    if (!this._preloaded) {\n      throw new Error(`${this.model.constructor.name}#${this.relationshipName} hasn't been preloaded`)\n    }\n\n    return this._loadedValue\n  }\n\n  /**\n   * @param {Record<string, any>} [attributes] - New model attributes.\n   * @returns {InstanceType<T>} - Built model.\n   */\n  build(attributes = {}) {\n    if (!this.targetModelClass) {\n      throw new Error(`No target model class configured for ${this.model.constructor.name}#${this.relationshipName}`)\n    }\n\n    const model = /** @type {InstanceType<T>} */ (new this.targetModelClass(attributes))\n\n    this.setLoaded(model)\n\n    return model\n  }\n}\n\n/**\n * Lightweight has-many relationship state holder for frontend model instances.\n * @template {typeof FrontendModelBase} S\n * @template {typeof FrontendModelBase} T\n */\nexport class FrontendModelHasManyRelationship {\n  /**\n   * @param {InstanceType<S>} model - Parent model.\n   * @param {string} relationshipName - Relationship name.\n   * @param {T | null} targetModelClass - Target model class.\n   */\n  constructor(model, relationshipName, targetModelClass) {\n    this.model = model\n    this.relationshipName = relationshipName\n    this.targetModelClass = targetModelClass\n    this._preloaded = false\n    this._loadedValue = []\n  }\n\n  /**\n   * @param {Array<InstanceType<T>>} loadedValue - Loaded relationship value.\n   * @returns {void}\n   */\n  setLoaded(loadedValue) {\n    this._loadedValue = Array.isArray(loadedValue) ? loadedValue : []\n    this._preloaded = true\n  }\n\n  /**\n   * @returns {boolean} - Whether relationship is preloaded.\n   */\n  getPreloaded() {\n    return this._preloaded\n  }\n\n  /**\n   * @returns {Array<InstanceType<T>>} - Loaded relationship values.\n   */\n  loaded() {\n    if (!this._preloaded) {\n      throw new Error(`${this.model.constructor.name}#${this.relationshipName} hasn't been preloaded`)\n    }\n\n    return this._loadedValue\n  }\n\n  /**\n   * @param {Array<InstanceType<T>>} models - Models to append.\n   * @returns {void}\n   */\n  addToLoaded(models) {\n    const loadedModels = this.getPreloaded() ? this.loaded() : []\n\n    this.setLoaded([...loadedModels, ...models])\n  }\n\n  /**\n   * @param {Record<string, any>} [attributes] - New model attributes.\n   * @returns {InstanceType<T>} - Built model.\n   */\n  build(attributes = {}) {\n    if (!this.targetModelClass) {\n      throw new Error(`No target model class configured for ${this.model.constructor.name}#${this.relationshipName}`)\n    }\n\n    const model = /** @type {InstanceType<T>} */ (new this.targetModelClass(attributes))\n\n    this.addToLoaded([model])\n\n    return model\n  }\n}\n\n/**\n * @param {string} relationshipType - Relationship type.\n * @returns {boolean} - Whether relationship type is has-many.\n */\nfunction relationshipTypeIsCollection(relationshipType) {\n  return relationshipType == \"hasMany\"\n}\n\n/**\n * @param {string | undefined | null} value - Base URL candidate.\n * @returns {string} - Normalized base URL without trailing slash.\n */\nfunction normalizeBaseUrl(value) {\n  if (typeof value !== \"string\") return \"\"\n\n  const trimmed = value.trim()\n\n  if (!trimmed.length) return \"\"\n\n  return trimmed.replace(/\\/+$/, \"\")\n}\n\n/**\n * @param {string | undefined | null} value - Path prefix candidate.\n * @returns {string} - Normalized path prefix with leading slash and no trailing slash.\n */\nfunction normalizePathPrefix(value) {\n  if (typeof value !== \"string\") return \"\"\n\n  const trimmed = value.trim()\n\n  if (!trimmed.length) return \"\"\n\n  const withLeadingSlash = trimmed.startsWith(\"/\") ? trimmed : `/${trimmed}`\n\n  return withLeadingSlash.replace(/\\/+$/, \"\")\n}\n\n/**\n * @param {string} resourcePath - Resource path (expected absolute path).\n * @param {string} commandName - Command name.\n * @returns {string} - Combined command URL.\n */\nfunction frontendModelCommandUrl(resourcePath, commandName) {\n  const resolvedBaseUrl = frontendModelTransportConfig.baseUrlResolver\n    ? frontendModelTransportConfig.baseUrlResolver()\n    : frontendModelTransportConfig.baseUrl\n  const baseUrl = normalizeBaseUrl(resolvedBaseUrl)\n  const resolvedPathPrefix = frontendModelTransportConfig.pathPrefixResolver\n    ? frontendModelTransportConfig.pathPrefixResolver()\n    : frontendModelTransportConfig.pathPrefix\n  const pathPrefix = normalizePathPrefix(resolvedPathPrefix)\n  const normalizedResourcePath = resourcePath.startsWith(\"/\") ? resourcePath : `/${resourcePath}`\n\n  return `${baseUrl}${pathPrefix}${normalizedResourcePath}/${commandName}`\n}\n\n/**\n * @param {Record<string, any>} conditions - findBy conditions.\n * @returns {Record<string, any>} - JSON-normalized conditions.\n */\nfunction normalizeFindConditions(conditions) {\n  try {\n    return /** @type {Record<string, any>} */ (JSON.parse(JSON.stringify(conditions)))\n  } catch (error) {\n    throw new Error(`findBy conditions could not be serialized: ${error instanceof Error ? error.message : String(error)}`)\n  }\n}\n\n/**\n * @param {unknown} conditions - findBy conditions.\n * @returns {void}\n */\nfunction assertFindByConditionsShape(conditions) {\n  if (!conditions || typeof conditions !== \"object\" || Array.isArray(conditions)) {\n    throw new Error(`findBy expects conditions to be a plain object, got: ${conditions}`)\n  }\n\n  const conditionsPrototype = Object.getPrototypeOf(conditions)\n\n  if (conditionsPrototype !== Object.prototype && conditionsPrototype !== null) {\n    throw new Error(`findBy expects conditions to be a plain object, got: ${conditions}`)\n  }\n\n  const symbolKeys = Object.getOwnPropertySymbols(conditions)\n\n  if (symbolKeys.length > 0) {\n    throw new Error(`findBy does not support symbol condition keys (keys: ${symbolKeys.map((key) => key.toString()).join(\", \")})`)\n  }\n}\n\n/**\n * @param {unknown} value - Condition value to validate.\n * @param {string} keyPath - Key path for error output.\n * @returns {void}\n */\nfunction assertDefinedFindByConditionValue(value, keyPath) {\n  if (value === undefined) {\n    throw new Error(`findBy does not support undefined condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"function\") {\n    throw new Error(`findBy does not support function condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"symbol\") {\n    throw new Error(`findBy does not support symbol condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"bigint\") {\n    throw new Error(`findBy does not support bigint condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"number\" && !Number.isFinite(value)) {\n    throw new Error(`findBy does not support non-finite number condition values (key: ${keyPath})`)\n  }\n\n  if (Array.isArray(value)) {\n    value.forEach((entry, index) => {\n      assertDefinedFindByConditionValue(entry, `${keyPath}[${index}]`)\n    })\n    return\n  }\n\n  if (value && typeof value === \"object\") {\n    if (value instanceof Date) {\n      return\n    }\n\n    const objectValue = /** @type {Record<string, unknown>} */ (value)\n    const prototype = Object.getPrototypeOf(objectValue)\n\n    if (prototype !== Object.prototype && prototype !== null) {\n      throw new Error(`findBy does not support non-plain object condition values (key: ${keyPath})`)\n    }\n\n    const symbolKeys = Object.getOwnPropertySymbols(objectValue)\n\n    if (symbolKeys.length > 0) {\n      throw new Error(`findBy does not support symbol condition keys (key: ${keyPath})`)\n    }\n\n    const valueObject = /** @type {Record<string, unknown>} */ (value)\n\n    Object.keys(valueObject).forEach((nestedKey) => {\n      assertDefinedFindByConditionValue(valueObject[nestedKey], `${keyPath}.${nestedKey}`)\n    })\n  }\n}\n\n/**\n * @param {unknown} originalValue - Original condition value.\n * @param {unknown} normalizedValue - JSON-normalized condition value.\n * @param {string} keyPath - Key path for error output.\n * @returns {void}\n */\nfunction assertFindByConditionSerializationPreservesValue(originalValue, normalizedValue, keyPath) {\n  if (originalValue === null) {\n    if (normalizedValue !== null) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    return\n  }\n\n  if (Array.isArray(originalValue)) {\n    if (!Array.isArray(normalizedValue)) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    if (originalValue.length !== normalizedValue.length) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    for (let index = 0; index < originalValue.length; index += 1) {\n      assertFindByConditionSerializationPreservesValue(originalValue[index], normalizedValue[index], `${keyPath}[${index}]`)\n    }\n\n    return\n  }\n\n  if (originalValue instanceof Date) {\n    if (typeof normalizedValue !== \"string\") {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    return\n  }\n\n  if (originalValue && typeof originalValue === \"object\") {\n    if (!normalizedValue || typeof normalizedValue !== \"object\" || Array.isArray(normalizedValue)) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    const normalizedObject = /** @type {Record<string, unknown>} */ (normalizedValue)\n    const originalObject = /** @type {Record<string, unknown>} */ (originalValue)\n\n    Object.keys(originalObject).forEach((nestedKey) => {\n      if (!(nestedKey in normalizedObject)) {\n        throw new Error(`findBy condition key was removed during serialization (key: ${keyPath}.${nestedKey})`)\n      }\n\n      assertFindByConditionSerializationPreservesValue(originalObject[nestedKey], normalizedObject[nestedKey], `${keyPath}.${nestedKey}`)\n    })\n\n    return\n  }\n\n  if (normalizedValue !== originalValue) {\n    throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n  }\n}\n\n/** Base class for generated frontend model classes. */\nexport default class FrontendModelBase {\n  /**\n   * @param {Record<string, any>} [attributes] - Initial attributes.\n   */\n  constructor(attributes = {}) {\n    this._attributes = {}\n    this._relationships = {}\n    this._selectedAttributes = null\n    this.assignAttributes(attributes)\n  }\n\n  /**\n   * @returns {FrontendModelResourceConfig} - Resource configuration.\n   */\n  static resourceConfig() {\n    throw new Error(\"resourceConfig() must be implemented by subclasses\")\n    // eslint-disable-next-line no-unreachable\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {Record<string, typeof FrontendModelBase>} - Relationship model classes keyed by relationship name.\n   */\n  static relationshipModelClasses() {\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {Record<string, {type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationship definitions keyed by relationship name.\n   */\n  static relationshipDefinitions() {\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} relationshipName - Relationship name.\n   * @returns {{type: \"belongsTo\" | \"hasOne\" | \"hasMany\"} | null} - Relationship definition.\n   */\n  static relationshipDefinition(relationshipName) {\n    const definitions = this.relationshipDefinitions()\n\n    return definitions[relationshipName] || null\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} relationshipName - Relationship name.\n   * @returns {typeof FrontendModelBase | null} - Target relationship model class.\n   */\n  static relationshipModelClass(relationshipName) {\n    const relationshipModelClasses = this.relationshipModelClasses()\n\n    return relationshipModelClasses[relationshipName] || null\n  }\n\n  /**\n   * @returns {Record<string, any>} - Attributes hash.\n   */\n  attributes() {\n    return this._attributes\n  }\n\n  /**\n   * @param {string} relationshipName - Relationship name.\n   * @returns {FrontendModelHasManyRelationship<any, any> | FrontendModelSingularRelationship<any, any>} - Relationship state object.\n   */\n  getRelationshipByName(relationshipName) {\n    if (!this._relationships[relationshipName]) {\n      const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n      const relationshipDefinition = ModelClass.relationshipDefinition(relationshipName)\n      const targetModelClass = ModelClass.relationshipModelClass(relationshipName)\n\n      if (relationshipDefinition && relationshipTypeIsCollection(relationshipDefinition.type)) {\n        this._relationships[relationshipName] = new FrontendModelHasManyRelationship(this, relationshipName, targetModelClass)\n      } else {\n        this._relationships[relationshipName] = new FrontendModelSingularRelationship(this, relationshipName, targetModelClass)\n      }\n    }\n\n    return this._relationships[relationshipName]\n  }\n\n  /**\n   * @param {Record<string, any>} attributes - Attributes to assign.\n   * @returns {void} - No return value.\n   */\n  assignAttributes(attributes) {\n    for (const key in attributes) {\n      this.setAttribute(key, attributes[key])\n    }\n  }\n\n  /**\n   * @returns {void} - Clears cached relationship state.\n   */\n  clearRelationshipCache() {\n    this._relationships = {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {string} - Primary key name.\n   */\n  static primaryKey() {\n    return this.resourceConfig().primaryKey || \"id\"\n  }\n\n  /**\n   * @returns {number | string} - Primary key value.\n   */\n  primaryKeyValue() {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n    const value = this.readAttribute(ModelClass.primaryKey())\n\n    if (value === undefined || value === null) {\n      throw new Error(`Missing primary key '${ModelClass.primaryKey()}' on ${ModelClass.name}`)\n    }\n\n    return value\n  }\n\n  /**\n   * @param {string} attributeName - Attribute name.\n   * @returns {any} - Attribute value.\n   */\n  readAttribute(attributeName) {\n    if (this._selectedAttributes && !this._selectedAttributes.has(attributeName)) {\n      throw new AttributeNotSelectedError(this.constructor.name, attributeName)\n    }\n\n    return this._attributes[attributeName]\n  }\n\n  /**\n   * @param {string} attributeName - Attribute name.\n   * @param {any} newValue - New value.\n   * @returns {any} - Assigned value.\n   */\n  setAttribute(attributeName, newValue) {\n    const previousValue = this._attributes[attributeName]\n\n    this._attributes[attributeName] = newValue\n\n    if (this._selectedAttributes) {\n      this._selectedAttributes.add(attributeName)\n    }\n\n    if (!Object.is(previousValue, newValue)) {\n      this.clearRelationshipCache()\n    }\n\n    return newValue\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {string} - Resource path.\n   */\n  static resourcePath() {\n    const path = this.resourceConfig().path\n\n    if (!path) throw new Error(`Missing resource path for ${this.name}`)\n\n    return path\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {\"find\" | \"index\" | \"update\" | \"destroy\"} commandType - Command type.\n   * @returns {string} - Resolved command name.\n   */\n  static commandName(commandType) {\n    const commands = this.resourceConfig().commands || {}\n\n    return commands[commandType] || commandType\n  }\n\n  /**\n   * @param {FrontendModelTransportConfig} config - Frontend model transport configuration.\n   * @returns {void} - No return value.\n   */\n  static configureTransport(config) {\n    if (!config || typeof config !== \"object\") {\n      return\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"baseUrl\")) {\n      frontendModelTransportConfig.baseUrl = config.baseUrl\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"baseUrlResolver\")) {\n      frontendModelTransportConfig.baseUrlResolver = config.baseUrlResolver\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"credentials\")) {\n      frontendModelTransportConfig.credentials = config.credentials\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"pathPrefix\")) {\n      frontendModelTransportConfig.pathPrefix = config.pathPrefix\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"pathPrefixResolver\")) {\n      frontendModelTransportConfig.pathPrefixResolver = config.pathPrefixResolver\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"request\")) {\n      frontendModelTransportConfig.request = config.request\n    }\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {object} response - Response payload.\n   * @returns {Record<string, any>} - Attributes from payload.\n   */\n  static attributesFromResponse(response) {\n    const modelData = this.modelDataFromResponse(response)\n\n    return modelData.attributes\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {object} response - Response payload.\n   * @returns {{attributes: Record<string, any>, preloadedRelationships: Record<string, any>, selectedAttributes: string[] | null}} - Attributes and preload/select payload.\n   */\n  static modelDataFromResponse(response) {\n    if (!response || typeof response !== \"object\") {\n      throw new Error(`Expected object response but got: ${response}`)\n    }\n\n    /** @type {Record<string, any>} */\n    let modelData\n\n    if (response.model && typeof response.model === \"object\") {\n      modelData = response.model\n    } else if (response.attributes && typeof response.attributes === \"object\") {\n      modelData = response.attributes\n    } else {\n      modelData = /** @type {Record<string, any>} */ (response)\n    }\n\n    const attributes = {...modelData}\n    const preloadedRelationships = isPlainObject(attributes[PRELOADED_RELATIONSHIPS_KEY])\n      ? /** @type {Record<string, any>} */ (attributes[PRELOADED_RELATIONSHIPS_KEY])\n      : {}\n    const selectedAttributesFromPayload = Array.isArray(attributes[SELECTED_ATTRIBUTES_KEY])\n      ? /** @type {string[]} */ (attributes[SELECTED_ATTRIBUTES_KEY]).filter((attributeName) => typeof attributeName === \"string\")\n      : null\n\n    delete attributes[PRELOADED_RELATIONSHIPS_KEY]\n    delete attributes[SELECTED_ATTRIBUTES_KEY]\n\n    const selectedAttributes = selectedAttributesFromPayload || Object.keys(attributes)\n\n    return {attributes, preloadedRelationships, selectedAttributes}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {FrontendModelBase} model - Model instance.\n   * @param {Record<string, any>} preloadedRelationships - Preloaded relationship payload.\n   * @returns {void}\n   */\n  static applyPreloadedRelationships(model, preloadedRelationships) {\n    for (const [relationshipName, relationshipPayload] of Object.entries(preloadedRelationships)) {\n      const relationship = model.getRelationshipByName(relationshipName)\n      const targetModelClass = this.relationshipModelClass(relationshipName)\n\n      if (Array.isArray(relationshipPayload)) {\n        relationship.setLoaded(relationshipPayload.map((entry) => this.instantiateRelationshipValue(entry, targetModelClass)))\n        continue\n      }\n\n      relationship.setLoaded(this.instantiateRelationshipValue(relationshipPayload, targetModelClass))\n    }\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {any} relationshipPayload - Relationship payload value.\n   * @param {typeof FrontendModelBase | null} targetModelClass - Target model class.\n   * @returns {any} - Instantiated relationship value.\n   */\n  static instantiateRelationshipValue(relationshipPayload, targetModelClass) {\n    if (!targetModelClass) return relationshipPayload\n\n    if (!relationshipPayload || typeof relationshipPayload !== \"object\") return relationshipPayload\n\n    return targetModelClass.instantiateFromResponse(relationshipPayload)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} response - Response payload.\n   * @returns {InstanceType<T>} - New model instance.\n   */\n  static instantiateFromResponse(response) {\n    const modelData = this.modelDataFromResponse(response)\n    const attributes = modelData.attributes\n    const preloadedRelationships = modelData.preloadedRelationships\n    const selectedAttributes = modelData.selectedAttributes\n    const model = /** @type {InstanceType<T>} */ (new this(attributes))\n    model._selectedAttributes = selectedAttributes ? new Set(selectedAttributes) : null\n\n    this.applyPreloadedRelationships(model, preloadedRelationships)\n\n    return model\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {number | string} id - Record identifier.\n   * @returns {Promise<InstanceType<T>>} - Resolved model.\n   */\n  static async find(id) {\n    return await this.query().find(id)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} conditions - Attribute match conditions.\n   * @returns {Promise<InstanceType<T> | null>} - Found model or null.\n   */\n  static async findBy(conditions) {\n    return await this.query().findBy(conditions)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} conditions - Attribute match conditions.\n   * @returns {Promise<InstanceType<T>>} - Found model.\n   */\n  static async findByOrFail(conditions) {\n    return await this.query().findByOrFail(conditions)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @returns {Promise<InstanceType<T>[]>} - Loaded model instances.\n   */\n  static async toArray() {\n    return await this.query().toArray()\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @returns {FrontendModelQuery<T>} - Query builder.\n   */\n  static query() {\n    return /** @type {FrontendModelQuery<T>} */ (new FrontendModelQuery({modelClass: this}))\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {import(\"../database/query/index.js\").NestedPreloadRecord | string | string[]} preload - Preload graph.\n   * @returns {FrontendModelQuery<T>} - Query with preload.\n   */\n  static preload(preload) {\n    return /** @type {FrontendModelQuery<T>} */ (this.query().preload(preload))\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, string[] | string>} select - Model-aware attribute select map.\n   * @returns {FrontendModelQuery<T>} - Query with selected attributes.\n   */\n  static select(select) {\n    return /** @type {FrontendModelQuery<T>} */ (this.query().select(select))\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {Record<string, any>} conditions - findBy conditions.\n   * @returns {void}\n   */\n  static assertFindByConditions(conditions) {\n    assertFindByConditionsShape(conditions)\n    const normalizedConditions = normalizeFindConditions(conditions)\n\n    Object.keys(conditions).forEach((key) => {\n      assertDefinedFindByConditionValue(conditions[key], key)\n      assertFindByConditionSerializationPreservesValue(conditions[key], normalizedConditions[key], key)\n    })\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {FrontendModelBase} model - Candidate model.\n   * @param {Record<string, any>} conditions - Match conditions.\n   * @returns {boolean} - Whether the model matches all conditions.\n   */\n  static matchesFindByConditions(model, conditions) {\n    const modelAttributes = model.attributes()\n\n    for (const key of Object.keys(conditions)) {\n      const expectedValue = conditions[key]\n      const actualValue = modelAttributes[key]\n\n      if (Array.isArray(expectedValue)) {\n        if (Array.isArray(actualValue)) {\n          if (!this.findByConditionValueMatches(actualValue, expectedValue)) {\n            return false\n          }\n        } else if (!expectedValue.some((entry) => this.findByConditionValueMatches(actualValue, entry))) {\n          return false\n        }\n      } else if (!this.findByConditionValueMatches(actualValue, expectedValue)) {\n        return false\n      }\n    }\n\n    return true\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {unknown} actualValue - Actual model value.\n   * @param {unknown} expectedValue - Expected find condition value.\n   * @returns {boolean} - Whether values match.\n   */\n  static findByConditionValueMatches(actualValue, expectedValue) {\n    if (expectedValue === null) {\n      return actualValue === null\n    }\n\n    if (Array.isArray(expectedValue)) {\n      if (!Array.isArray(actualValue)) {\n        return false\n      }\n\n      if (actualValue.length !== expectedValue.length) {\n        return false\n      }\n\n      for (let index = 0; index < expectedValue.length; index += 1) {\n        if (!this.findByConditionValueMatches(actualValue[index], expectedValue[index])) {\n          return false\n        }\n      }\n\n      return true\n    }\n\n    if (expectedValue && typeof expectedValue === \"object\") {\n      if (!actualValue || typeof actualValue !== \"object\" || Array.isArray(actualValue)) {\n        return false\n      }\n\n      const actualObject = /** @type {Record<string, unknown>} */ (actualValue)\n      const expectedObject = /** @type {Record<string, unknown>} */ (expectedValue)\n      const actualKeys = Object.keys(actualObject)\n      const expectedKeys = Object.keys(expectedObject)\n\n      if (actualKeys.length !== expectedKeys.length) {\n        return false\n      }\n\n      for (const key of expectedKeys) {\n        if (!Object.prototype.hasOwnProperty.call(actualObject, key)) {\n          return false\n        }\n\n        if (!this.findByConditionValueMatches(actualObject[key], expectedObject[key])) {\n          return false\n        }\n      }\n\n      return true\n    }\n\n    if (actualValue === expectedValue) {\n      return true\n    }\n\n    return this.findByPrimitiveValuesMatch(actualValue, expectedValue)\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {unknown} actualValue - Actual model value.\n   * @param {unknown} expectedValue - Expected find condition value.\n   * @returns {boolean} - Whether primitive values match after safe coercion.\n   */\n  static findByPrimitiveValuesMatch(actualValue, expectedValue) {\n    if (actualValue instanceof Date && typeof expectedValue === \"string\") {\n      return actualValue.toISOString() === expectedValue\n    }\n\n    if (typeof actualValue === \"string\" && expectedValue instanceof Date) {\n      return actualValue === expectedValue.toISOString()\n    }\n\n    if (actualValue instanceof Date && expectedValue instanceof Date) {\n      return actualValue.toISOString() === expectedValue.toISOString()\n    }\n\n    if (typeof actualValue === \"number\" && typeof expectedValue === \"string\") {\n      return this.findByNumericStringMatchesNumber(expectedValue, actualValue)\n    }\n\n    if (typeof actualValue === \"string\" && typeof expectedValue === \"number\") {\n      return this.findByNumericStringMatchesNumber(actualValue, expectedValue)\n    }\n\n    return false\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} numericString - Numeric string value.\n   * @param {number} expectedNumber - Number value.\n   * @returns {boolean} - Whether values represent the same number.\n   */\n  static findByNumericStringMatchesNumber(numericString, expectedNumber) {\n    if (!Number.isFinite(expectedNumber)) {\n      return false\n    }\n\n    if (!/^-?\\d+(?:\\.\\d+)?$/.test(numericString)) {\n      return false\n    }\n\n    return Number(numericString) === expectedNumber\n  }\n\n  /**\n   * @param {Record<string, any>} [newAttributes] - New values to assign before update.\n   * @returns {Promise<this>} - Updated model.\n   */\n  async update(newAttributes = {}) {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n\n    this.assignAttributes(newAttributes)\n\n    const response = await ModelClass.executeCommand(\"update\", {\n      attributes: this.attributes(),\n      id: this.primaryKeyValue()\n    })\n\n    this.assignAttributes(ModelClass.attributesFromResponse(response))\n\n    return this\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when destroyed on backend.\n   */\n  async destroy() {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n\n    await ModelClass.executeCommand(\"destroy\", {\n      id: this.primaryKeyValue()\n    })\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {\"find\" | \"index\" | \"update\" | \"destroy\"} commandType - Command type.\n   * @param {Record<string, any>} payload - Command payload.\n   * @returns {Promise<Record<string, any>>} - Parsed JSON response.\n   */\n  static async executeCommand(commandType, payload) {\n    const commandName = this.commandName(commandType)\n    const url = frontendModelCommandUrl(this.resourcePath(), commandName)\n    const serializedPayload = /** @type {Record<string, any>} */ (serializeFrontendModelTransportValue(payload))\n\n    if (frontendModelTransportConfig.request) {\n      const customResponse = await frontendModelTransportConfig.request({\n        commandName,\n        commandType,\n        modelClass: this,\n        payload: serializedPayload,\n        url\n      })\n\n      return /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(customResponse))\n    }\n\n    const response = await fetch(url, {\n      body: JSON.stringify(serializedPayload),\n      credentials: frontendModelTransportConfig.credentials,\n      headers: {\n        \"Content-Type\": \"application/json\"\n      },\n      method: \"POST\"\n    })\n\n    if (!response.ok) {\n      throw new Error(`Request failed (${response.status}) for ${this.name}#${commandType}`)\n    }\n\n    const responseText = await response.text()\n    const json = responseText.length > 0 ? JSON.parse(responseText) : {}\n\n    return /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(json))\n  }\n}\n"]}
897
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../../src/frontend-models/base.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport FrontendModelQuery from \"./query.js\"\nimport {deserializeFrontendModelTransportValue, serializeFrontendModelTransportValue} from \"./transport-serialization.js\"\n\n/** @typedef {{commands?: Record<string, string>, path?: string, primaryKey?: string}} FrontendModelResourceConfig */\n/**\n * @typedef {object} FrontendModelTransportConfig\n * @property {string} [baseUrl] - Optional base URL prefixed before resource paths.\n * @property {(() => string | undefined | null)} [baseUrlResolver] - Optional resolver used per request for dynamic base URL.\n * @property {string} [pathPrefix] - Optional path prefix inserted between base URL and resource path.\n * @property {(() => string | undefined | null)} [pathPrefixResolver] - Optional resolver used per request for dynamic path prefix.\n * @property {\"omit\" | \"same-origin\" | \"include\"} [credentials] - Optional credentials mode forwarded to fetch.\n * @property {((args: {commandName: string, commandType: \"find\" | \"index\" | \"update\" | \"destroy\", modelClass: typeof FrontendModelBase, payload: Record<string, any>, url: string}) => Promise<Record<string, any>>)} [request] - Optional custom transport handler.\n */\n\n/** @type {FrontendModelTransportConfig} */\nconst frontendModelTransportConfig = {}\nconst PRELOADED_RELATIONSHIPS_KEY = \"__preloadedRelationships\"\nconst SELECTED_ATTRIBUTES_KEY = \"__selectedAttributes\"\n\n/** Error raised when reading an attribute that was not selected in query payloads. */\nexport class AttributeNotSelectedError extends Error {\n  /**\n   * @param {string} modelName - Model class name.\n   * @param {string} attributeName - Attribute that was requested.\n   */\n  constructor(modelName, attributeName) {\n    super(`${modelName}#${attributeName} was not selected`)\n    this.name = \"AttributeNotSelectedError\"\n  }\n}\n\n/**\n * @param {unknown} value - Candidate value.\n * @returns {value is Record<string, any>} - Whether value is a plain object.\n */\nfunction isPlainObject(value) {\n  if (!value || typeof value !== \"object\" || Array.isArray(value)) return false\n\n  const prototype = Object.getPrototypeOf(value)\n\n  return prototype === Object.prototype || prototype === null\n}\n\n/**\n * Lightweight singular relationship state holder for frontend model instances.\n * @template {typeof FrontendModelBase} S\n * @template {typeof FrontendModelBase} T\n */\nexport class FrontendModelSingularRelationship {\n  /**\n   * @param {InstanceType<S>} model - Parent model.\n   * @param {string} relationshipName - Relationship name.\n   * @param {T | null} targetModelClass - Target model class.\n   */\n  constructor(model, relationshipName, targetModelClass) {\n    this.model = model\n    this.relationshipName = relationshipName\n    this.targetModelClass = targetModelClass\n    this._preloaded = false\n    this._loadedValue = null\n  }\n\n  /**\n   * @param {any} loadedValue - Loaded relationship value.\n   * @returns {void}\n   */\n  setLoaded(loadedValue) {\n    this._loadedValue = loadedValue == undefined ? null : loadedValue\n    this._preloaded = true\n  }\n\n  /**\n   * @returns {boolean} - Whether relationship is preloaded.\n   */\n  getPreloaded() {\n    return this._preloaded\n  }\n\n  /**\n   * @returns {any} - Loaded relationship value.\n   */\n  loaded() {\n    if (!this._preloaded) {\n      throw new Error(`${this.model.constructor.name}#${this.relationshipName} hasn't been preloaded`)\n    }\n\n    return this._loadedValue\n  }\n\n  /**\n   * @param {Record<string, any>} [attributes] - New model attributes.\n   * @returns {InstanceType<T>} - Built model.\n   */\n  build(attributes = {}) {\n    if (!this.targetModelClass) {\n      throw new Error(`No target model class configured for ${this.model.constructor.name}#${this.relationshipName}`)\n    }\n\n    const model = /** @type {InstanceType<T>} */ (new this.targetModelClass(attributes))\n\n    this.setLoaded(model)\n\n    return model\n  }\n}\n\n/**\n * Lightweight has-many relationship state holder for frontend model instances.\n * @template {typeof FrontendModelBase} S\n * @template {typeof FrontendModelBase} T\n */\nexport class FrontendModelHasManyRelationship {\n  /**\n   * @param {InstanceType<S>} model - Parent model.\n   * @param {string} relationshipName - Relationship name.\n   * @param {T | null} targetModelClass - Target model class.\n   */\n  constructor(model, relationshipName, targetModelClass) {\n    this.model = model\n    this.relationshipName = relationshipName\n    this.targetModelClass = targetModelClass\n    this._preloaded = false\n    this._loadedValue = []\n  }\n\n  /**\n   * @param {Array<InstanceType<T>>} loadedValue - Loaded relationship value.\n   * @returns {void}\n   */\n  setLoaded(loadedValue) {\n    this._loadedValue = Array.isArray(loadedValue) ? loadedValue : []\n    this._preloaded = true\n  }\n\n  /**\n   * @returns {boolean} - Whether relationship is preloaded.\n   */\n  getPreloaded() {\n    return this._preloaded\n  }\n\n  /**\n   * @returns {Array<InstanceType<T>>} - Loaded relationship values.\n   */\n  loaded() {\n    if (!this._preloaded) {\n      throw new Error(`${this.model.constructor.name}#${this.relationshipName} hasn't been preloaded`)\n    }\n\n    return this._loadedValue\n  }\n\n  /**\n   * @param {Array<InstanceType<T>>} models - Models to append.\n   * @returns {void}\n   */\n  addToLoaded(models) {\n    const loadedModels = this.getPreloaded() ? this.loaded() : []\n\n    this.setLoaded([...loadedModels, ...models])\n  }\n\n  /**\n   * @param {Record<string, any>} [attributes] - New model attributes.\n   * @returns {InstanceType<T>} - Built model.\n   */\n  build(attributes = {}) {\n    if (!this.targetModelClass) {\n      throw new Error(`No target model class configured for ${this.model.constructor.name}#${this.relationshipName}`)\n    }\n\n    const model = /** @type {InstanceType<T>} */ (new this.targetModelClass(attributes))\n\n    this.addToLoaded([model])\n\n    return model\n  }\n}\n\n/**\n * @param {string} relationshipType - Relationship type.\n * @returns {boolean} - Whether relationship type is has-many.\n */\nfunction relationshipTypeIsCollection(relationshipType) {\n  return relationshipType == \"hasMany\"\n}\n\n/**\n * @param {string | undefined | null} value - Base URL candidate.\n * @returns {string} - Normalized base URL without trailing slash.\n */\nfunction normalizeBaseUrl(value) {\n  if (typeof value !== \"string\") return \"\"\n\n  const trimmed = value.trim()\n\n  if (!trimmed.length) return \"\"\n\n  return trimmed.replace(/\\/+$/, \"\")\n}\n\n/**\n * @param {string | undefined | null} value - Path prefix candidate.\n * @returns {string} - Normalized path prefix with leading slash and no trailing slash.\n */\nfunction normalizePathPrefix(value) {\n  if (typeof value !== \"string\") return \"\"\n\n  const trimmed = value.trim()\n\n  if (!trimmed.length) return \"\"\n\n  const withLeadingSlash = trimmed.startsWith(\"/\") ? trimmed : `/${trimmed}`\n\n  return withLeadingSlash.replace(/\\/+$/, \"\")\n}\n\n/**\n * @param {string} resourcePath - Resource path (expected absolute path).\n * @param {string} commandName - Command name.\n * @returns {string} - Combined command URL.\n */\nfunction frontendModelCommandUrl(resourcePath, commandName) {\n  const resolvedBaseUrl = frontendModelTransportConfig.baseUrlResolver\n    ? frontendModelTransportConfig.baseUrlResolver()\n    : frontendModelTransportConfig.baseUrl\n  const baseUrl = normalizeBaseUrl(resolvedBaseUrl)\n  const resolvedPathPrefix = frontendModelTransportConfig.pathPrefixResolver\n    ? frontendModelTransportConfig.pathPrefixResolver()\n    : frontendModelTransportConfig.pathPrefix\n  const pathPrefix = normalizePathPrefix(resolvedPathPrefix)\n  const normalizedResourcePath = resourcePath.startsWith(\"/\") ? resourcePath : `/${resourcePath}`\n\n  return `${baseUrl}${pathPrefix}${normalizedResourcePath}/${commandName}`\n}\n\n/**\n * @param {Record<string, any>} conditions - findBy conditions.\n * @returns {Record<string, any>} - JSON-normalized conditions.\n */\nfunction normalizeFindConditions(conditions) {\n  try {\n    return /** @type {Record<string, any>} */ (JSON.parse(JSON.stringify(conditions)))\n  } catch (error) {\n    throw new Error(`findBy conditions could not be serialized: ${error instanceof Error ? error.message : String(error)}`)\n  }\n}\n\n/**\n * @param {unknown} conditions - findBy conditions.\n * @returns {void}\n */\nfunction assertFindByConditionsShape(conditions) {\n  if (!conditions || typeof conditions !== \"object\" || Array.isArray(conditions)) {\n    throw new Error(`findBy expects conditions to be a plain object, got: ${conditions}`)\n  }\n\n  const conditionsPrototype = Object.getPrototypeOf(conditions)\n\n  if (conditionsPrototype !== Object.prototype && conditionsPrototype !== null) {\n    throw new Error(`findBy expects conditions to be a plain object, got: ${conditions}`)\n  }\n\n  const symbolKeys = Object.getOwnPropertySymbols(conditions)\n\n  if (symbolKeys.length > 0) {\n    throw new Error(`findBy does not support symbol condition keys (keys: ${symbolKeys.map((key) => key.toString()).join(\", \")})`)\n  }\n}\n\n/**\n * @param {unknown} value - Condition value to validate.\n * @param {string} keyPath - Key path for error output.\n * @returns {void}\n */\nfunction assertDefinedFindByConditionValue(value, keyPath) {\n  if (value === undefined) {\n    throw new Error(`findBy does not support undefined condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"function\") {\n    throw new Error(`findBy does not support function condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"symbol\") {\n    throw new Error(`findBy does not support symbol condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"bigint\") {\n    throw new Error(`findBy does not support bigint condition values (key: ${keyPath})`)\n  }\n\n  if (typeof value === \"number\" && !Number.isFinite(value)) {\n    throw new Error(`findBy does not support non-finite number condition values (key: ${keyPath})`)\n  }\n\n  if (Array.isArray(value)) {\n    value.forEach((entry, index) => {\n      assertDefinedFindByConditionValue(entry, `${keyPath}[${index}]`)\n    })\n    return\n  }\n\n  if (value && typeof value === \"object\") {\n    if (value instanceof Date) {\n      return\n    }\n\n    const objectValue = /** @type {Record<string, unknown>} */ (value)\n    const prototype = Object.getPrototypeOf(objectValue)\n\n    if (prototype !== Object.prototype && prototype !== null) {\n      throw new Error(`findBy does not support non-plain object condition values (key: ${keyPath})`)\n    }\n\n    const symbolKeys = Object.getOwnPropertySymbols(objectValue)\n\n    if (symbolKeys.length > 0) {\n      throw new Error(`findBy does not support symbol condition keys (key: ${keyPath})`)\n    }\n\n    const valueObject = /** @type {Record<string, unknown>} */ (value)\n\n    Object.keys(valueObject).forEach((nestedKey) => {\n      assertDefinedFindByConditionValue(valueObject[nestedKey], `${keyPath}.${nestedKey}`)\n    })\n  }\n}\n\n/**\n * @param {unknown} originalValue - Original condition value.\n * @param {unknown} normalizedValue - JSON-normalized condition value.\n * @param {string} keyPath - Key path for error output.\n * @returns {void}\n */\nfunction assertFindByConditionSerializationPreservesValue(originalValue, normalizedValue, keyPath) {\n  if (originalValue === null) {\n    if (normalizedValue !== null) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    return\n  }\n\n  if (Array.isArray(originalValue)) {\n    if (!Array.isArray(normalizedValue)) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    if (originalValue.length !== normalizedValue.length) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    for (let index = 0; index < originalValue.length; index += 1) {\n      assertFindByConditionSerializationPreservesValue(originalValue[index], normalizedValue[index], `${keyPath}[${index}]`)\n    }\n\n    return\n  }\n\n  if (originalValue instanceof Date) {\n    if (typeof normalizedValue !== \"string\") {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    return\n  }\n\n  if (originalValue && typeof originalValue === \"object\") {\n    if (!normalizedValue || typeof normalizedValue !== \"object\" || Array.isArray(normalizedValue)) {\n      throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n    }\n\n    const normalizedObject = /** @type {Record<string, unknown>} */ (normalizedValue)\n    const originalObject = /** @type {Record<string, unknown>} */ (originalValue)\n\n    Object.keys(originalObject).forEach((nestedKey) => {\n      if (!(nestedKey in normalizedObject)) {\n        throw new Error(`findBy condition key was removed during serialization (key: ${keyPath}.${nestedKey})`)\n      }\n\n      assertFindByConditionSerializationPreservesValue(originalObject[nestedKey], normalizedObject[nestedKey], `${keyPath}.${nestedKey}`)\n    })\n\n    return\n  }\n\n  if (normalizedValue !== originalValue) {\n    throw new Error(`findBy condition changed during serialization (key: ${keyPath})`)\n  }\n}\n\n/** Base class for generated frontend model classes. */\nexport default class FrontendModelBase {\n  /**\n   * @param {Record<string, any>} [attributes] - Initial attributes.\n   */\n  constructor(attributes = {}) {\n    this._attributes = {}\n    this._relationships = {}\n    this._selectedAttributes = null\n    this.assignAttributes(attributes)\n  }\n\n  /**\n   * @returns {FrontendModelResourceConfig} - Resource configuration.\n   */\n  static resourceConfig() {\n    throw new Error(\"resourceConfig() must be implemented by subclasses\")\n    // eslint-disable-next-line no-unreachable\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {Record<string, typeof FrontendModelBase>} - Relationship model classes keyed by relationship name.\n   */\n  static relationshipModelClasses() {\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {Record<string, {type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationship definitions keyed by relationship name.\n   */\n  static relationshipDefinitions() {\n    return {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} relationshipName - Relationship name.\n   * @returns {{type: \"belongsTo\" | \"hasOne\" | \"hasMany\"} | null} - Relationship definition.\n   */\n  static relationshipDefinition(relationshipName) {\n    const definitions = this.relationshipDefinitions()\n\n    return definitions[relationshipName] || null\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} relationshipName - Relationship name.\n   * @returns {typeof FrontendModelBase | null} - Target relationship model class.\n   */\n  static relationshipModelClass(relationshipName) {\n    const relationshipModelClasses = this.relationshipModelClasses()\n\n    return relationshipModelClasses[relationshipName] || null\n  }\n\n  /**\n   * @returns {Record<string, any>} - Attributes hash.\n   */\n  attributes() {\n    return this._attributes\n  }\n\n  /**\n   * @param {string} relationshipName - Relationship name.\n   * @returns {FrontendModelHasManyRelationship<any, any> | FrontendModelSingularRelationship<any, any>} - Relationship state object.\n   */\n  getRelationshipByName(relationshipName) {\n    if (!this._relationships[relationshipName]) {\n      const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n      const relationshipDefinition = ModelClass.relationshipDefinition(relationshipName)\n      const targetModelClass = ModelClass.relationshipModelClass(relationshipName)\n\n      if (relationshipDefinition && relationshipTypeIsCollection(relationshipDefinition.type)) {\n        this._relationships[relationshipName] = new FrontendModelHasManyRelationship(this, relationshipName, targetModelClass)\n      } else {\n        this._relationships[relationshipName] = new FrontendModelSingularRelationship(this, relationshipName, targetModelClass)\n      }\n    }\n\n    return this._relationships[relationshipName]\n  }\n\n  /**\n   * @param {Record<string, any>} attributes - Attributes to assign.\n   * @returns {void} - No return value.\n   */\n  assignAttributes(attributes) {\n    for (const key in attributes) {\n      this.setAttribute(key, attributes[key])\n    }\n  }\n\n  /**\n   * @returns {void} - Clears cached relationship state.\n   */\n  clearRelationshipCache() {\n    this._relationships = {}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {string} - Primary key name.\n   */\n  static primaryKey() {\n    return this.resourceConfig().primaryKey || \"id\"\n  }\n\n  /**\n   * @returns {number | string} - Primary key value.\n   */\n  primaryKeyValue() {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n    const value = this.readAttribute(ModelClass.primaryKey())\n\n    if (value === undefined || value === null) {\n      throw new Error(`Missing primary key '${ModelClass.primaryKey()}' on ${ModelClass.name}`)\n    }\n\n    return value\n  }\n\n  /**\n   * @param {string} attributeName - Attribute name.\n   * @returns {any} - Attribute value.\n   */\n  readAttribute(attributeName) {\n    if (this._selectedAttributes && !this._selectedAttributes.has(attributeName)) {\n      throw new AttributeNotSelectedError(this.constructor.name, attributeName)\n    }\n\n    return this._attributes[attributeName]\n  }\n\n  /**\n   * @param {string} attributeName - Attribute name.\n   * @param {any} newValue - New value.\n   * @returns {any} - Assigned value.\n   */\n  setAttribute(attributeName, newValue) {\n    const previousValue = this._attributes[attributeName]\n\n    this._attributes[attributeName] = newValue\n\n    if (this._selectedAttributes) {\n      this._selectedAttributes.add(attributeName)\n    }\n\n    if (!Object.is(previousValue, newValue)) {\n      this.clearRelationshipCache()\n    }\n\n    return newValue\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {string} - Resource path.\n   */\n  static resourcePath() {\n    const path = this.resourceConfig().path\n\n    if (!path) throw new Error(`Missing resource path for ${this.name}`)\n\n    return path\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {\"find\" | \"index\" | \"update\" | \"destroy\"} commandType - Command type.\n   * @returns {string} - Resolved command name.\n   */\n  static commandName(commandType) {\n    const commands = this.resourceConfig().commands || {}\n\n    return commands[commandType] || commandType\n  }\n\n  /**\n   * @param {FrontendModelTransportConfig} config - Frontend model transport configuration.\n   * @returns {void} - No return value.\n   */\n  static configureTransport(config) {\n    if (!config || typeof config !== \"object\") {\n      return\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"baseUrl\")) {\n      frontendModelTransportConfig.baseUrl = config.baseUrl\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"baseUrlResolver\")) {\n      frontendModelTransportConfig.baseUrlResolver = config.baseUrlResolver\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"credentials\")) {\n      frontendModelTransportConfig.credentials = config.credentials\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"pathPrefix\")) {\n      frontendModelTransportConfig.pathPrefix = config.pathPrefix\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"pathPrefixResolver\")) {\n      frontendModelTransportConfig.pathPrefixResolver = config.pathPrefixResolver\n    }\n\n    if (Object.prototype.hasOwnProperty.call(config, \"request\")) {\n      frontendModelTransportConfig.request = config.request\n    }\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {object} response - Response payload.\n   * @returns {Record<string, any>} - Attributes from payload.\n   */\n  static attributesFromResponse(response) {\n    const modelData = this.modelDataFromResponse(response)\n\n    return modelData.attributes\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {object} response - Response payload.\n   * @returns {{attributes: Record<string, any>, preloadedRelationships: Record<string, any>, selectedAttributes: string[] | null}} - Attributes and preload/select payload.\n   */\n  static modelDataFromResponse(response) {\n    if (!response || typeof response !== \"object\") {\n      throw new Error(`Expected object response but got: ${response}`)\n    }\n\n    /** @type {Record<string, any>} */\n    let modelData\n\n    if (response.model && typeof response.model === \"object\") {\n      modelData = response.model\n    } else if (response.attributes && typeof response.attributes === \"object\") {\n      modelData = response.attributes\n    } else {\n      modelData = /** @type {Record<string, any>} */ (response)\n    }\n\n    const attributes = {...modelData}\n    const preloadedRelationships = isPlainObject(attributes[PRELOADED_RELATIONSHIPS_KEY])\n      ? /** @type {Record<string, any>} */ (attributes[PRELOADED_RELATIONSHIPS_KEY])\n      : {}\n    const selectedAttributesFromPayload = Array.isArray(attributes[SELECTED_ATTRIBUTES_KEY])\n      ? /** @type {string[]} */ (attributes[SELECTED_ATTRIBUTES_KEY]).filter((attributeName) => typeof attributeName === \"string\")\n      : null\n\n    delete attributes[PRELOADED_RELATIONSHIPS_KEY]\n    delete attributes[SELECTED_ATTRIBUTES_KEY]\n\n    const selectedAttributes = selectedAttributesFromPayload || Object.keys(attributes)\n\n    return {attributes, preloadedRelationships, selectedAttributes}\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {FrontendModelBase} model - Model instance.\n   * @param {Record<string, any>} preloadedRelationships - Preloaded relationship payload.\n   * @returns {void}\n   */\n  static applyPreloadedRelationships(model, preloadedRelationships) {\n    for (const [relationshipName, relationshipPayload] of Object.entries(preloadedRelationships)) {\n      const relationship = model.getRelationshipByName(relationshipName)\n      const targetModelClass = this.relationshipModelClass(relationshipName)\n\n      if (Array.isArray(relationshipPayload)) {\n        relationship.setLoaded(relationshipPayload.map((entry) => this.instantiateRelationshipValue(entry, targetModelClass)))\n        continue\n      }\n\n      relationship.setLoaded(this.instantiateRelationshipValue(relationshipPayload, targetModelClass))\n    }\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {any} relationshipPayload - Relationship payload value.\n   * @param {typeof FrontendModelBase | null} targetModelClass - Target model class.\n   * @returns {any} - Instantiated relationship value.\n   */\n  static instantiateRelationshipValue(relationshipPayload, targetModelClass) {\n    if (!targetModelClass) return relationshipPayload\n\n    if (!relationshipPayload || typeof relationshipPayload !== \"object\") return relationshipPayload\n\n    return targetModelClass.instantiateFromResponse(relationshipPayload)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} response - Response payload.\n   * @returns {InstanceType<T>} - New model instance.\n   */\n  static instantiateFromResponse(response) {\n    const modelData = this.modelDataFromResponse(response)\n    const attributes = modelData.attributes\n    const preloadedRelationships = modelData.preloadedRelationships\n    const selectedAttributes = modelData.selectedAttributes\n    const model = /** @type {InstanceType<T>} */ (new this(attributes))\n    model._selectedAttributes = selectedAttributes ? new Set(selectedAttributes) : null\n\n    this.applyPreloadedRelationships(model, preloadedRelationships)\n\n    return model\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {number | string} id - Record identifier.\n   * @returns {Promise<InstanceType<T>>} - Resolved model.\n   */\n  static async find(id) {\n    return await this.query().find(id)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} conditions - Attribute match conditions.\n   * @returns {Promise<InstanceType<T> | null>} - Found model or null.\n   */\n  static async findBy(conditions) {\n    return await this.query().findBy(conditions)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, any>} conditions - Attribute match conditions.\n   * @returns {Promise<InstanceType<T>>} - Found model.\n   */\n  static async findByOrFail(conditions) {\n    return await this.query().findByOrFail(conditions)\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @returns {Promise<InstanceType<T>[]>} - Loaded model instances.\n   */\n  static async toArray() {\n    return await this.query().toArray()\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @returns {FrontendModelQuery<T>} - Query builder.\n   */\n  static query() {\n    return /** @type {FrontendModelQuery<T>} */ (new FrontendModelQuery({modelClass: this}))\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {import(\"../database/query/index.js\").NestedPreloadRecord | string | string[]} preload - Preload graph.\n   * @returns {FrontendModelQuery<T>} - Query with preload.\n   */\n  static preload(preload) {\n    return /** @type {FrontendModelQuery<T>} */ (this.query().preload(preload))\n  }\n\n  /**\n   * @template {typeof FrontendModelBase} T\n   * @this {T}\n   * @param {Record<string, string[] | string>} select - Model-aware attribute select map.\n   * @returns {FrontendModelQuery<T>} - Query with selected attributes.\n   */\n  static select(select) {\n    return /** @type {FrontendModelQuery<T>} */ (this.query().select(select))\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {Record<string, any>} conditions - findBy conditions.\n   * @returns {void}\n   */\n  static assertFindByConditions(conditions) {\n    assertFindByConditionsShape(conditions)\n    const normalizedConditions = normalizeFindConditions(conditions)\n\n    Object.keys(conditions).forEach((key) => {\n      assertDefinedFindByConditionValue(conditions[key], key)\n      assertFindByConditionSerializationPreservesValue(conditions[key], normalizedConditions[key], key)\n    })\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {FrontendModelBase} model - Candidate model.\n   * @param {Record<string, any>} conditions - Match conditions.\n   * @returns {boolean} - Whether the model matches all conditions.\n   */\n  static matchesFindByConditions(model, conditions) {\n    const modelAttributes = model.attributes()\n\n    for (const key of Object.keys(conditions)) {\n      const expectedValue = conditions[key]\n      const actualValue = modelAttributes[key]\n\n      if (Array.isArray(expectedValue)) {\n        if (Array.isArray(actualValue)) {\n          if (!this.findByConditionValueMatches(actualValue, expectedValue)) {\n            return false\n          }\n        } else if (!expectedValue.some((entry) => this.findByConditionValueMatches(actualValue, entry))) {\n          return false\n        }\n      } else if (!this.findByConditionValueMatches(actualValue, expectedValue)) {\n        return false\n      }\n    }\n\n    return true\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {unknown} actualValue - Actual model value.\n   * @param {unknown} expectedValue - Expected find condition value.\n   * @returns {boolean} - Whether values match.\n   */\n  static findByConditionValueMatches(actualValue, expectedValue) {\n    if (expectedValue === null) {\n      return actualValue === null\n    }\n\n    if (Array.isArray(expectedValue)) {\n      if (!Array.isArray(actualValue)) {\n        return false\n      }\n\n      if (actualValue.length !== expectedValue.length) {\n        return false\n      }\n\n      for (let index = 0; index < expectedValue.length; index += 1) {\n        if (!this.findByConditionValueMatches(actualValue[index], expectedValue[index])) {\n          return false\n        }\n      }\n\n      return true\n    }\n\n    if (expectedValue && typeof expectedValue === \"object\") {\n      if (!actualValue || typeof actualValue !== \"object\" || Array.isArray(actualValue)) {\n        return false\n      }\n\n      const actualObject = /** @type {Record<string, unknown>} */ (actualValue)\n      const expectedObject = /** @type {Record<string, unknown>} */ (expectedValue)\n      const actualKeys = Object.keys(actualObject)\n      const expectedKeys = Object.keys(expectedObject)\n\n      if (actualKeys.length !== expectedKeys.length) {\n        return false\n      }\n\n      for (const key of expectedKeys) {\n        if (!Object.prototype.hasOwnProperty.call(actualObject, key)) {\n          return false\n        }\n\n        if (!this.findByConditionValueMatches(actualObject[key], expectedObject[key])) {\n          return false\n        }\n      }\n\n      return true\n    }\n\n    if (actualValue === expectedValue) {\n      return true\n    }\n\n    return this.findByPrimitiveValuesMatch(actualValue, expectedValue)\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {unknown} actualValue - Actual model value.\n   * @param {unknown} expectedValue - Expected find condition value.\n   * @returns {boolean} - Whether primitive values match after safe coercion.\n   */\n  static findByPrimitiveValuesMatch(actualValue, expectedValue) {\n    if (actualValue instanceof Date && typeof expectedValue === \"string\") {\n      return actualValue.toISOString() === expectedValue\n    }\n\n    if (typeof actualValue === \"string\" && expectedValue instanceof Date) {\n      return actualValue === expectedValue.toISOString()\n    }\n\n    if (actualValue instanceof Date && expectedValue instanceof Date) {\n      return actualValue.toISOString() === expectedValue.toISOString()\n    }\n\n    if (typeof actualValue === \"number\" && typeof expectedValue === \"string\") {\n      return this.findByNumericStringMatchesNumber(expectedValue, actualValue)\n    }\n\n    if (typeof actualValue === \"string\" && typeof expectedValue === \"number\") {\n      return this.findByNumericStringMatchesNumber(actualValue, expectedValue)\n    }\n\n    return false\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {string} numericString - Numeric string value.\n   * @param {number} expectedNumber - Number value.\n   * @returns {boolean} - Whether values represent the same number.\n   */\n  static findByNumericStringMatchesNumber(numericString, expectedNumber) {\n    if (!Number.isFinite(expectedNumber)) {\n      return false\n    }\n\n    if (!/^-?\\d+(?:\\.\\d+)?$/.test(numericString)) {\n      return false\n    }\n\n    return Number(numericString) === expectedNumber\n  }\n\n  /**\n   * @param {Record<string, any>} [newAttributes] - New values to assign before update.\n   * @returns {Promise<this>} - Updated model.\n   */\n  async update(newAttributes = {}) {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n\n    this.assignAttributes(newAttributes)\n\n    const response = await ModelClass.executeCommand(\"update\", {\n      attributes: this.attributes(),\n      id: this.primaryKeyValue()\n    })\n\n    this.assignAttributes(ModelClass.attributesFromResponse(response))\n\n    return this\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when destroyed on backend.\n   */\n  async destroy() {\n    const ModelClass = /** @type {typeof FrontendModelBase} */ (this.constructor)\n\n    await ModelClass.executeCommand(\"destroy\", {\n      id: this.primaryKeyValue()\n    })\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {\"find\" | \"index\" | \"update\" | \"destroy\"} commandType - Command type.\n   * @param {Record<string, any>} payload - Command payload.\n   * @returns {Promise<Record<string, any>>} - Parsed JSON response.\n   */\n  static async executeCommand(commandType, payload) {\n    const commandName = this.commandName(commandType)\n    const url = frontendModelCommandUrl(this.resourcePath(), commandName)\n    const serializedPayload = /** @type {Record<string, any>} */ (serializeFrontendModelTransportValue(payload))\n\n    if (frontendModelTransportConfig.request) {\n      const customResponse = await frontendModelTransportConfig.request({\n        commandName,\n        commandType,\n        modelClass: this,\n        payload: serializedPayload,\n        url\n      })\n\n      const decodedResponse = /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(customResponse))\n\n      this.throwOnErrorFrontendModelResponse({\n        commandType,\n        response: decodedResponse\n      })\n\n      return decodedResponse\n    }\n\n    const response = await fetch(url, {\n      body: JSON.stringify(serializedPayload),\n      credentials: frontendModelTransportConfig.credentials,\n      headers: {\n        \"Content-Type\": \"application/json\"\n      },\n      method: \"POST\"\n    })\n\n    if (!response.ok) {\n      throw new Error(`Request failed (${response.status}) for ${this.name}#${commandType}`)\n    }\n\n    const responseText = await response.text()\n    const json = responseText.length > 0 ? JSON.parse(responseText) : {}\n    const decodedResponse = /** @type {Record<string, any>} */ (deserializeFrontendModelTransportValue(json))\n\n    this.throwOnErrorFrontendModelResponse({\n      commandType,\n      response: decodedResponse\n    })\n\n    return decodedResponse\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @param {object} args - Arguments.\n   * @param {\"find\" | \"index\" | \"update\" | \"destroy\"} args.commandType - Command type.\n   * @param {Record<string, any>} args.response - Decoded response.\n   * @returns {void}\n   */\n  static throwOnErrorFrontendModelResponse({commandType, response}) {\n    if (response?.status !== \"error\") return\n\n    const responseKeys = Object.keys(response)\n    const hasOnlyStatus = responseKeys.length === 1 && responseKeys[0] === \"status\"\n    const hasErrorMessage = typeof response.errorMessage === \"string\" && response.errorMessage.length > 0\n    const hasErrorEnvelopeKeys = Boolean(\n      response.code !== undefined\n      || response.error !== undefined\n      || response.errors !== undefined\n      || response.message !== undefined\n    )\n    const nonStatusKeys = responseKeys.filter((key) => key !== \"status\")\n    const configuredAttributeNames = this.configuredFrontendModelAttributeNames()\n    const looksLikeRawModelPayload = nonStatusKeys.length > 0\n      && nonStatusKeys.every((key) => configuredAttributeNames.has(key))\n\n    if (!hasErrorMessage && !hasOnlyStatus && !hasErrorEnvelopeKeys && looksLikeRawModelPayload) return\n\n    const errorMessage = hasErrorMessage\n      ? response.errorMessage\n      : `Request failed for ${this.name}#${commandType}`\n\n    throw new Error(errorMessage)\n  }\n\n  /**\n   * @this {typeof FrontendModelBase}\n   * @returns {Set<string>} - Configured frontend model attribute names.\n   */\n  static configuredFrontendModelAttributeNames() {\n    const resourceConfig = /** @type {Record<string, any>} */ (this.resourceConfig())\n    const attributes = resourceConfig.attributes\n\n    if (Array.isArray(attributes)) {\n      return new Set(attributes.filter((attributeName) => typeof attributeName === \"string\"))\n    }\n\n    if (attributes && typeof attributes === \"object\") {\n      return new Set(Object.keys(attributes))\n    }\n\n    return new Set()\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"get-route.d.ts","sourceRoot":"","sources":["../../../src/routes/get-route.js"],"names":[],"mappings":";AAOA;IACE;;;OAGG;IACH,mCAFG;QAAqB,IAAI,EAAjB,MAAM;KAChB,EAMA;IAFC,aAAgB;IAChB,eAA+D;CA8BlE;sBA1CqB,iBAAiB"}
1
+ {"version":3,"file":"get-route.d.ts","sourceRoot":"","sources":["../../../src/routes/get-route.js"],"names":[],"mappings":";AA4BA;IACE;;;OAGG;IACH,mCAFG;QAAqB,IAAI,EAAjB,MAAM;KAChB,EAMA;IAFC,aAAgB;IAChB,eAA+D;CA8BlE;sBA/DqB,iBAAiB"}
@@ -2,6 +2,23 @@
2
2
  import escapeStringRegexp from "escape-string-regexp";
3
3
  import BaseRoute from "./base-route.js";
4
4
  import restArgsError from "../utils/rest-args-error.js";
5
+ /**
6
+ * @param {Record<string, any>} params - Route params object.
7
+ * @param {string} name - Route name.
8
+ * @returns {void} - No return value.
9
+ */
10
+ function assignActionAndController(params, name) {
11
+ const segments = name.split("/").filter((segment) => segment.length > 0);
12
+ if (segments.length <= 1) {
13
+ params.action = name;
14
+ return;
15
+ }
16
+ const actionSegment = segments[segments.length - 1];
17
+ const controllerSuffix = segments.slice(0, -1).join("/");
18
+ const existingController = typeof params.controller === "string" && params.controller.length > 0 ? params.controller : null;
19
+ params.action = actionSegment;
20
+ params.controller = existingController ? `${existingController}/${controllerSuffix}` : controllerSuffix;
21
+ }
5
22
  class VelociousRouteGetRoute extends BaseRoute {
6
23
  /**
7
24
  * @param {object} args - Options object.
@@ -32,11 +49,11 @@ class VelociousRouteGetRoute extends BaseRoute {
32
49
  // Prevent partial prefix matches (e.g., "params" matching "params-with-query")
33
50
  if (restPath && !restPath.startsWith("/"))
34
51
  return;
35
- params.action = this.name;
52
+ assignActionAndController(params, this.name);
36
53
  return { restPath };
37
54
  }
38
55
  }
39
56
  }
40
57
  BaseRoute.registerRouteGetType(VelociousRouteGetRoute);
41
58
  export default VelociousRouteGetRoute;
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LXJvdXRlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3JvdXRlcy9nZXQtcm91dGUuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsWUFBWTtBQUVaLE9BQU8sa0JBQWtCLE1BQU0sc0JBQXNCLENBQUE7QUFFckQsT0FBTyxTQUFTLE1BQU0saUJBQWlCLENBQUE7QUFDdkMsT0FBTyxhQUFhLE1BQU0sNkJBQTZCLENBQUE7QUFFdkQsTUFBTSxzQkFBdUIsU0FBUSxTQUFTO0lBQzVDOzs7T0FHRztJQUNILFlBQVksRUFBQyxJQUFJLEVBQUUsR0FBRyxRQUFRLEVBQUM7UUFDN0IsS0FBSyxFQUFFLENBQUE7UUFDUCxhQUFhLENBQUMsUUFBUSxDQUFDLENBQUE7UUFDdkIsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUE7UUFDaEIsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxLQUFLLGtCQUFrQixDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQTtJQUNqRSxDQUFDO0lBRUQsYUFBYTtRQUNYLE9BQU87WUFDTCxFQUFDLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUM7U0FDcEQsQ0FBQTtJQUNILENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxhQUFhLENBQUMsRUFBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBQztRQUNuQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUVyQyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsTUFBTSxDQUFDLGNBQWMsRUFBRSxZQUFZLEVBQUUsUUFBUSxDQUFDLEdBQUcsS0FBSyxDQUFBLENBQUMscUNBQXFDO1lBRTVGLCtFQUErRTtZQUMvRSxJQUFJLFFBQVEsSUFBSSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDO2dCQUFFLE9BQU07WUFFakQsTUFBTSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFBO1lBRXpCLE9BQU8sRUFBQyxRQUFRLEVBQUMsQ0FBQTtRQUNuQixDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBRUQsU0FBUyxDQUFDLG9CQUFvQixDQUFDLHNCQUFzQixDQUFDLENBQUE7QUFFdEQsZUFBZSxzQkFBc0IsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQgZXNjYXBlU3RyaW5nUmVnZXhwIGZyb20gXCJlc2NhcGUtc3RyaW5nLXJlZ2V4cFwiXG5cbmltcG9ydCBCYXNlUm91dGUgZnJvbSBcIi4vYmFzZS1yb3V0ZS5qc1wiXG5pbXBvcnQgcmVzdEFyZ3NFcnJvciBmcm9tIFwiLi4vdXRpbHMvcmVzdC1hcmdzLWVycm9yLmpzXCJcblxuY2xhc3MgVmVsb2Npb3VzUm91dGVHZXRSb3V0ZSBleHRlbmRzIEJhc2VSb3V0ZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gYXJncy5uYW1lIC0gTmFtZS5cbiAgICovXG4gIGNvbnN0cnVjdG9yKHtuYW1lLCAuLi5yZXN0QXJnc30pIHtcbiAgICBzdXBlcigpXG4gICAgcmVzdEFyZ3NFcnJvcihyZXN0QXJncylcbiAgICB0aGlzLm5hbWUgPSBuYW1lXG4gICAgdGhpcy5yZWdFeHAgPSBuZXcgUmVnRXhwKGBeKCR7ZXNjYXBlU3RyaW5nUmVnZXhwKG5hbWUpfSkoLiopJGApXG4gIH1cblxuICBnZXRIdW1hblBhdGhzKCkge1xuICAgIHJldHVybiBbXG4gICAgICB7bWV0aG9kOiBcIkdFVFwiLCBhY3Rpb246IHRoaXMubmFtZSwgcGF0aDogdGhpcy5uYW1lfVxuICAgIF1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IGFyZ3MucGFyYW1zIC0gUGFyYW1ldGVycyBvYmplY3QuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLnBhdGggLSBQYXRoLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2h0dHAtc2VydmVyL2NsaWVudC9yZXF1ZXN0LmpzXCIpLmRlZmF1bHQgfCBpbXBvcnQoXCIuLi9odHRwLXNlcnZlci9jbGllbnQvd2Vic29ja2V0LXJlcXVlc3QuanNcIikuZGVmYXVsdH0gYXJncy5yZXF1ZXN0IC0gUmVxdWVzdCBvYmplY3QuXG4gICAqIEByZXR1cm5zIHt7cmVzdFBhdGg6IHN0cmluZ30gfCB1bmRlZmluZWR9IC0gUkVTVCBwYXRoIG1ldGFkYXRhIGZvciB0aGlzIHJvdXRlLlxuICAgKi9cbiAgbWF0Y2hXaXRoUGF0aCh7cGFyYW1zLCBwYXRoLCByZXF1ZXN0fSkgeyAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIG5vLXVudXNlZC12YXJzXG4gICAgY29uc3QgbWF0Y2ggPSBwYXRoLm1hdGNoKHRoaXMucmVnRXhwKVxuXG4gICAgaWYgKG1hdGNoKSB7XG4gICAgICBjb25zdCBbX2JlZ2lubmlnU2xhc2gsIF9tYXRjaGVkTmFtZSwgcmVzdFBhdGhdID0gbWF0Y2ggLy8gZXNsaW50LWRpc2FibGUtbGluZSBuby11bnVzZWQtdmFyc1xuXG4gICAgICAvLyBQcmV2ZW50IHBhcnRpYWwgcHJlZml4IG1hdGNoZXMgKGUuZy4sIFwicGFyYW1zXCIgbWF0Y2hpbmcgXCJwYXJhbXMtd2l0aC1xdWVyeVwiKVxuICAgICAgaWYgKHJlc3RQYXRoICYmICFyZXN0UGF0aC5zdGFydHNXaXRoKFwiL1wiKSkgcmV0dXJuXG5cbiAgICAgIHBhcmFtcy5hY3Rpb24gPSB0aGlzLm5hbWVcblxuICAgICAgcmV0dXJuIHtyZXN0UGF0aH1cbiAgICB9XG4gIH1cbn1cblxuQmFzZVJvdXRlLnJlZ2lzdGVyUm91dGVHZXRUeXBlKFZlbG9jaW91c1JvdXRlR2V0Um91dGUpXG5cbmV4cG9ydCBkZWZhdWx0IFZlbG9jaW91c1JvdXRlR2V0Um91dGVcblxuIl19
59
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LXJvdXRlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3JvdXRlcy9nZXQtcm91dGUuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsWUFBWTtBQUVaLE9BQU8sa0JBQWtCLE1BQU0sc0JBQXNCLENBQUE7QUFFckQsT0FBTyxTQUFTLE1BQU0saUJBQWlCLENBQUE7QUFDdkMsT0FBTyxhQUFhLE1BQU0sNkJBQTZCLENBQUE7QUFFdkQ7Ozs7R0FJRztBQUNILFNBQVMseUJBQXlCLENBQUMsTUFBTSxFQUFFLElBQUk7SUFDN0MsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUE7SUFFeEUsSUFBSSxRQUFRLENBQUMsTUFBTSxJQUFJLENBQUMsRUFBRSxDQUFDO1FBQ3pCLE1BQU0sQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFBO1FBQ3BCLE9BQU07SUFDUixDQUFDO0lBRUQsTUFBTSxhQUFhLEdBQUcsUUFBUSxDQUFDLFFBQVEsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUE7SUFDbkQsTUFBTSxnQkFBZ0IsR0FBRyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQTtJQUN4RCxNQUFNLGtCQUFrQixHQUFHLE9BQU8sTUFBTSxDQUFDLFVBQVUsS0FBSyxRQUFRLElBQUksTUFBTSxDQUFDLFVBQVUsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUE7SUFFM0gsTUFBTSxDQUFDLE1BQU0sR0FBRyxhQUFhLENBQUE7SUFDN0IsTUFBTSxDQUFDLFVBQVUsR0FBRyxrQkFBa0IsQ0FBQyxDQUFDLENBQUMsR0FBRyxrQkFBa0IsSUFBSSxnQkFBZ0IsRUFBRSxDQUFDLENBQUMsQ0FBQyxnQkFBZ0IsQ0FBQTtBQUN6RyxDQUFDO0FBRUQsTUFBTSxzQkFBdUIsU0FBUSxTQUFTO0lBQzVDOzs7T0FHRztJQUNILFlBQVksRUFBQyxJQUFJLEVBQUUsR0FBRyxRQUFRLEVBQUM7UUFDN0IsS0FBSyxFQUFFLENBQUE7UUFDUCxhQUFhLENBQUMsUUFBUSxDQUFDLENBQUE7UUFDdkIsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUE7UUFDaEIsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLE1BQU0sQ0FBQyxLQUFLLGtCQUFrQixDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQTtJQUNqRSxDQUFDO0lBRUQsYUFBYTtRQUNYLE9BQU87WUFDTCxFQUFDLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUM7U0FDcEQsQ0FBQTtJQUNILENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxhQUFhLENBQUMsRUFBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBQztRQUNuQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUVyQyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsTUFBTSxDQUFDLGNBQWMsRUFBRSxZQUFZLEVBQUUsUUFBUSxDQUFDLEdBQUcsS0FBSyxDQUFBLENBQUMscUNBQXFDO1lBRTVGLCtFQUErRTtZQUMvRSxJQUFJLFFBQVEsSUFBSSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDO2dCQUFFLE9BQU07WUFFakQseUJBQXlCLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQTtZQUU1QyxPQUFPLEVBQUMsUUFBUSxFQUFDLENBQUE7UUFDbkIsQ0FBQztJQUNILENBQUM7Q0FDRjtBQUVELFNBQVMsQ0FBQyxvQkFBb0IsQ0FBQyxzQkFBc0IsQ0FBQyxDQUFBO0FBRXRELGVBQWUsc0JBQXNCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IGVzY2FwZVN0cmluZ1JlZ2V4cCBmcm9tIFwiZXNjYXBlLXN0cmluZy1yZWdleHBcIlxuXG5pbXBvcnQgQmFzZVJvdXRlIGZyb20gXCIuL2Jhc2Utcm91dGUuanNcIlxuaW1wb3J0IHJlc3RBcmdzRXJyb3IgZnJvbSBcIi4uL3V0aWxzL3Jlc3QtYXJncy1lcnJvci5qc1wiXG5cbi8qKlxuICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBwYXJhbXMgLSBSb3V0ZSBwYXJhbXMgb2JqZWN0LlxuICogQHBhcmFtIHtzdHJpbmd9IG5hbWUgLSBSb3V0ZSBuYW1lLlxuICogQHJldHVybnMge3ZvaWR9IC0gTm8gcmV0dXJuIHZhbHVlLlxuICovXG5mdW5jdGlvbiBhc3NpZ25BY3Rpb25BbmRDb250cm9sbGVyKHBhcmFtcywgbmFtZSkge1xuICBjb25zdCBzZWdtZW50cyA9IG5hbWUuc3BsaXQoXCIvXCIpLmZpbHRlcigoc2VnbWVudCkgPT4gc2VnbWVudC5sZW5ndGggPiAwKVxuXG4gIGlmIChzZWdtZW50cy5sZW5ndGggPD0gMSkge1xuICAgIHBhcmFtcy5hY3Rpb24gPSBuYW1lXG4gICAgcmV0dXJuXG4gIH1cblxuICBjb25zdCBhY3Rpb25TZWdtZW50ID0gc2VnbWVudHNbc2VnbWVudHMubGVuZ3RoIC0gMV1cbiAgY29uc3QgY29udHJvbGxlclN1ZmZpeCA9IHNlZ21lbnRzLnNsaWNlKDAsIC0xKS5qb2luKFwiL1wiKVxuICBjb25zdCBleGlzdGluZ0NvbnRyb2xsZXIgPSB0eXBlb2YgcGFyYW1zLmNvbnRyb2xsZXIgPT09IFwic3RyaW5nXCIgJiYgcGFyYW1zLmNvbnRyb2xsZXIubGVuZ3RoID4gMCA/IHBhcmFtcy5jb250cm9sbGVyIDogbnVsbFxuXG4gIHBhcmFtcy5hY3Rpb24gPSBhY3Rpb25TZWdtZW50XG4gIHBhcmFtcy5jb250cm9sbGVyID0gZXhpc3RpbmdDb250cm9sbGVyID8gYCR7ZXhpc3RpbmdDb250cm9sbGVyfS8ke2NvbnRyb2xsZXJTdWZmaXh9YCA6IGNvbnRyb2xsZXJTdWZmaXhcbn1cblxuY2xhc3MgVmVsb2Npb3VzUm91dGVHZXRSb3V0ZSBleHRlbmRzIEJhc2VSb3V0ZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gYXJncy5uYW1lIC0gTmFtZS5cbiAgICovXG4gIGNvbnN0cnVjdG9yKHtuYW1lLCAuLi5yZXN0QXJnc30pIHtcbiAgICBzdXBlcigpXG4gICAgcmVzdEFyZ3NFcnJvcihyZXN0QXJncylcbiAgICB0aGlzLm5hbWUgPSBuYW1lXG4gICAgdGhpcy5yZWdFeHAgPSBuZXcgUmVnRXhwKGBeKCR7ZXNjYXBlU3RyaW5nUmVnZXhwKG5hbWUpfSkoLiopJGApXG4gIH1cblxuICBnZXRIdW1hblBhdGhzKCkge1xuICAgIHJldHVybiBbXG4gICAgICB7bWV0aG9kOiBcIkdFVFwiLCBhY3Rpb246IHRoaXMubmFtZSwgcGF0aDogdGhpcy5uYW1lfVxuICAgIF1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IGFyZ3MucGFyYW1zIC0gUGFyYW1ldGVycyBvYmplY3QuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLnBhdGggLSBQYXRoLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2h0dHAtc2VydmVyL2NsaWVudC9yZXF1ZXN0LmpzXCIpLmRlZmF1bHQgfCBpbXBvcnQoXCIuLi9odHRwLXNlcnZlci9jbGllbnQvd2Vic29ja2V0LXJlcXVlc3QuanNcIikuZGVmYXVsdH0gYXJncy5yZXF1ZXN0IC0gUmVxdWVzdCBvYmplY3QuXG4gICAqIEByZXR1cm5zIHt7cmVzdFBhdGg6IHN0cmluZ30gfCB1bmRlZmluZWR9IC0gUkVTVCBwYXRoIG1ldGFkYXRhIGZvciB0aGlzIHJvdXRlLlxuICAgKi9cbiAgbWF0Y2hXaXRoUGF0aCh7cGFyYW1zLCBwYXRoLCByZXF1ZXN0fSkgeyAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIG5vLXVudXNlZC12YXJzXG4gICAgY29uc3QgbWF0Y2ggPSBwYXRoLm1hdGNoKHRoaXMucmVnRXhwKVxuXG4gICAgaWYgKG1hdGNoKSB7XG4gICAgICBjb25zdCBbX2JlZ2lubmlnU2xhc2gsIF9tYXRjaGVkTmFtZSwgcmVzdFBhdGhdID0gbWF0Y2ggLy8gZXNsaW50LWRpc2FibGUtbGluZSBuby11bnVzZWQtdmFyc1xuXG4gICAgICAvLyBQcmV2ZW50IHBhcnRpYWwgcHJlZml4IG1hdGNoZXMgKGUuZy4sIFwicGFyYW1zXCIgbWF0Y2hpbmcgXCJwYXJhbXMtd2l0aC1xdWVyeVwiKVxuICAgICAgaWYgKHJlc3RQYXRoICYmICFyZXN0UGF0aC5zdGFydHNXaXRoKFwiL1wiKSkgcmV0dXJuXG5cbiAgICAgIGFzc2lnbkFjdGlvbkFuZENvbnRyb2xsZXIocGFyYW1zLCB0aGlzLm5hbWUpXG5cbiAgICAgIHJldHVybiB7cmVzdFBhdGh9XG4gICAgfVxuICB9XG59XG5cbkJhc2VSb3V0ZS5yZWdpc3RlclJvdXRlR2V0VHlwZShWZWxvY2lvdXNSb3V0ZUdldFJvdXRlKVxuXG5leHBvcnQgZGVmYXVsdCBWZWxvY2lvdXNSb3V0ZUdldFJvdXRlXG4iXX0=
@@ -1 +1 @@
1
- {"version":3,"file":"post-route.d.ts","sourceRoot":"","sources":["../../../src/routes/post-route.js"],"names":[],"mappings":";AAMA;IACE;;;OAGG;IACH,mCAFG;QAAqB,IAAI,EAAjB,MAAM;KAChB,EAMA;IAFC,aAAgB;IAChB,eAA+D;CA2BlE;sBAxCqB,iBAAiB"}
1
+ {"version":3,"file":"post-route.d.ts","sourceRoot":"","sources":["../../../src/routes/post-route.js"],"names":[],"mappings":";AA2BA;IACE;;;OAGG;IACH,mCAFG;QAAqB,IAAI,EAAjB,MAAM;KAChB,EAMA;IAFC,aAAgB;IAChB,eAA+D;CA8BlE;sBAhEqB,iBAAiB"}
@@ -2,6 +2,23 @@
2
2
  import BaseRoute from "./base-route.js";
3
3
  import escapeStringRegexp from "escape-string-regexp";
4
4
  import restArgsError from "../utils/rest-args-error.js";
5
+ /**
6
+ * @param {Record<string, any>} params - Route params object.
7
+ * @param {string} name - Route name.
8
+ * @returns {void} - No return value.
9
+ */
10
+ function assignActionAndController(params, name) {
11
+ const segments = name.split("/").filter((segment) => segment.length > 0);
12
+ if (segments.length <= 1) {
13
+ params.action = name;
14
+ return;
15
+ }
16
+ const actionSegment = segments[segments.length - 1];
17
+ const controllerSuffix = segments.slice(0, -1).join("/");
18
+ const existingController = typeof params.controller === "string" && params.controller.length > 0 ? params.controller : null;
19
+ params.action = actionSegment;
20
+ params.controller = existingController ? `${existingController}/${controllerSuffix}` : controllerSuffix;
21
+ }
5
22
  class VelociousRoutePostRoute extends BaseRoute {
6
23
  /**
7
24
  * @param {object} args - Options object.
@@ -30,11 +47,14 @@ class VelociousRoutePostRoute extends BaseRoute {
30
47
  const match = path.match(this.regExp);
31
48
  if (match) {
32
49
  const [_beginnigSlash, _matchedName, restPath] = match; // eslint-disable-line no-unused-vars
33
- params.action = this.name;
50
+ // Prevent partial prefix matches (e.g., "update" matching "update-password")
51
+ if (restPath && !restPath.startsWith("/"))
52
+ return;
53
+ assignActionAndController(params, this.name);
34
54
  return { restPath };
35
55
  }
36
56
  }
37
57
  }
38
58
  BaseRoute.registerRoutePostType(VelociousRoutePostRoute);
39
59
  export default VelociousRoutePostRoute;
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9zdC1yb3V0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9yb3V0ZXMvcG9zdC1yb3V0ZS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosT0FBTyxTQUFTLE1BQU0saUJBQWlCLENBQUE7QUFDdkMsT0FBTyxrQkFBa0IsTUFBTSxzQkFBc0IsQ0FBQTtBQUNyRCxPQUFPLGFBQWEsTUFBTSw2QkFBNkIsQ0FBQTtBQUV2RCxNQUFNLHVCQUF3QixTQUFRLFNBQVM7SUFDN0M7OztPQUdHO0lBQ0gsWUFBWSxFQUFDLElBQUksRUFBRSxHQUFHLFFBQVEsRUFBQztRQUM3QixLQUFLLEVBQUUsQ0FBQTtRQUNQLGFBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQTtRQUN2QixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQTtRQUNoQixJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLEtBQUssa0JBQWtCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO0lBQ2pFLENBQUM7SUFFRCxhQUFhO1FBQ1gsT0FBTyxDQUFDO2dCQUNOLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJO2FBQUM7U0FDcEQsQ0FBQTtJQUNILENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxhQUFhLENBQUMsRUFBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBQztRQUNuQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUVyQyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsTUFBTSxDQUFDLGNBQWMsRUFBRSxZQUFZLEVBQUUsUUFBUSxDQUFDLEdBQUcsS0FBSyxDQUFBLENBQUMscUNBQXFDO1lBRTVGLE1BQU0sQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQTtZQUV6QixPQUFPLEVBQUMsUUFBUSxFQUFDLENBQUE7UUFDbkIsQ0FBQztJQUNILENBQUM7Q0FDRjtBQUVELFNBQVMsQ0FBQyxxQkFBcUIsQ0FBQyx1QkFBdUIsQ0FBQyxDQUFBO0FBRXhELGVBQWUsdUJBQXVCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IEJhc2VSb3V0ZSBmcm9tIFwiLi9iYXNlLXJvdXRlLmpzXCJcbmltcG9ydCBlc2NhcGVTdHJpbmdSZWdleHAgZnJvbSBcImVzY2FwZS1zdHJpbmctcmVnZXhwXCJcbmltcG9ydCByZXN0QXJnc0Vycm9yIGZyb20gXCIuLi91dGlscy9yZXN0LWFyZ3MtZXJyb3IuanNcIlxuXG5jbGFzcyBWZWxvY2lvdXNSb3V0ZVBvc3RSb3V0ZSBleHRlbmRzIEJhc2VSb3V0ZSB7XG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gYXJncy5uYW1lIC0gTmFtZS5cbiAgICovXG4gIGNvbnN0cnVjdG9yKHtuYW1lLCAuLi5yZXN0QXJnc30pIHtcbiAgICBzdXBlcigpXG4gICAgcmVzdEFyZ3NFcnJvcihyZXN0QXJncylcbiAgICB0aGlzLm5hbWUgPSBuYW1lXG4gICAgdGhpcy5yZWdFeHAgPSBuZXcgUmVnRXhwKGBeKCR7ZXNjYXBlU3RyaW5nUmVnZXhwKG5hbWUpfSkoLiopJGApXG4gIH1cblxuICBnZXRIdW1hblBhdGhzKCkge1xuICAgIHJldHVybiBbe1xuICAgICAgbWV0aG9kOiBcIlBPU1RcIiwgYWN0aW9uOiB0aGlzLm5hbWUsIHBhdGg6IHRoaXMubmFtZX1cbiAgICBdXG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHtvYmplY3R9IGFyZ3MgLSBPcHRpb25zIG9iamVjdC5cbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+fSBhcmdzLnBhcmFtcyAtIFBhcmFtZXRlcnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge3N0cmluZ30gYXJncy5wYXRoIC0gUGF0aC5cbiAgICogQHBhcmFtIHtpbXBvcnQoXCIuLi9odHRwLXNlcnZlci9jbGllbnQvcmVxdWVzdC5qc1wiKS5kZWZhdWx0IHwgaW1wb3J0KFwiLi4vaHR0cC1zZXJ2ZXIvY2xpZW50L3dlYnNvY2tldC1yZXF1ZXN0LmpzXCIpLmRlZmF1bHR9IGFyZ3MucmVxdWVzdCAtIFJlcXVlc3Qgb2JqZWN0LlxuICAgKiBAcmV0dXJucyB7e3Jlc3RQYXRoOiBzdHJpbmd9IHwgdW5kZWZpbmVkfSAtIFJFU1QgcGF0aCBtZXRhZGF0YSBmb3IgdGhpcyByb3V0ZS5cbiAgICovXG4gIG1hdGNoV2l0aFBhdGgoe3BhcmFtcywgcGF0aCwgcmVxdWVzdH0pIHsgLy8gZXNsaW50LWRpc2FibGUtbGluZSBuby11bnVzZWQtdmFyc1xuICAgIGNvbnN0IG1hdGNoID0gcGF0aC5tYXRjaCh0aGlzLnJlZ0V4cClcblxuICAgIGlmIChtYXRjaCkge1xuICAgICAgY29uc3QgW19iZWdpbm5pZ1NsYXNoLCBfbWF0Y2hlZE5hbWUsIHJlc3RQYXRoXSA9IG1hdGNoIC8vIGVzbGludC1kaXNhYmxlLWxpbmUgbm8tdW51c2VkLXZhcnNcblxuICAgICAgcGFyYW1zLmFjdGlvbiA9IHRoaXMubmFtZVxuXG4gICAgICByZXR1cm4ge3Jlc3RQYXRofVxuICAgIH1cbiAgfVxufVxuXG5CYXNlUm91dGUucmVnaXN0ZXJSb3V0ZVBvc3RUeXBlKFZlbG9jaW91c1JvdXRlUG9zdFJvdXRlKVxuXG5leHBvcnQgZGVmYXVsdCBWZWxvY2lvdXNSb3V0ZVBvc3RSb3V0ZVxuXG4iXX0=
60
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9zdC1yb3V0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9yb3V0ZXMvcG9zdC1yb3V0ZS5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosT0FBTyxTQUFTLE1BQU0saUJBQWlCLENBQUE7QUFDdkMsT0FBTyxrQkFBa0IsTUFBTSxzQkFBc0IsQ0FBQTtBQUNyRCxPQUFPLGFBQWEsTUFBTSw2QkFBNkIsQ0FBQTtBQUV2RDs7OztHQUlHO0FBQ0gsU0FBUyx5QkFBeUIsQ0FBQyxNQUFNLEVBQUUsSUFBSTtJQUM3QyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsT0FBTyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQTtJQUV4RSxJQUFJLFFBQVEsQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFLENBQUM7UUFDekIsTUFBTSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUE7UUFDcEIsT0FBTTtJQUNSLENBQUM7SUFFRCxNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsUUFBUSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQTtJQUNuRCxNQUFNLGdCQUFnQixHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFBO0lBQ3hELE1BQU0sa0JBQWtCLEdBQUcsT0FBTyxNQUFNLENBQUMsVUFBVSxLQUFLLFFBQVEsSUFBSSxNQUFNLENBQUMsVUFBVSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQTtJQUUzSCxNQUFNLENBQUMsTUFBTSxHQUFHLGFBQWEsQ0FBQTtJQUM3QixNQUFNLENBQUMsVUFBVSxHQUFHLGtCQUFrQixDQUFDLENBQUMsQ0FBQyxHQUFHLGtCQUFrQixJQUFJLGdCQUFnQixFQUFFLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixDQUFBO0FBQ3pHLENBQUM7QUFFRCxNQUFNLHVCQUF3QixTQUFRLFNBQVM7SUFDN0M7OztPQUdHO0lBQ0gsWUFBWSxFQUFDLElBQUksRUFBRSxHQUFHLFFBQVEsRUFBQztRQUM3QixLQUFLLEVBQUUsQ0FBQTtRQUNQLGFBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQTtRQUN2QixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQTtRQUNoQixJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLEtBQUssa0JBQWtCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO0lBQ2pFLENBQUM7SUFFRCxhQUFhO1FBQ1gsT0FBTyxDQUFDO2dCQUNOLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJO2FBQUM7U0FDcEQsQ0FBQTtJQUNILENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxhQUFhLENBQUMsRUFBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBQztRQUNuQyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUVyQyxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsTUFBTSxDQUFDLGNBQWMsRUFBRSxZQUFZLEVBQUUsUUFBUSxDQUFDLEdBQUcsS0FBSyxDQUFBLENBQUMscUNBQXFDO1lBRTVGLDZFQUE2RTtZQUM3RSxJQUFJLFFBQVEsSUFBSSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDO2dCQUFFLE9BQU07WUFFakQseUJBQXlCLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQTtZQUU1QyxPQUFPLEVBQUMsUUFBUSxFQUFDLENBQUE7UUFDbkIsQ0FBQztJQUNILENBQUM7Q0FDRjtBQUVELFNBQVMsQ0FBQyxxQkFBcUIsQ0FBQyx1QkFBdUIsQ0FBQyxDQUFBO0FBRXhELGVBQWUsdUJBQXVCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBAdHMtY2hlY2tcblxuaW1wb3J0IEJhc2VSb3V0ZSBmcm9tIFwiLi9iYXNlLXJvdXRlLmpzXCJcbmltcG9ydCBlc2NhcGVTdHJpbmdSZWdleHAgZnJvbSBcImVzY2FwZS1zdHJpbmctcmVnZXhwXCJcbmltcG9ydCByZXN0QXJnc0Vycm9yIGZyb20gXCIuLi91dGlscy9yZXN0LWFyZ3MtZXJyb3IuanNcIlxuXG4vKipcbiAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55Pn0gcGFyYW1zIC0gUm91dGUgcGFyYW1zIG9iamVjdC5cbiAqIEBwYXJhbSB7c3RyaW5nfSBuYW1lIC0gUm91dGUgbmFtZS5cbiAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAqL1xuZnVuY3Rpb24gYXNzaWduQWN0aW9uQW5kQ29udHJvbGxlcihwYXJhbXMsIG5hbWUpIHtcbiAgY29uc3Qgc2VnbWVudHMgPSBuYW1lLnNwbGl0KFwiL1wiKS5maWx0ZXIoKHNlZ21lbnQpID0+IHNlZ21lbnQubGVuZ3RoID4gMClcblxuICBpZiAoc2VnbWVudHMubGVuZ3RoIDw9IDEpIHtcbiAgICBwYXJhbXMuYWN0aW9uID0gbmFtZVxuICAgIHJldHVyblxuICB9XG5cbiAgY29uc3QgYWN0aW9uU2VnbWVudCA9IHNlZ21lbnRzW3NlZ21lbnRzLmxlbmd0aCAtIDFdXG4gIGNvbnN0IGNvbnRyb2xsZXJTdWZmaXggPSBzZWdtZW50cy5zbGljZSgwLCAtMSkuam9pbihcIi9cIilcbiAgY29uc3QgZXhpc3RpbmdDb250cm9sbGVyID0gdHlwZW9mIHBhcmFtcy5jb250cm9sbGVyID09PSBcInN0cmluZ1wiICYmIHBhcmFtcy5jb250cm9sbGVyLmxlbmd0aCA+IDAgPyBwYXJhbXMuY29udHJvbGxlciA6IG51bGxcblxuICBwYXJhbXMuYWN0aW9uID0gYWN0aW9uU2VnbWVudFxuICBwYXJhbXMuY29udHJvbGxlciA9IGV4aXN0aW5nQ29udHJvbGxlciA/IGAke2V4aXN0aW5nQ29udHJvbGxlcn0vJHtjb250cm9sbGVyU3VmZml4fWAgOiBjb250cm9sbGVyU3VmZml4XG59XG5cbmNsYXNzIFZlbG9jaW91c1JvdXRlUG9zdFJvdXRlIGV4dGVuZHMgQmFzZVJvdXRlIHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7b2JqZWN0fSBhcmdzIC0gT3B0aW9ucyBvYmplY3QuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLm5hbWUgLSBOYW1lLlxuICAgKi9cbiAgY29uc3RydWN0b3Ioe25hbWUsIC4uLnJlc3RBcmdzfSkge1xuICAgIHN1cGVyKClcbiAgICByZXN0QXJnc0Vycm9yKHJlc3RBcmdzKVxuICAgIHRoaXMubmFtZSA9IG5hbWVcbiAgICB0aGlzLnJlZ0V4cCA9IG5ldyBSZWdFeHAoYF4oJHtlc2NhcGVTdHJpbmdSZWdleHAobmFtZSl9KSguKikkYClcbiAgfVxuXG4gIGdldEh1bWFuUGF0aHMoKSB7XG4gICAgcmV0dXJuIFt7XG4gICAgICBtZXRob2Q6IFwiUE9TVFwiLCBhY3Rpb246IHRoaXMubmFtZSwgcGF0aDogdGhpcy5uYW1lfVxuICAgIF1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge29iamVjdH0gYXJncyAtIE9wdGlvbnMgb2JqZWN0LlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IGFyZ3MucGFyYW1zIC0gUGFyYW1ldGVycyBvYmplY3QuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLnBhdGggLSBQYXRoLlxuICAgKiBAcGFyYW0ge2ltcG9ydChcIi4uL2h0dHAtc2VydmVyL2NsaWVudC9yZXF1ZXN0LmpzXCIpLmRlZmF1bHQgfCBpbXBvcnQoXCIuLi9odHRwLXNlcnZlci9jbGllbnQvd2Vic29ja2V0LXJlcXVlc3QuanNcIikuZGVmYXVsdH0gYXJncy5yZXF1ZXN0IC0gUmVxdWVzdCBvYmplY3QuXG4gICAqIEByZXR1cm5zIHt7cmVzdFBhdGg6IHN0cmluZ30gfCB1bmRlZmluZWR9IC0gUkVTVCBwYXRoIG1ldGFkYXRhIGZvciB0aGlzIHJvdXRlLlxuICAgKi9cbiAgbWF0Y2hXaXRoUGF0aCh7cGFyYW1zLCBwYXRoLCByZXF1ZXN0fSkgeyAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIG5vLXVudXNlZC12YXJzXG4gICAgY29uc3QgbWF0Y2ggPSBwYXRoLm1hdGNoKHRoaXMucmVnRXhwKVxuXG4gICAgaWYgKG1hdGNoKSB7XG4gICAgICBjb25zdCBbX2JlZ2lubmlnU2xhc2gsIF9tYXRjaGVkTmFtZSwgcmVzdFBhdGhdID0gbWF0Y2ggLy8gZXNsaW50LWRpc2FibGUtbGluZSBuby11bnVzZWQtdmFyc1xuXG4gICAgICAvLyBQcmV2ZW50IHBhcnRpYWwgcHJlZml4IG1hdGNoZXMgKGUuZy4sIFwidXBkYXRlXCIgbWF0Y2hpbmcgXCJ1cGRhdGUtcGFzc3dvcmRcIilcbiAgICAgIGlmIChyZXN0UGF0aCAmJiAhcmVzdFBhdGguc3RhcnRzV2l0aChcIi9cIikpIHJldHVyblxuXG4gICAgICBhc3NpZ25BY3Rpb25BbmRDb250cm9sbGVyKHBhcmFtcywgdGhpcy5uYW1lKVxuXG4gICAgICByZXR1cm4ge3Jlc3RQYXRofVxuICAgIH1cbiAgfVxufVxuXG5CYXNlUm91dGUucmVnaXN0ZXJSb3V0ZVBvc3RUeXBlKFZlbG9jaW91c1JvdXRlUG9zdFJvdXRlKVxuXG5leHBvcnQgZGVmYXVsdCBWZWxvY2lvdXNSb3V0ZVBvc3RSb3V0ZVxuIl19
@@ -1 +1 @@
1
- {"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"AAWA;IAIE;;;;;OAKG;IACH,kDAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACmF,OAAO,EAA/H,OAAO,kCAAkC,EAAE,OAAO,GAAG,OAAO,4CAA4C,EAAE,OAAO;QACvD,QAAQ,EAAlE,OAAO,mCAAmC,EAAE,OAAO;KAC7D,EAcA;IAtBD,iCAAiC;IACjC,QADW,MAAM,GAAG,SAAS,CACvB;IAaJ,qDAAkC;IAGlC,YAAgC;IAGhC,2HAAsB;IACtB,8DAAwB;IAG1B,yBA6FC;IAED;;;;OAIG;IACH,2BAJW,OAAO,iBAAiB,EAAE,OAAO,QACjC,MAAM,GACJ;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAC,GAAG,SAAS,CAsB1C;IAED;;;OAGG;IACH,uCAHW,MAAM,GACJ,OAAO,CAAC,OAAO,2BAA2B,EAAE,uBAAuB,GAAG,IAAI,CAAC,CAqCvF;IAED;;;;;OAKG;IACH,6CAJG;QAAqB,MAAM,EAAnB,MAAM;QAC0C,eAAe,EAA/D,cAAc,kBAAkB,EAAE,OAAO;KACjD,GAAU,OAAO,CAAC,IAAI,CAAC,CAiBzB;IAED;;;OAGG;IACH,uBAHW,IAAI,GACF,MAAM,CAqBlB;IAED;;;OAGG;IACH,iCAHW,GAAG,GACD,GAAG,CA2Bf;CACF;mBA/QkB,cAAc"}
1
+ {"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"AAmBA;IAIE;;;;;OAKG;IACH,kDAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACmF,OAAO,EAA/H,OAAO,kCAAkC,EAAE,OAAO,GAAG,OAAO,4CAA4C,EAAE,OAAO;QACvD,QAAQ,EAAlE,OAAO,mCAAmC,EAAE,OAAO;KAC7D,EAcA;IAtBD,iCAAiC;IACjC,QADW,MAAM,GAAG,SAAS,CACvB;IAaJ,qDAAkC;IAGlC,YAAgC;IAGhC,2HAAsB;IACtB,8DAAwB;IAG1B,yBA6FC;IAED;;;;OAIG;IACH,2BAJW,OAAO,iBAAiB,EAAE,OAAO,QACjC,MAAM,GACJ;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAC,GAAG,SAAS,CAiC1C;IAED;;;OAGG;IACH,uCAHW,MAAM,GACJ,OAAO,CAAC,OAAO,2BAA2B,EAAE,uBAAuB,GAAG,IAAI,CAAC,CAqCvF;IAED;;;;;OAKG;IACH,6CAJG;QAAqB,MAAM,EAAnB,MAAM;QAC0C,eAAe,EAA/D,cAAc,kBAAkB,EAAE,OAAO;KACjD,GAAU,OAAO,CAAC,IAAI,CAAC,CAiBzB;IAED;;;OAGG;IACH,uBAHW,IAAI,GACF,MAAM,CAqBlB;IAED;;;OAGG;IACH,iCAHW,GAAG,GACD,GAAG,CA2Bf;CACF;mBAlSkB,cAAc"}
@@ -7,6 +7,13 @@ import Logger from "../logger.js";
7
7
  import UploadedFile from "../http-server/client/uploaded-file/uploaded-file.js";
8
8
  import ensureError from "../utils/ensure-error.js";
9
9
  import toImportSpecifier from "../utils/to-import-specifier.js";
10
+ /**
11
+ * @param {string} actionName - Raw action name from route params or route hook.
12
+ * @returns {string} - Normalized controller method name.
13
+ */
14
+ function normalizeActionName(actionName) {
15
+ return inflection.camelize(actionName.replaceAll("-", "_").replaceAll("/", "_"), true);
16
+ }
10
17
  export default class VelociousRoutesResolver {
11
18
  /** @type {Logger | undefined} */
12
19
  logger;
@@ -43,11 +50,11 @@ export default class VelociousRoutesResolver {
43
50
  const actionParam = this.params.action;
44
51
  const controllerParam = this.params.controller;
45
52
  const actionValue = typeof actionParam == "string" ? actionParam : (Array.isArray(actionParam) ? actionParam[0] : undefined);
46
- let action = typeof actionValue == "string" ? inflection.camelize(actionValue.replaceAll("-", "_"), true) : undefined;
53
+ let action = typeof actionValue == "string" ? normalizeActionName(actionValue) : undefined;
47
54
  let controller = typeof controllerParam == "string" ? controllerParam : (Array.isArray(controllerParam) ? controllerParam[0] : undefined);
48
55
  if (routeResolverHookMatch) {
49
56
  controller = routeResolverHookMatch.controller;
50
- action = inflection.camelize(routeResolverHookMatch.action.replaceAll("-", "_"), true);
57
+ action = normalizeActionName(routeResolverHookMatch.action);
51
58
  this.params.controller = controller;
52
59
  this.params.action = routeResolverHookMatch.action;
53
60
  controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`;
@@ -126,16 +133,24 @@ export default class VelociousRoutesResolver {
126
133
  matchPathWithRoutes(route, path) {
127
134
  const pathWithoutSlash = path.replace(/^\//, "").split("?")[0];
128
135
  for (const subRoute of route.routes) {
136
+ const paramsSnapshot = { ...this.params };
129
137
  const matchResult = subRoute.matchWithPath({
130
138
  params: this.params,
131
139
  path: pathWithoutSlash,
132
140
  request: this.request
133
141
  });
134
- if (!matchResult)
142
+ if (!matchResult) {
143
+ this.params = paramsSnapshot;
135
144
  continue;
145
+ }
136
146
  const { restPath } = matchResult;
137
147
  if (restPath) {
138
- return this.matchPathWithRoutes(subRoute, restPath);
148
+ const recursiveMatch = this.matchPathWithRoutes(subRoute, restPath);
149
+ if (recursiveMatch) {
150
+ return recursiveMatch;
151
+ }
152
+ this.params = paramsSnapshot;
153
+ continue;
139
154
  }
140
155
  return matchResult;
141
156
  }
@@ -241,4 +256,4 @@ export default class VelociousRoutesResolver {
241
256
  return value;
242
257
  }
243
258
  }
244
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport Logger from \"../logger.js\"\nimport UploadedFile from \"../http-server/client/uploaded-file/uploaded-file.js\"\nimport ensureError from \"../utils/ensure-error.js\"\nimport toImportSpecifier from \"../utils/to-import-specifier.js\"\n\nexport default class VelociousRoutesResolver {\n  /** @type {Logger | undefined} */\n  logger\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"../http-server/client/request.js\").default | import(\"../http-server/client/websocket-request.js\").default} args.request - Request object.\n   * @param {import(\"../http-server/client/response.js\").default} args.response - Response object.\n   */\n  constructor({configuration, request, response}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n\n    this.configuration = configuration\n    this.logger = new Logger(\"RoutesResolver\", {configuration})\n    const requestParams = request.params() || {}\n    this.params = {...requestParams}\n    delete this.params.action\n    delete this.params.controller\n    this.request = request\n    this.response = response\n  }\n\n  async resolve() {\n    let controllerPath\n    let currentRoute = this.configuration.routes.rootRoute\n    const rawPath = this.request.path()\n    const currentPath = rawPath.split(\"?\")[0]\n    let viewPath\n\n    const routeResolverHookMatch = await this.resolveRouteResolverHooks(currentPath)\n    const matchResult = routeResolverHookMatch ? undefined : this.matchPathWithRoutes(currentRoute, currentPath)\n    const actionParam = this.params.action\n    const controllerParam = this.params.controller\n    const actionValue = typeof actionParam == \"string\" ? actionParam : (Array.isArray(actionParam) ? actionParam[0] : undefined)\n    let action = typeof actionValue == \"string\" ? inflection.camelize(actionValue.replaceAll(\"-\", \"_\"), true) : undefined\n    let controller = typeof controllerParam == \"string\" ? controllerParam : (Array.isArray(controllerParam) ? controllerParam[0] : undefined)\n\n    if (routeResolverHookMatch) {\n      controller = routeResolverHookMatch.controller\n      action = inflection.camelize(routeResolverHookMatch.action.replaceAll(\"-\", \"_\"), true)\n      this.params.controller = controller\n      this.params.action = routeResolverHookMatch.action\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else if (!matchResult) {\n        const __filename = fileURLToPath(import.meta.url)\n        const __dirname = dirname(__filename)\n        const requestedPath = currentPath.replace(/^\\//, \"\") || \"_root\"\n        const attemptedControllerPath = `${this.configuration.getDirectory()}/src/routes/${requestedPath}/controller.js`\n\n        await this.logger.warn(`No route matched for ${rawPath}. Tried controller at ${attemptedControllerPath}`)\n\n        controller = \"errors\"\n        controllerPath = \"./built-in/errors/controller.js\"\n        action = \"notFound\"\n        viewPath = await fs.realpath(`${__dirname}/built-in/errors`)\n    } else if (action) {\n      if (!controller) controller = \"_root\"\n\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else {\n      throw new Error(`Matched the route but didn't know what to do with it: ${rawPath} (action: ${action}, controller: ${controller}, params: ${JSON.stringify(this.params)})`)\n    }\n\n    const controllerClassImport = await import(toImportSpecifier(controllerPath))\n    const controllerClass = controllerClassImport.default\n    const controllerInstance = new controllerClass({\n      action,\n      configuration: this.configuration,\n      controller,\n      params: this.params,\n      request: this.request,\n      response: this.response,\n      viewPath\n    })\n\n    if (!(action in controllerInstance)) {\n      throw new Error(`Missing action on controller: ${controller}#${action}`)\n    }\n\n    await this._logActionStart({action, controllerClass})\n\n    try {\n      await this.configuration.ensureConnections(async () => {\n        const ability = await this.configuration.resolveAbility({\n          params: this.params,\n          request: this.request,\n          response: this.response\n        })\n\n        await this.configuration.runWithAbility(ability, async () => {\n          await controllerInstance._runBeforeCallbacks()\n          await controllerInstance[action]()\n        })\n      })\n    } catch (error) {\n      const ensuredError = ensureError(error)\n      const errorContext = {\n        action,\n        controller,\n        httpMethod: this.request.httpMethod(),\n        path: this.request.path(),\n        stage: \"controller-action\"\n      }\n\n      const errorWithContext = /** @type {{velociousContext?: object}} */ (ensuredError)\n\n      errorWithContext.velociousContext = {\n        ...(errorWithContext.velociousContext || {}),\n        controllerAction: errorContext\n      }\n\n      throw ensuredError\n    }\n  }\n\n  /**\n   * @param {import(\"./base-route.js\").default} route - Route.\n   * @param {string} path - Path.\n   * @returns {{restPath: string} | undefined} - REST path metadata for this route.\n   */\n  matchPathWithRoutes(route, path) {\n    const pathWithoutSlash = path.replace(/^\\//, \"\").split(\"?\")[0]\n\n    for (const subRoute of route.routes) {\n      const matchResult = subRoute.matchWithPath({\n        params: this.params,\n        path: pathWithoutSlash,\n        request: this.request\n      })\n\n      if (!matchResult) continue\n\n      const {restPath} = matchResult\n\n      if (restPath) {\n        return this.matchPathWithRoutes(subRoute, restPath)\n      }\n\n      return matchResult\n    }\n  }\n\n  /**\n   * @param {string} currentPath - Request path without query string.\n   * @returns {Promise<import(\"../configuration-types.js\").RouteResolverHookResult | null>} - Matched action/controller from hooks.\n   */\n  async resolveRouteResolverHooks(currentPath) {\n    const hooks = this.configuration.getRouteResolverHooks?.() || []\n\n    for (const hook of hooks) {\n      const hookResult = await hook({\n        configuration: this.configuration,\n        currentPath,\n        params: this.params,\n        request: this.request,\n        resolver: this,\n        response: this.response\n      })\n\n      if (!hookResult) continue\n\n      if (typeof hookResult.action !== \"string\" || hookResult.action.length < 1) {\n        throw new Error(`Expected route resolver hook action to be a string, got: ${hookResult.action}`)\n      }\n\n      if (typeof hookResult.controller !== \"string\" || hookResult.controller.length < 1) {\n        throw new Error(`Expected route resolver hook controller to be a string, got: ${hookResult.controller}`)\n      }\n\n      if (hookResult.params && typeof hookResult.params !== \"object\") {\n        throw new Error(`Expected route resolver hook params to be an object, got: ${hookResult.params}`)\n      }\n\n      if (hookResult.params) {\n        Object.assign(this.params, hookResult.params)\n      }\n\n      return hookResult\n    }\n\n    return null\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../controller.js\").default} args.controllerClass - Controller class.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _logActionStart({action, controllerClass}) {\n    const request = this.request\n    const timestamp = this._formatTimestamp(new Date())\n    const remoteAddress = request.remoteAddress?.() || request.header(\"x-forwarded-for\") || \"unknown\"\n    const loggedParams = /** @type {Record<string, unknown>} */ (this._sanitizeParamsForLogging(this.params))\n    const logMethod = this.configuration.getEnvironment() === \"test\" ? \"debug\" : \"info\"\n\n    delete loggedParams.action\n    delete loggedParams.controller\n\n    const controllerLogger = new Logger(controllerClass.name, {configuration: this.configuration})\n\n    await controllerLogger[logMethod](() => `Started ${request.httpMethod()} \"${request.path()}\" for ${remoteAddress} at ${timestamp}`)\n    await controllerLogger[logMethod](() => `Processing by ${controllerClass.name}#${action}`)\n    await controllerLogger[logMethod](() => [`  Parameters:`, loggedParams])\n  }\n\n  /**\n   * @param {Date} date - Date value.\n   * @returns {string} - The timestamp.\n   */\n  _formatTimestamp(date) {\n    /**\n     * @param {number} num - Num.\n     * @returns {string} - The pad.\n     */\n    const pad = (num) => String(num).padStart(2, \"0\")\n    const year = date.getFullYear()\n    const month = pad(date.getMonth() + 1)\n    const day = pad(date.getDate())\n    const hours = pad(date.getHours())\n    const minutes = pad(date.getMinutes())\n    const seconds = pad(date.getSeconds())\n    const offsetMinutes = date.getTimezoneOffset()\n    const offsetSign = offsetMinutes > 0 ? \"-\" : \"+\"\n    const offsetTotalMinutes = Math.abs(offsetMinutes)\n    const offsetHours = pad(Math.floor(offsetTotalMinutes / 60))\n    const offsetRemainingMinutes = pad(offsetTotalMinutes % 60)\n\n    return `${year}-${month}-${day} ${hours}:${minutes}:${seconds} ${offsetSign}${offsetHours}${offsetRemainingMinutes}`\n  }\n\n  /**\n   * @param {any} value - Value to use.\n   * @returns {any} - The sanitize params for logging.\n   */\n  _sanitizeParamsForLogging(value) {\n    if (value instanceof UploadedFile) {\n      return {\n        className: value.constructor.name,\n        filename: value.filename(),\n        size: value.size()\n      }\n    }\n\n    if (Array.isArray(value)) {\n      return value.map((item) => this._sanitizeParamsForLogging(item))\n    }\n\n    if (value && typeof value === \"object\") {\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key of Object.keys(value)) {\n        result[key] = this._sanitizeParamsForLogging(value[key])\n      }\n\n      return result\n    }\n\n    return value\n  }\n}\n"]}
259
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport Logger from \"../logger.js\"\nimport UploadedFile from \"../http-server/client/uploaded-file/uploaded-file.js\"\nimport ensureError from \"../utils/ensure-error.js\"\nimport toImportSpecifier from \"../utils/to-import-specifier.js\"\n\n/**\n * @param {string} actionName - Raw action name from route params or route hook.\n * @returns {string} - Normalized controller method name.\n */\nfunction normalizeActionName(actionName) {\n  return inflection.camelize(actionName.replaceAll(\"-\", \"_\").replaceAll(\"/\", \"_\"), true)\n}\n\nexport default class VelociousRoutesResolver {\n  /** @type {Logger | undefined} */\n  logger\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"../http-server/client/request.js\").default | import(\"../http-server/client/websocket-request.js\").default} args.request - Request object.\n   * @param {import(\"../http-server/client/response.js\").default} args.response - Response object.\n   */\n  constructor({configuration, request, response}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n\n    this.configuration = configuration\n    this.logger = new Logger(\"RoutesResolver\", {configuration})\n    const requestParams = request.params() || {}\n    this.params = {...requestParams}\n    delete this.params.action\n    delete this.params.controller\n    this.request = request\n    this.response = response\n  }\n\n  async resolve() {\n    let controllerPath\n    let currentRoute = this.configuration.routes.rootRoute\n    const rawPath = this.request.path()\n    const currentPath = rawPath.split(\"?\")[0]\n    let viewPath\n\n    const routeResolverHookMatch = await this.resolveRouteResolverHooks(currentPath)\n    const matchResult = routeResolverHookMatch ? undefined : this.matchPathWithRoutes(currentRoute, currentPath)\n    const actionParam = this.params.action\n    const controllerParam = this.params.controller\n    const actionValue = typeof actionParam == \"string\" ? actionParam : (Array.isArray(actionParam) ? actionParam[0] : undefined)\n    let action = typeof actionValue == \"string\" ? normalizeActionName(actionValue) : undefined\n    let controller = typeof controllerParam == \"string\" ? controllerParam : (Array.isArray(controllerParam) ? controllerParam[0] : undefined)\n\n    if (routeResolverHookMatch) {\n      controller = routeResolverHookMatch.controller\n      action = normalizeActionName(routeResolverHookMatch.action)\n      this.params.controller = controller\n      this.params.action = routeResolverHookMatch.action\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else if (!matchResult) {\n        const __filename = fileURLToPath(import.meta.url)\n        const __dirname = dirname(__filename)\n        const requestedPath = currentPath.replace(/^\\//, \"\") || \"_root\"\n        const attemptedControllerPath = `${this.configuration.getDirectory()}/src/routes/${requestedPath}/controller.js`\n\n        await this.logger.warn(`No route matched for ${rawPath}. Tried controller at ${attemptedControllerPath}`)\n\n        controller = \"errors\"\n        controllerPath = \"./built-in/errors/controller.js\"\n        action = \"notFound\"\n        viewPath = await fs.realpath(`${__dirname}/built-in/errors`)\n    } else if (action) {\n      if (!controller) controller = \"_root\"\n\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else {\n      throw new Error(`Matched the route but didn't know what to do with it: ${rawPath} (action: ${action}, controller: ${controller}, params: ${JSON.stringify(this.params)})`)\n    }\n\n    const controllerClassImport = await import(toImportSpecifier(controllerPath))\n    const controllerClass = controllerClassImport.default\n    const controllerInstance = new controllerClass({\n      action,\n      configuration: this.configuration,\n      controller,\n      params: this.params,\n      request: this.request,\n      response: this.response,\n      viewPath\n    })\n\n    if (!(action in controllerInstance)) {\n      throw new Error(`Missing action on controller: ${controller}#${action}`)\n    }\n\n    await this._logActionStart({action, controllerClass})\n\n    try {\n      await this.configuration.ensureConnections(async () => {\n        const ability = await this.configuration.resolveAbility({\n          params: this.params,\n          request: this.request,\n          response: this.response\n        })\n\n        await this.configuration.runWithAbility(ability, async () => {\n          await controllerInstance._runBeforeCallbacks()\n          await controllerInstance[action]()\n        })\n      })\n    } catch (error) {\n      const ensuredError = ensureError(error)\n      const errorContext = {\n        action,\n        controller,\n        httpMethod: this.request.httpMethod(),\n        path: this.request.path(),\n        stage: \"controller-action\"\n      }\n\n      const errorWithContext = /** @type {{velociousContext?: object}} */ (ensuredError)\n\n      errorWithContext.velociousContext = {\n        ...(errorWithContext.velociousContext || {}),\n        controllerAction: errorContext\n      }\n\n      throw ensuredError\n    }\n  }\n\n  /**\n   * @param {import(\"./base-route.js\").default} route - Route.\n   * @param {string} path - Path.\n   * @returns {{restPath: string} | undefined} - REST path metadata for this route.\n   */\n  matchPathWithRoutes(route, path) {\n    const pathWithoutSlash = path.replace(/^\\//, \"\").split(\"?\")[0]\n\n    for (const subRoute of route.routes) {\n      const paramsSnapshot = {...this.params}\n      const matchResult = subRoute.matchWithPath({\n        params: this.params,\n        path: pathWithoutSlash,\n        request: this.request\n      })\n\n      if (!matchResult) {\n        this.params = paramsSnapshot\n        continue\n      }\n\n      const {restPath} = matchResult\n\n      if (restPath) {\n        const recursiveMatch = this.matchPathWithRoutes(subRoute, restPath)\n\n        if (recursiveMatch) {\n          return recursiveMatch\n        }\n\n        this.params = paramsSnapshot\n        continue\n      }\n\n      return matchResult\n    }\n  }\n\n  /**\n   * @param {string} currentPath - Request path without query string.\n   * @returns {Promise<import(\"../configuration-types.js\").RouteResolverHookResult | null>} - Matched action/controller from hooks.\n   */\n  async resolveRouteResolverHooks(currentPath) {\n    const hooks = this.configuration.getRouteResolverHooks?.() || []\n\n    for (const hook of hooks) {\n      const hookResult = await hook({\n        configuration: this.configuration,\n        currentPath,\n        params: this.params,\n        request: this.request,\n        resolver: this,\n        response: this.response\n      })\n\n      if (!hookResult) continue\n\n      if (typeof hookResult.action !== \"string\" || hookResult.action.length < 1) {\n        throw new Error(`Expected route resolver hook action to be a string, got: ${hookResult.action}`)\n      }\n\n      if (typeof hookResult.controller !== \"string\" || hookResult.controller.length < 1) {\n        throw new Error(`Expected route resolver hook controller to be a string, got: ${hookResult.controller}`)\n      }\n\n      if (hookResult.params && typeof hookResult.params !== \"object\") {\n        throw new Error(`Expected route resolver hook params to be an object, got: ${hookResult.params}`)\n      }\n\n      if (hookResult.params) {\n        Object.assign(this.params, hookResult.params)\n      }\n\n      return hookResult\n    }\n\n    return null\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../controller.js\").default} args.controllerClass - Controller class.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _logActionStart({action, controllerClass}) {\n    const request = this.request\n    const timestamp = this._formatTimestamp(new Date())\n    const remoteAddress = request.remoteAddress?.() || request.header(\"x-forwarded-for\") || \"unknown\"\n    const loggedParams = /** @type {Record<string, unknown>} */ (this._sanitizeParamsForLogging(this.params))\n    const logMethod = this.configuration.getEnvironment() === \"test\" ? \"debug\" : \"info\"\n\n    delete loggedParams.action\n    delete loggedParams.controller\n\n    const controllerLogger = new Logger(controllerClass.name, {configuration: this.configuration})\n\n    await controllerLogger[logMethod](() => `Started ${request.httpMethod()} \"${request.path()}\" for ${remoteAddress} at ${timestamp}`)\n    await controllerLogger[logMethod](() => `Processing by ${controllerClass.name}#${action}`)\n    await controllerLogger[logMethod](() => [`  Parameters:`, loggedParams])\n  }\n\n  /**\n   * @param {Date} date - Date value.\n   * @returns {string} - The timestamp.\n   */\n  _formatTimestamp(date) {\n    /**\n     * @param {number} num - Num.\n     * @returns {string} - The pad.\n     */\n    const pad = (num) => String(num).padStart(2, \"0\")\n    const year = date.getFullYear()\n    const month = pad(date.getMonth() + 1)\n    const day = pad(date.getDate())\n    const hours = pad(date.getHours())\n    const minutes = pad(date.getMinutes())\n    const seconds = pad(date.getSeconds())\n    const offsetMinutes = date.getTimezoneOffset()\n    const offsetSign = offsetMinutes > 0 ? \"-\" : \"+\"\n    const offsetTotalMinutes = Math.abs(offsetMinutes)\n    const offsetHours = pad(Math.floor(offsetTotalMinutes / 60))\n    const offsetRemainingMinutes = pad(offsetTotalMinutes % 60)\n\n    return `${year}-${month}-${day} ${hours}:${minutes}:${seconds} ${offsetSign}${offsetHours}${offsetRemainingMinutes}`\n  }\n\n  /**\n   * @param {any} value - Value to use.\n   * @returns {any} - The sanitize params for logging.\n   */\n  _sanitizeParamsForLogging(value) {\n    if (value instanceof UploadedFile) {\n      return {\n        className: value.constructor.name,\n        filename: value.filename(),\n        size: value.size()\n      }\n    }\n\n    if (Array.isArray(value)) {\n      return value.map((item) => this._sanitizeParamsForLogging(item))\n    }\n\n    if (value && typeof value === \"object\") {\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key of Object.keys(value)) {\n        result[key] = this._sanitizeParamsForLogging(value[key])\n      }\n\n      return result\n    }\n\n    return value\n  }\n}\n"]}
package/package.json CHANGED
@@ -3,7 +3,7 @@
3
3
  "velocious": "build/bin/velocious.js"
4
4
  },
5
5
  "name": "velocious",
6
- "version": "1.0.217",
6
+ "version": "1.0.219",
7
7
  "main": "build/index.js",
8
8
  "types": "build/index.d.ts",
9
9
  "files": [