velocious 1.0.349 → 1.0.350

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
@@ -293,13 +293,6 @@ import FrontendModelBaseResource from "velocious/build/src/frontend-model-resour
293
293
  class UserResource extends FrontendModelBaseResource {
294
294
  static resourceConfig() {
295
295
  return {
296
- abilities: {
297
- create: "create",
298
- destroy: "destroy",
299
- find: "read",
300
- index: "read",
301
- update: "update"
302
- },
303
296
  attributes: ["id", "name", "email"],
304
297
  relationships: {
305
298
  projects: {type: "hasMany", model: "Project"}
@@ -323,6 +316,15 @@ export default new Configuration({
323
316
 
324
317
  `frontendModels` entries must be `FrontendModelBaseResource` subclasses. Built-in CRUD/find/index/serialize behavior lives in the base class, and app resources override only the pieces they actually need.
325
318
 
319
+ Resources expose the full CRUD ability set (`create`, `destroy`, `read`, `update`) by default. To restrict the API surface — for example to a read-only resource — declare an explicit subset:
320
+
321
+ ```js
322
+ class AuditLogResource extends FrontendModelBaseResource {
323
+ static abilities = ["read"]
324
+ static attributes = ["id", "message", "createdAt"]
325
+ }
326
+ ```
327
+
326
328
  Generate classes:
327
329
 
328
330
  ```bash
@@ -1 +1 @@
1
- {"version":3,"file":"resource-definition.d.ts","sourceRoot":"","sources":["../../../src/frontend-models/resource-definition.js"],"names":[],"mappings":"AAOA;;;GAGG;AACH,wEAHW,OAAO,2BAA2B,EAAE,2BAA2B,GAC7D,MAAM,CAAC,MAAM,EAAE,OAAO,yBAAyB,CAAC,CAc5D;AAED;;;GAGG;AACH,8DAHW,OAAO,GACL,KAAK,IAAI,OAAO,yBAAyB,CAIrD;AAED;;;GAGG;AACH,6EAHW,OAAO,GACL,OAAO,yBAAyB,GAAG,IAAI,CAInD;AAED;;;GAGG;AACH,qFAHW,OAAO,GACL,OAAO,2BAA2B,EAAE,4CAA4C,GAAG,IAAI,CAMnG;AAuLD;;;;GAIG;AACH,qDAJW,MAAM,sBACN,OAAO,GACL,MAAM,CAUlB;AAED;;;;;;GAMG;AACH,8FALG;IAAqB,WAAW,EAAxB,MAAM;IACO,SAAS,EAAtB,MAAM;IACQ,kBAAkB,EAAhC,OAAO;CACf,GAAU,SAAS,GAAG,MAAM,GAAG,OAAO,GAAG,QAAQ,GAAG,QAAQ,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,GAAG,IAAI,CA2BrG;AAED;;;;;GAKG;AACH,oFAJG;IAAgF,eAAe,EAAvF,OAAO,2BAA2B,EAAE,2BAA2B,EAAE;IACpD,WAAW,EAAxB,MAAM;CACd,GAAU;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAAC,UAAU,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,YAAY,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,YAAY,GAAG,QAAQ,CAAA;CAAC,GAAG,IAAI,CA8DxJ;sCAxVqC,6CAA6C"}
1
+ {"version":3,"file":"resource-definition.d.ts","sourceRoot":"","sources":["../../../src/frontend-models/resource-definition.js"],"names":[],"mappings":"AAOA;;;GAGG;AACH,wEAHW,OAAO,2BAA2B,EAAE,2BAA2B,GAC7D,MAAM,CAAC,MAAM,EAAE,OAAO,yBAAyB,CAAC,CAc5D;AAED;;;GAGG;AACH,8DAHW,OAAO,GACL,KAAK,IAAI,OAAO,yBAAyB,CAIrD;AAED;;;GAGG;AACH,6EAHW,OAAO,GACL,OAAO,yBAAyB,GAAG,IAAI,CAInD;AAED;;;GAGG;AACH,qFAHW,OAAO,GACL,OAAO,2BAA2B,EAAE,4CAA4C,GAAG,IAAI,CAMnG;AAkMD;;;;GAIG;AACH,qDAJW,MAAM,sBACN,OAAO,GACL,MAAM,CAUlB;AAED;;;;;;GAMG;AACH,8FALG;IAAqB,WAAW,EAAxB,MAAM;IACO,SAAS,EAAtB,MAAM;IACQ,kBAAkB,EAAhC,OAAO;CACf,GAAU,SAAS,GAAG,MAAM,GAAG,OAAO,GAAG,QAAQ,GAAG,QAAQ,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,GAAG,IAAI,CA2BrG;AAED;;;;;GAKG;AACH,oFAJG;IAAgF,eAAe,EAAvF,OAAO,2BAA2B,EAAE,2BAA2B,EAAE;IACpD,WAAW,EAAxB,MAAM;CACd,GAAU;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAAC,UAAU,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,YAAY,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,YAAY,GAAG,QAAQ,CAAA;CAAC,GAAG,IAAI,CA8DxJ;sCAnWqC,6CAA6C"}
@@ -78,22 +78,23 @@ function normalizeFrontendModelResourceConfiguration(resourceConfiguration) {
78
78
  * @returns {Record<string, string>} - Normalized abilities config.
79
79
  */
80
80
  function normalizeFrontendModelResourceAbilities(abilities) {
81
- if (!abilities) {
82
- return { find: "read", index: "read" };
81
+ if (abilities === undefined) {
82
+ return defaultCrudAbilities();
83
83
  }
84
84
  if (!Array.isArray(abilities)) {
85
85
  throw new Error("Resource abilities must be an array of action names. Object form is no longer supported.");
86
86
  }
87
- /** @type {Record<string, string>} */
88
- const normalized = {};
89
87
  if (abilities.includes("manage")) {
90
- normalized.create = "manage";
91
- normalized.destroy = "manage";
92
- normalized.find = "manage";
93
- normalized.index = "manage";
94
- normalized.update = "manage";
95
- return normalized;
88
+ return {
89
+ create: "manage",
90
+ destroy: "manage",
91
+ find: "manage",
92
+ index: "manage",
93
+ update: "manage"
94
+ };
96
95
  }
96
+ /** @type {Record<string, string>} */
97
+ const normalized = {};
97
98
  if (abilities.includes("create"))
98
99
  normalized.create = "create";
99
100
  if (abilities.includes("destroy"))
@@ -106,6 +107,16 @@ function normalizeFrontendModelResourceAbilities(abilities) {
106
107
  normalized.update = "update";
107
108
  return normalized;
108
109
  }
110
+ /** @returns {Record<string, string>} - Default CRUD ability map. */
111
+ function defaultCrudAbilities() {
112
+ return {
113
+ create: "create",
114
+ destroy: "destroy",
115
+ find: "read",
116
+ index: "read",
117
+ update: "update"
118
+ };
119
+ }
109
120
  /**
110
121
  * @param {import("../configuration-types.js").FrontendModelResourceConfiguration} resourceConfiguration - Raw resource configuration.
111
122
  * @returns {{builtInCollectionCommands: Record<string, string>, builtInMemberCommands: Record<string, string>, collectionCommands: Record<string, string>, memberCommands: Record<string, string>}} - Normalized command configuration.
@@ -305,4 +316,4 @@ function normalizeFrontendModelResourcePathForMatch(path) {
305
316
  }
306
317
  return withLeadingSlash;
307
318
  }
308
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-definition.js","sourceRoot":"","sources":["../../../src/frontend-models/resource-definition.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport * as inflection from \"inflection\"\nimport FrontendModelBaseResource from \"../frontend-model-resource/base-resource.js\"\nimport restArgsError from \"../utils/rest-args-error.js\"\nimport {validateFrontendModelResourceCommandName} from \"./resource-config-validation.js\"\n\n/**\n * @param {import(\"../configuration-types.js\").BackendProjectConfiguration} backendProject - Backend project config.\n * @returns {Record<string, typeof FrontendModelBaseResource>} - Resource definitions keyed by model name.\n */\nexport function frontendModelResourcesForBackendProject(backendProject) {\n  const resources = backendProject.frontendModels\n\n  if (resources !== undefined) {\n    if (!resources || typeof resources !== \"object\") {\n      throw new Error(`Expected backend project frontendModels object but got: ${resources}`)\n    }\n\n    return resources\n  }\n\n  return {}\n}\n\n/**\n * @param {unknown} value - Candidate resource definition.\n * @returns {value is typeof FrontendModelBaseResource} - Whether value is a resource class.\n */\nexport function frontendModelResourceDefinitionIsClass(value) {\n  return typeof value === \"function\" && (value === FrontendModelBaseResource || value.prototype instanceof FrontendModelBaseResource)\n}\n\n/**\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {typeof FrontendModelBaseResource | null} - Resource class when definition is class-based.\n */\nexport function frontendModelResourceClassFromDefinition(resourceDefinition) {\n  return frontendModelResourceDefinitionIsClass(resourceDefinition) ? resourceDefinition : null\n}\n\n/**\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {import(\"../configuration-types.js\").NormalizedFrontendModelResourceConfiguration | null} - Normalized resource configuration.\n */\nexport function frontendModelResourceConfigurationFromDefinition(resourceDefinition) {\n  if (!frontendModelResourceDefinitionIsClass(resourceDefinition)) return null\n\n  return normalizeFrontendModelResourceConfiguration(resourceDefinition.resourceConfig())\n}\n\n/**\n * @param {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Raw resource configuration.\n * @returns {import(\"../configuration-types.js\").NormalizedFrontendModelResourceConfiguration} - Normalized resource configuration.\n */\nfunction normalizeFrontendModelResourceConfiguration(resourceConfiguration) {\n  const restArgs = /** @type {Record<string, any>} */ ({...resourceConfiguration})\n\n  for (const key of [\n    \"abilities\",\n    \"attributes\",\n    \"attachments\",\n    \"builtInCollectionCommands\",\n    \"builtInMemberCommands\",\n    \"collectionCommands\",\n    \"commands\",\n    \"memberCommands\",\n    \"modelName\",\n    \"primaryKey\",\n    \"relationships\",\n    \"server\"\n  ]) {\n    delete restArgs[key]\n  }\n\n  restArgsError(restArgs)\n\n  const normalizedCommands = normalizeFrontendModelResourceCommands(resourceConfiguration)\n\n  return {\n    ...resourceConfiguration,\n    abilities: normalizeFrontendModelResourceAbilities(resourceConfiguration.abilities),\n    builtInCollectionCommands: normalizedCommands.builtInCollectionCommands,\n    builtInMemberCommands: normalizedCommands.builtInMemberCommands,\n    collectionCommands: normalizedCommands.collectionCommands,\n    memberCommands: normalizedCommands.memberCommands\n  }\n}\n\n/**\n * @param {string[] | undefined} abilities - Resource abilities config (camelCase action list).\n * @returns {Record<string, string>} - Normalized abilities config.\n */\nfunction normalizeFrontendModelResourceAbilities(abilities) {\n  if (!abilities) {\n    return {find: \"read\", index: \"read\"}\n  }\n\n  if (!Array.isArray(abilities)) {\n    throw new Error(\"Resource abilities must be an array of action names. Object form is no longer supported.\")\n  }\n\n  /** @type {Record<string, string>} */\n  const normalized = {}\n\n  if (abilities.includes(\"manage\")) {\n    normalized.create = \"manage\"\n    normalized.destroy = \"manage\"\n    normalized.find = \"manage\"\n    normalized.index = \"manage\"\n    normalized.update = \"manage\"\n\n    return normalized\n  }\n\n  if (abilities.includes(\"create\")) normalized.create = \"create\"\n  if (abilities.includes(\"destroy\")) normalized.destroy = \"destroy\"\n  if (abilities.includes(\"read\")) {\n    normalized.find = \"read\"\n    normalized.index = \"read\"\n  }\n  if (abilities.includes(\"update\")) normalized.update = \"update\"\n\n  return normalized\n}\n\n/**\n * @param {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Raw resource configuration.\n * @returns {{builtInCollectionCommands: Record<string, string>, builtInMemberCommands: Record<string, string>, collectionCommands: Record<string, string>, memberCommands: Record<string, string>}} - Normalized command configuration.\n */\nfunction normalizeFrontendModelResourceCommands(resourceConfiguration) {\n  const builtInCollectionCommands = resourceConfiguration.builtInCollectionCommands\n  const builtInMemberCommands = resourceConfiguration.builtInMemberCommands\n  const customCollectionCommands = resourceConfiguration.collectionCommands\n  const customMemberCommands = resourceConfiguration.memberCommands\n  const normalizedBuiltInCollectionCommands = normalizeFrontendModelBuiltInCommands({\n    commandDefaults: {\n      create: \"create\",\n      index: \"index\"\n    },\n    commandsConfig: builtInCollectionCommands,\n    modelName: \"CollectionCommand\"\n  })\n  const normalizedBuiltInMemberCommands = normalizeFrontendModelBuiltInCommands({\n    commandDefaults: {\n      attach: \"attach\",\n      destroy: \"destroy\",\n      download: \"download\",\n      find: \"find\",\n      update: \"update\",\n      url: \"url\"\n    },\n    commandsConfig: builtInMemberCommands,\n    modelName: \"MemberCommand\"\n  })\n\n  return {\n    builtInCollectionCommands: normalizedBuiltInCollectionCommands,\n    builtInMemberCommands: normalizedBuiltInMemberCommands,\n    collectionCommands: normalizeFrontendModelCustomCommands({commandsConfig: customCollectionCommands, modelName: \"CollectionCommand\"}),\n    memberCommands: normalizeFrontendModelCustomCommands({commandsConfig: customMemberCommands, modelName: \"MemberCommand\"})\n  }\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {Record<string, string>} args.commandDefaults - Built-in default command names.\n * @param {string[] | undefined} args.commandsConfig - Built-in commands config (camelCase command type list).\n * @param {string} args.modelName - Diagnostic model name.\n * @returns {Record<string, string>} - Normalized built-in command config.\n */\nfunction normalizeFrontendModelBuiltInCommands({commandDefaults, commandsConfig, modelName}) {\n  if (!commandsConfig) {\n    return commandDefaults\n  }\n\n  if (!Array.isArray(commandsConfig)) {\n    throw new Error(`${modelName} configuration must use the array form. Object form is no longer supported.`)\n  }\n\n  /** @type {Record<string, string>} */\n  const normalizedCommands = {}\n\n  for (const commandType of commandsConfig) {\n    const defaultCommandName = commandDefaults[commandType]\n\n    if (!defaultCommandName) {\n      throw new Error(`Unknown built-in frontend model command '${commandType}' for ${modelName}`)\n    }\n\n    normalizedCommands[commandType] = validateFrontendModelResourceCommandName({\n      commandName: defaultCommandName,\n      commandType: defaultCommandName,\n      modelName\n    })\n  }\n\n  return normalizedCommands\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {string[] | undefined} args.commandsConfig - Custom commands config (camelCase method-name list).\n * @param {string} args.modelName - Diagnostic model name.\n * @returns {Record<string, string>} - Normalized custom command config (camelCase method name → kebab-case command slug).\n */\nfunction normalizeFrontendModelCustomCommands({commandsConfig, modelName}) {\n  if (!commandsConfig) {\n    return {}\n  }\n\n  if (!Array.isArray(commandsConfig)) {\n    throw new Error(`${modelName} configuration must use the array form. Object form is no longer supported.`)\n  }\n\n  /** @type {Record<string, string>} */\n  const normalizedCommands = {}\n\n  for (const methodName of commandsConfig) {\n    const validatedMethodName = validateFrontendModelResourceCommandName({\n      commandName: methodName,\n      commandType: methodName,\n      modelName\n    })\n    const commandSlug = inflection.dasherize(inflection.underscore(validatedMethodName))\n\n    normalizedCommands[validatedMethodName] = commandSlug\n  }\n\n  return normalizedCommands\n}\n\n/**\n * @param {string} modelName - Model class name.\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {string} - Normalized resource path.\n */\nexport function frontendModelResourcePath(modelName, resourceDefinition) {\n  const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n  if (!resourceConfiguration) {\n    throw new Error(`Invalid frontend model resource definition for ${modelName}`)\n  }\n\n  return `/${inflection.dasherize(inflection.pluralize(inflection.underscore(modelName)))}`\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {string} args.commandName - Command path segment.\n * @param {string} args.modelName - Model class name.\n * @param {unknown} args.resourceDefinition - Resource definition.\n * @returns {\"destroy\" | \"find\" | \"index\" | \"create\" | \"update\" | \"attach\" | \"download\" | \"url\" | null} - Frontend action.\n */\nexport function frontendModelActionForCommand({commandName, modelName, resourceDefinition}) {\n  const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n  if (!resourceConfiguration) {\n    throw new Error(`Invalid frontend model resource definition for ${modelName}`)\n  }\n\n  for (const [action, configuredCommandName] of Object.entries({\n    ...resourceConfiguration.builtInCollectionCommands,\n    ...resourceConfiguration.builtInMemberCommands\n  })) {\n    if (configuredCommandName === undefined) continue\n\n    const validatedCommandName = validateFrontendModelResourceCommandName({\n      commandName: configuredCommandName,\n      commandType: /** @type {\"attach\" | \"create\" | \"destroy\" | \"download\" | \"find\" | \"index\" | \"update\" | \"url\"} */ (action),\n      modelName\n    })\n\n    if (commandName === validatedCommandName) {\n      return /** @type {\"attach\" | \"create\" | \"destroy\" | \"download\" | \"find\" | \"index\" | \"update\" | \"url\"} */ (action)\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {import(\"../configuration-types.js\").BackendProjectConfiguration[]} args.backendProjects - Backend projects to scan.\n * @param {string} args.currentPath - Request path without query.\n * @returns {{commandName: string, memberId?: string, methodName: string, modelName: string, resourcePath: string, scope: \"collection\" | \"member\"} | null} - Matched custom command metadata.\n */\nexport function frontendModelCustomCommandForPath({backendProjects, currentPath}) {\n  const normalizedCurrentPath = normalizeFrontendModelResourcePathForMatch(currentPath)\n\n  for (const backendProject of backendProjects) {\n    const resources = frontendModelResourcesForBackendProject(backendProject)\n\n    for (const modelName in resources) {\n      const resourceDefinition = resources[modelName]\n      const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n      if (!resourceConfiguration) {\n        continue\n      }\n\n      const resourcePath = normalizeFrontendModelResourcePathForMatch(frontendModelResourcePath(modelName, resourceDefinition))\n      const expectedPrefix = `${resourcePath}/`\n\n      if (!normalizedCurrentPath.startsWith(expectedPrefix)) {\n        continue\n      }\n\n      const pathSegments = normalizedCurrentPath\n        .slice(expectedPrefix.length)\n        .split(\"/\")\n        .filter(Boolean)\n\n      if (pathSegments.length === 1) {\n        const matchedCollectionCommand = Object.entries(resourceConfiguration.collectionCommands)\n          .find(([, commandName]) => commandName === pathSegments[0])\n\n        if (matchedCollectionCommand) {\n          return {\n            commandName: matchedCollectionCommand[1],\n            methodName: matchedCollectionCommand[0],\n            modelName,\n            resourcePath,\n            scope: \"collection\"\n          }\n        }\n      }\n\n      if (pathSegments.length === 2) {\n        const matchedMemberCommand = Object.entries(resourceConfiguration.memberCommands)\n          .find(([, commandName]) => commandName === pathSegments[1])\n\n        if (matchedMemberCommand) {\n          return {\n            commandName: matchedMemberCommand[1],\n            memberId: decodeURIComponent(pathSegments[0]),\n            methodName: matchedMemberCommand[0],\n            modelName,\n            resourcePath,\n            scope: \"member\"\n          }\n        }\n      }\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {string} path - Path value.\n * @returns {string} - Normalized path with leading slash and no trailing slash.\n */\nfunction normalizeFrontendModelResourcePathForMatch(path) {\n  const withLeadingSlash = path.startsWith(\"/\") ? path : `/${path}`\n\n  if (withLeadingSlash.length > 1) {\n    return withLeadingSlash.replace(/\\/+$/, \"\")\n  }\n\n  return withLeadingSlash\n}\n"]}
319
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-definition.js","sourceRoot":"","sources":["../../../src/frontend-models/resource-definition.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,KAAK,UAAU,MAAM,YAAY,CAAA;AACxC,OAAO,yBAAyB,MAAM,6CAA6C,CAAA;AACnF,OAAO,aAAa,MAAM,6BAA6B,CAAA;AACvD,OAAO,EAAC,wCAAwC,EAAC,MAAM,iCAAiC,CAAA;AAExF;;;GAGG;AACH,MAAM,UAAU,uCAAuC,CAAC,cAAc;IACpE,MAAM,SAAS,GAAG,cAAc,CAAC,cAAc,CAAA;IAE/C,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;QAC5B,IAAI,CAAC,SAAS,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;YAChD,MAAM,IAAI,KAAK,CAAC,2DAA2D,SAAS,EAAE,CAAC,CAAA;QACzF,CAAC;QAED,OAAO,SAAS,CAAA;IAClB,CAAC;IAED,OAAO,EAAE,CAAA;AACX,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,sCAAsC,CAAC,KAAK;IAC1D,OAAO,OAAO,KAAK,KAAK,UAAU,IAAI,CAAC,KAAK,KAAK,yBAAyB,IAAI,KAAK,CAAC,SAAS,YAAY,yBAAyB,CAAC,CAAA;AACrI,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,wCAAwC,CAAC,kBAAkB;IACzE,OAAO,sCAAsC,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAA;AAC/F,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,gDAAgD,CAAC,kBAAkB;IACjF,IAAI,CAAC,sCAAsC,CAAC,kBAAkB,CAAC;QAAE,OAAO,IAAI,CAAA;IAE5E,OAAO,2CAA2C,CAAC,kBAAkB,CAAC,cAAc,EAAE,CAAC,CAAA;AACzF,CAAC;AAED;;;GAGG;AACH,SAAS,2CAA2C,CAAC,qBAAqB;IACxE,MAAM,QAAQ,GAAG,kCAAkC,CAAC,CAAC,EAAC,GAAG,qBAAqB,EAAC,CAAC,CAAA;IAEhF,KAAK,MAAM,GAAG,IAAI;QAChB,WAAW;QACX,YAAY;QACZ,aAAa;QACb,2BAA2B;QAC3B,uBAAuB;QACvB,oBAAoB;QACpB,UAAU;QACV,gBAAgB;QAChB,WAAW;QACX,YAAY;QACZ,eAAe;QACf,QAAQ;KACT,EAAE,CAAC;QACF,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAA;IACtB,CAAC;IAED,aAAa,CAAC,QAAQ,CAAC,CAAA;IAEvB,MAAM,kBAAkB,GAAG,sCAAsC,CAAC,qBAAqB,CAAC,CAAA;IAExF,OAAO;QACL,GAAG,qBAAqB;QACxB,SAAS,EAAE,uCAAuC,CAAC,qBAAqB,CAAC,SAAS,CAAC;QACnF,yBAAyB,EAAE,kBAAkB,CAAC,yBAAyB;QACvE,qBAAqB,EAAE,kBAAkB,CAAC,qBAAqB;QAC/D,kBAAkB,EAAE,kBAAkB,CAAC,kBAAkB;QACzD,cAAc,EAAE,kBAAkB,CAAC,cAAc;KAClD,CAAA;AACH,CAAC;AAED;;;GAGG;AACH,SAAS,uCAAuC,CAAC,SAAS;IACxD,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;QAC5B,OAAO,oBAAoB,EAAE,CAAA;IAC/B,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CAAC,0FAA0F,CAAC,CAAA;IAC7G,CAAC;IAED,IAAI,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;QACjC,OAAO;YACL,MAAM,EAAE,QAAQ;YAChB,OAAO,EAAE,QAAQ;YACjB,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,QAAQ;YACf,MAAM,EAAE,QAAQ;SACjB,CAAA;IACH,CAAC;IAED,qCAAqC;IACrC,MAAM,UAAU,GAAG,EAAE,CAAA;IAErB,IAAI,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC;QAAE,UAAU,CAAC,MAAM,GAAG,QAAQ,CAAA;IAC9D,IAAI,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC;QAAE,UAAU,CAAC,OAAO,GAAG,SAAS,CAAA;IACjE,IAAI,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/B,UAAU,CAAC,IAAI,GAAG,MAAM,CAAA;QACxB,UAAU,CAAC,KAAK,GAAG,MAAM,CAAA;IAC3B,CAAC;IACD,IAAI,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC;QAAE,UAAU,CAAC,MAAM,GAAG,QAAQ,CAAA;IAE9D,OAAO,UAAU,CAAA;AACnB,CAAC;AAED,oEAAoE;AACpE,SAAS,oBAAoB;IAC3B,OAAO;QACL,MAAM,EAAE,QAAQ;QAChB,OAAO,EAAE,SAAS;QAClB,IAAI,EAAE,MAAM;QACZ,KAAK,EAAE,MAAM;QACb,MAAM,EAAE,QAAQ;KACjB,CAAA;AACH,CAAC;AAED;;;GAGG;AACH,SAAS,sCAAsC,CAAC,qBAAqB;IACnE,MAAM,yBAAyB,GAAG,qBAAqB,CAAC,yBAAyB,CAAA;IACjF,MAAM,qBAAqB,GAAG,qBAAqB,CAAC,qBAAqB,CAAA;IACzE,MAAM,wBAAwB,GAAG,qBAAqB,CAAC,kBAAkB,CAAA;IACzE,MAAM,oBAAoB,GAAG,qBAAqB,CAAC,cAAc,CAAA;IACjE,MAAM,mCAAmC,GAAG,qCAAqC,CAAC;QAChF,eAAe,EAAE;YACf,MAAM,EAAE,QAAQ;YAChB,KAAK,EAAE,OAAO;SACf;QACD,cAAc,EAAE,yBAAyB;QACzC,SAAS,EAAE,mBAAmB;KAC/B,CAAC,CAAA;IACF,MAAM,+BAA+B,GAAG,qCAAqC,CAAC;QAC5E,eAAe,EAAE;YACf,MAAM,EAAE,QAAQ;YAChB,OAAO,EAAE,SAAS;YAClB,QAAQ,EAAE,UAAU;YACpB,IAAI,EAAE,MAAM;YACZ,MAAM,EAAE,QAAQ;YAChB,GAAG,EAAE,KAAK;SACX;QACD,cAAc,EAAE,qBAAqB;QACrC,SAAS,EAAE,eAAe;KAC3B,CAAC,CAAA;IAEF,OAAO;QACL,yBAAyB,EAAE,mCAAmC;QAC9D,qBAAqB,EAAE,+BAA+B;QACtD,kBAAkB,EAAE,oCAAoC,CAAC,EAAC,cAAc,EAAE,wBAAwB,EAAE,SAAS,EAAE,mBAAmB,EAAC,CAAC;QACpI,cAAc,EAAE,oCAAoC,CAAC,EAAC,cAAc,EAAE,oBAAoB,EAAE,SAAS,EAAE,eAAe,EAAC,CAAC;KACzH,CAAA;AACH,CAAC;AAED;;;;;;GAMG;AACH,SAAS,qCAAqC,CAAC,EAAC,eAAe,EAAE,cAAc,EAAE,SAAS,EAAC;IACzF,IAAI,CAAC,cAAc,EAAE,CAAC;QACpB,OAAO,eAAe,CAAA;IACxB,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;QACnC,MAAM,IAAI,KAAK,CAAC,GAAG,SAAS,6EAA6E,CAAC,CAAA;IAC5G,CAAC;IAED,qCAAqC;IACrC,MAAM,kBAAkB,GAAG,EAAE,CAAA;IAE7B,KAAK,MAAM,WAAW,IAAI,cAAc,EAAE,CAAC;QACzC,MAAM,kBAAkB,GAAG,eAAe,CAAC,WAAW,CAAC,CAAA;QAEvD,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,4CAA4C,WAAW,SAAS,SAAS,EAAE,CAAC,CAAA;QAC9F,CAAC;QAED,kBAAkB,CAAC,WAAW,CAAC,GAAG,wCAAwC,CAAC;YACzE,WAAW,EAAE,kBAAkB;YAC/B,WAAW,EAAE,kBAAkB;YAC/B,SAAS;SACV,CAAC,CAAA;IACJ,CAAC;IAED,OAAO,kBAAkB,CAAA;AAC3B,CAAC;AAED;;;;;GAKG;AACH,SAAS,oCAAoC,CAAC,EAAC,cAAc,EAAE,SAAS,EAAC;IACvE,IAAI,CAAC,cAAc,EAAE,CAAC;QACpB,OAAO,EAAE,CAAA;IACX,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;QACnC,MAAM,IAAI,KAAK,CAAC,GAAG,SAAS,6EAA6E,CAAC,CAAA;IAC5G,CAAC;IAED,qCAAqC;IACrC,MAAM,kBAAkB,GAAG,EAAE,CAAA;IAE7B,KAAK,MAAM,UAAU,IAAI,cAAc,EAAE,CAAC;QACxC,MAAM,mBAAmB,GAAG,wCAAwC,CAAC;YACnE,WAAW,EAAE,UAAU;YACvB,WAAW,EAAE,UAAU;YACvB,SAAS;SACV,CAAC,CAAA;QACF,MAAM,WAAW,GAAG,UAAU,CAAC,SAAS,CAAC,UAAU,CAAC,UAAU,CAAC,mBAAmB,CAAC,CAAC,CAAA;QAEpF,kBAAkB,CAAC,mBAAmB,CAAC,GAAG,WAAW,CAAA;IACvD,CAAC;IAED,OAAO,kBAAkB,CAAA;AAC3B,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,yBAAyB,CAAC,SAAS,EAAE,kBAAkB;IACrE,MAAM,qBAAqB,GAAG,gDAAgD,CAAC,kBAAkB,CAAC,CAAA;IAElG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CAAC,kDAAkD,SAAS,EAAE,CAAC,CAAA;IAChF,CAAC;IAED,OAAO,IAAI,UAAU,CAAC,SAAS,CAAC,UAAU,CAAC,SAAS,CAAC,UAAU,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAA;AAC3F,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,6BAA6B,CAAC,EAAC,WAAW,EAAE,SAAS,EAAE,kBAAkB,EAAC;IACxF,MAAM,qBAAqB,GAAG,gDAAgD,CAAC,kBAAkB,CAAC,CAAA;IAElG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CAAC,kDAAkD,SAAS,EAAE,CAAC,CAAA;IAChF,CAAC;IAED,KAAK,MAAM,CAAC,MAAM,EAAE,qBAAqB,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC;QAC3D,GAAG,qBAAqB,CAAC,yBAAyB;QAClD,GAAG,qBAAqB,CAAC,qBAAqB;KAC/C,CAAC,EAAE,CAAC;QACH,IAAI,qBAAqB,KAAK,SAAS;YAAE,SAAQ;QAEjD,MAAM,oBAAoB,GAAG,wCAAwC,CAAC;YACpE,WAAW,EAAE,qBAAqB;YAClC,WAAW,EAAE,iGAAiG,CAAC,CAAC,MAAM,CAAC;YACvH,SAAS;SACV,CAAC,CAAA;QAEF,IAAI,WAAW,KAAK,oBAAoB,EAAE,CAAC;YACzC,OAAO,iGAAiG,CAAC,CAAC,MAAM,CAAC,CAAA;QACnH,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAA;AACb,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,iCAAiC,CAAC,EAAC,eAAe,EAAE,WAAW,EAAC;IAC9E,MAAM,qBAAqB,GAAG,0CAA0C,CAAC,WAAW,CAAC,CAAA;IAErF,KAAK,MAAM,cAAc,IAAI,eAAe,EAAE,CAAC;QAC7C,MAAM,SAAS,GAAG,uCAAuC,CAAC,cAAc,CAAC,CAAA;QAEzE,KAAK,MAAM,SAAS,IAAI,SAAS,EAAE,CAAC;YAClC,MAAM,kBAAkB,GAAG,SAAS,CAAC,SAAS,CAAC,CAAA;YAC/C,MAAM,qBAAqB,GAAG,gDAAgD,CAAC,kBAAkB,CAAC,CAAA;YAElG,IAAI,CAAC,qBAAqB,EAAE,CAAC;gBAC3B,SAAQ;YACV,CAAC;YAED,MAAM,YAAY,GAAG,0CAA0C,CAAC,yBAAyB,CAAC,SAAS,EAAE,kBAAkB,CAAC,CAAC,CAAA;YACzH,MAAM,cAAc,GAAG,GAAG,YAAY,GAAG,CAAA;YAEzC,IAAI,CAAC,qBAAqB,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE,CAAC;gBACtD,SAAQ;YACV,CAAC;YAED,MAAM,YAAY,GAAG,qBAAqB;iBACvC,KAAK,CAAC,cAAc,CAAC,MAAM,CAAC;iBAC5B,KAAK,CAAC,GAAG,CAAC;iBACV,MAAM,CAAC,OAAO,CAAC,CAAA;YAElB,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC9B,MAAM,wBAAwB,GAAG,MAAM,CAAC,OAAO,CAAC,qBAAqB,CAAC,kBAAkB,CAAC;qBACtF,IAAI,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,EAAE,EAAE,CAAC,WAAW,KAAK,YAAY,CAAC,CAAC,CAAC,CAAC,CAAA;gBAE7D,IAAI,wBAAwB,EAAE,CAAC;oBAC7B,OAAO;wBACL,WAAW,EAAE,wBAAwB,CAAC,CAAC,CAAC;wBACxC,UAAU,EAAE,wBAAwB,CAAC,CAAC,CAAC;wBACvC,SAAS;wBACT,YAAY;wBACZ,KAAK,EAAE,YAAY;qBACpB,CAAA;gBACH,CAAC;YACH,CAAC;YAED,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC9B,MAAM,oBAAoB,GAAG,MAAM,CAAC,OAAO,CAAC,qBAAqB,CAAC,cAAc,CAAC;qBAC9E,IAAI,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,EAAE,EAAE,CAAC,WAAW,KAAK,YAAY,CAAC,CAAC,CAAC,CAAC,CAAA;gBAE7D,IAAI,oBAAoB,EAAE,CAAC;oBACzB,OAAO;wBACL,WAAW,EAAE,oBAAoB,CAAC,CAAC,CAAC;wBACpC,QAAQ,EAAE,kBAAkB,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;wBAC7C,UAAU,EAAE,oBAAoB,CAAC,CAAC,CAAC;wBACnC,SAAS;wBACT,YAAY;wBACZ,KAAK,EAAE,QAAQ;qBAChB,CAAA;gBACH,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAA;AACb,CAAC;AAED;;;GAGG;AACH,SAAS,0CAA0C,CAAC,IAAI;IACtD,MAAM,gBAAgB,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAA;IAEjE,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChC,OAAO,gBAAgB,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAA;IAC7C,CAAC;IAED,OAAO,gBAAgB,CAAA;AACzB,CAAC","sourcesContent":["// @ts-check\n\nimport * as inflection from \"inflection\"\nimport FrontendModelBaseResource from \"../frontend-model-resource/base-resource.js\"\nimport restArgsError from \"../utils/rest-args-error.js\"\nimport {validateFrontendModelResourceCommandName} from \"./resource-config-validation.js\"\n\n/**\n * @param {import(\"../configuration-types.js\").BackendProjectConfiguration} backendProject - Backend project config.\n * @returns {Record<string, typeof FrontendModelBaseResource>} - Resource definitions keyed by model name.\n */\nexport function frontendModelResourcesForBackendProject(backendProject) {\n  const resources = backendProject.frontendModels\n\n  if (resources !== undefined) {\n    if (!resources || typeof resources !== \"object\") {\n      throw new Error(`Expected backend project frontendModels object but got: ${resources}`)\n    }\n\n    return resources\n  }\n\n  return {}\n}\n\n/**\n * @param {unknown} value - Candidate resource definition.\n * @returns {value is typeof FrontendModelBaseResource} - Whether value is a resource class.\n */\nexport function frontendModelResourceDefinitionIsClass(value) {\n  return typeof value === \"function\" && (value === FrontendModelBaseResource || value.prototype instanceof FrontendModelBaseResource)\n}\n\n/**\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {typeof FrontendModelBaseResource | null} - Resource class when definition is class-based.\n */\nexport function frontendModelResourceClassFromDefinition(resourceDefinition) {\n  return frontendModelResourceDefinitionIsClass(resourceDefinition) ? resourceDefinition : null\n}\n\n/**\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {import(\"../configuration-types.js\").NormalizedFrontendModelResourceConfiguration | null} - Normalized resource configuration.\n */\nexport function frontendModelResourceConfigurationFromDefinition(resourceDefinition) {\n  if (!frontendModelResourceDefinitionIsClass(resourceDefinition)) return null\n\n  return normalizeFrontendModelResourceConfiguration(resourceDefinition.resourceConfig())\n}\n\n/**\n * @param {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Raw resource configuration.\n * @returns {import(\"../configuration-types.js\").NormalizedFrontendModelResourceConfiguration} - Normalized resource configuration.\n */\nfunction normalizeFrontendModelResourceConfiguration(resourceConfiguration) {\n  const restArgs = /** @type {Record<string, any>} */ ({...resourceConfiguration})\n\n  for (const key of [\n    \"abilities\",\n    \"attributes\",\n    \"attachments\",\n    \"builtInCollectionCommands\",\n    \"builtInMemberCommands\",\n    \"collectionCommands\",\n    \"commands\",\n    \"memberCommands\",\n    \"modelName\",\n    \"primaryKey\",\n    \"relationships\",\n    \"server\"\n  ]) {\n    delete restArgs[key]\n  }\n\n  restArgsError(restArgs)\n\n  const normalizedCommands = normalizeFrontendModelResourceCommands(resourceConfiguration)\n\n  return {\n    ...resourceConfiguration,\n    abilities: normalizeFrontendModelResourceAbilities(resourceConfiguration.abilities),\n    builtInCollectionCommands: normalizedCommands.builtInCollectionCommands,\n    builtInMemberCommands: normalizedCommands.builtInMemberCommands,\n    collectionCommands: normalizedCommands.collectionCommands,\n    memberCommands: normalizedCommands.memberCommands\n  }\n}\n\n/**\n * @param {string[] | undefined} abilities - Resource abilities config (camelCase action list).\n * @returns {Record<string, string>} - Normalized abilities config.\n */\nfunction normalizeFrontendModelResourceAbilities(abilities) {\n  if (abilities === undefined) {\n    return defaultCrudAbilities()\n  }\n\n  if (!Array.isArray(abilities)) {\n    throw new Error(\"Resource abilities must be an array of action names. Object form is no longer supported.\")\n  }\n\n  if (abilities.includes(\"manage\")) {\n    return {\n      create: \"manage\",\n      destroy: \"manage\",\n      find: \"manage\",\n      index: \"manage\",\n      update: \"manage\"\n    }\n  }\n\n  /** @type {Record<string, string>} */\n  const normalized = {}\n\n  if (abilities.includes(\"create\")) normalized.create = \"create\"\n  if (abilities.includes(\"destroy\")) normalized.destroy = \"destroy\"\n  if (abilities.includes(\"read\")) {\n    normalized.find = \"read\"\n    normalized.index = \"read\"\n  }\n  if (abilities.includes(\"update\")) normalized.update = \"update\"\n\n  return normalized\n}\n\n/** @returns {Record<string, string>} - Default CRUD ability map. */\nfunction defaultCrudAbilities() {\n  return {\n    create: \"create\",\n    destroy: \"destroy\",\n    find: \"read\",\n    index: \"read\",\n    update: \"update\"\n  }\n}\n\n/**\n * @param {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Raw resource configuration.\n * @returns {{builtInCollectionCommands: Record<string, string>, builtInMemberCommands: Record<string, string>, collectionCommands: Record<string, string>, memberCommands: Record<string, string>}} - Normalized command configuration.\n */\nfunction normalizeFrontendModelResourceCommands(resourceConfiguration) {\n  const builtInCollectionCommands = resourceConfiguration.builtInCollectionCommands\n  const builtInMemberCommands = resourceConfiguration.builtInMemberCommands\n  const customCollectionCommands = resourceConfiguration.collectionCommands\n  const customMemberCommands = resourceConfiguration.memberCommands\n  const normalizedBuiltInCollectionCommands = normalizeFrontendModelBuiltInCommands({\n    commandDefaults: {\n      create: \"create\",\n      index: \"index\"\n    },\n    commandsConfig: builtInCollectionCommands,\n    modelName: \"CollectionCommand\"\n  })\n  const normalizedBuiltInMemberCommands = normalizeFrontendModelBuiltInCommands({\n    commandDefaults: {\n      attach: \"attach\",\n      destroy: \"destroy\",\n      download: \"download\",\n      find: \"find\",\n      update: \"update\",\n      url: \"url\"\n    },\n    commandsConfig: builtInMemberCommands,\n    modelName: \"MemberCommand\"\n  })\n\n  return {\n    builtInCollectionCommands: normalizedBuiltInCollectionCommands,\n    builtInMemberCommands: normalizedBuiltInMemberCommands,\n    collectionCommands: normalizeFrontendModelCustomCommands({commandsConfig: customCollectionCommands, modelName: \"CollectionCommand\"}),\n    memberCommands: normalizeFrontendModelCustomCommands({commandsConfig: customMemberCommands, modelName: \"MemberCommand\"})\n  }\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {Record<string, string>} args.commandDefaults - Built-in default command names.\n * @param {string[] | undefined} args.commandsConfig - Built-in commands config (camelCase command type list).\n * @param {string} args.modelName - Diagnostic model name.\n * @returns {Record<string, string>} - Normalized built-in command config.\n */\nfunction normalizeFrontendModelBuiltInCommands({commandDefaults, commandsConfig, modelName}) {\n  if (!commandsConfig) {\n    return commandDefaults\n  }\n\n  if (!Array.isArray(commandsConfig)) {\n    throw new Error(`${modelName} configuration must use the array form. Object form is no longer supported.`)\n  }\n\n  /** @type {Record<string, string>} */\n  const normalizedCommands = {}\n\n  for (const commandType of commandsConfig) {\n    const defaultCommandName = commandDefaults[commandType]\n\n    if (!defaultCommandName) {\n      throw new Error(`Unknown built-in frontend model command '${commandType}' for ${modelName}`)\n    }\n\n    normalizedCommands[commandType] = validateFrontendModelResourceCommandName({\n      commandName: defaultCommandName,\n      commandType: defaultCommandName,\n      modelName\n    })\n  }\n\n  return normalizedCommands\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {string[] | undefined} args.commandsConfig - Custom commands config (camelCase method-name list).\n * @param {string} args.modelName - Diagnostic model name.\n * @returns {Record<string, string>} - Normalized custom command config (camelCase method name → kebab-case command slug).\n */\nfunction normalizeFrontendModelCustomCommands({commandsConfig, modelName}) {\n  if (!commandsConfig) {\n    return {}\n  }\n\n  if (!Array.isArray(commandsConfig)) {\n    throw new Error(`${modelName} configuration must use the array form. Object form is no longer supported.`)\n  }\n\n  /** @type {Record<string, string>} */\n  const normalizedCommands = {}\n\n  for (const methodName of commandsConfig) {\n    const validatedMethodName = validateFrontendModelResourceCommandName({\n      commandName: methodName,\n      commandType: methodName,\n      modelName\n    })\n    const commandSlug = inflection.dasherize(inflection.underscore(validatedMethodName))\n\n    normalizedCommands[validatedMethodName] = commandSlug\n  }\n\n  return normalizedCommands\n}\n\n/**\n * @param {string} modelName - Model class name.\n * @param {unknown} resourceDefinition - Resource definition.\n * @returns {string} - Normalized resource path.\n */\nexport function frontendModelResourcePath(modelName, resourceDefinition) {\n  const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n  if (!resourceConfiguration) {\n    throw new Error(`Invalid frontend model resource definition for ${modelName}`)\n  }\n\n  return `/${inflection.dasherize(inflection.pluralize(inflection.underscore(modelName)))}`\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {string} args.commandName - Command path segment.\n * @param {string} args.modelName - Model class name.\n * @param {unknown} args.resourceDefinition - Resource definition.\n * @returns {\"destroy\" | \"find\" | \"index\" | \"create\" | \"update\" | \"attach\" | \"download\" | \"url\" | null} - Frontend action.\n */\nexport function frontendModelActionForCommand({commandName, modelName, resourceDefinition}) {\n  const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n  if (!resourceConfiguration) {\n    throw new Error(`Invalid frontend model resource definition for ${modelName}`)\n  }\n\n  for (const [action, configuredCommandName] of Object.entries({\n    ...resourceConfiguration.builtInCollectionCommands,\n    ...resourceConfiguration.builtInMemberCommands\n  })) {\n    if (configuredCommandName === undefined) continue\n\n    const validatedCommandName = validateFrontendModelResourceCommandName({\n      commandName: configuredCommandName,\n      commandType: /** @type {\"attach\" | \"create\" | \"destroy\" | \"download\" | \"find\" | \"index\" | \"update\" | \"url\"} */ (action),\n      modelName\n    })\n\n    if (commandName === validatedCommandName) {\n      return /** @type {\"attach\" | \"create\" | \"destroy\" | \"download\" | \"find\" | \"index\" | \"update\" | \"url\"} */ (action)\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {object} args - Arguments.\n * @param {import(\"../configuration-types.js\").BackendProjectConfiguration[]} args.backendProjects - Backend projects to scan.\n * @param {string} args.currentPath - Request path without query.\n * @returns {{commandName: string, memberId?: string, methodName: string, modelName: string, resourcePath: string, scope: \"collection\" | \"member\"} | null} - Matched custom command metadata.\n */\nexport function frontendModelCustomCommandForPath({backendProjects, currentPath}) {\n  const normalizedCurrentPath = normalizeFrontendModelResourcePathForMatch(currentPath)\n\n  for (const backendProject of backendProjects) {\n    const resources = frontendModelResourcesForBackendProject(backendProject)\n\n    for (const modelName in resources) {\n      const resourceDefinition = resources[modelName]\n      const resourceConfiguration = frontendModelResourceConfigurationFromDefinition(resourceDefinition)\n\n      if (!resourceConfiguration) {\n        continue\n      }\n\n      const resourcePath = normalizeFrontendModelResourcePathForMatch(frontendModelResourcePath(modelName, resourceDefinition))\n      const expectedPrefix = `${resourcePath}/`\n\n      if (!normalizedCurrentPath.startsWith(expectedPrefix)) {\n        continue\n      }\n\n      const pathSegments = normalizedCurrentPath\n        .slice(expectedPrefix.length)\n        .split(\"/\")\n        .filter(Boolean)\n\n      if (pathSegments.length === 1) {\n        const matchedCollectionCommand = Object.entries(resourceConfiguration.collectionCommands)\n          .find(([, commandName]) => commandName === pathSegments[0])\n\n        if (matchedCollectionCommand) {\n          return {\n            commandName: matchedCollectionCommand[1],\n            methodName: matchedCollectionCommand[0],\n            modelName,\n            resourcePath,\n            scope: \"collection\"\n          }\n        }\n      }\n\n      if (pathSegments.length === 2) {\n        const matchedMemberCommand = Object.entries(resourceConfiguration.memberCommands)\n          .find(([, commandName]) => commandName === pathSegments[1])\n\n        if (matchedMemberCommand) {\n          return {\n            commandName: matchedMemberCommand[1],\n            memberId: decodeURIComponent(pathSegments[0]),\n            methodName: matchedMemberCommand[0],\n            modelName,\n            resourcePath,\n            scope: \"member\"\n          }\n        }\n      }\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {string} path - Path value.\n * @returns {string} - Normalized path with leading slash and no trailing slash.\n */\nfunction normalizeFrontendModelResourcePathForMatch(path) {\n  const withLeadingSlash = path.startsWith(\"/\") ? path : `/${path}`\n\n  if (withLeadingSlash.length > 1) {\n    return withLeadingSlash.replace(/\\/+$/, \"\")\n  }\n\n  return withLeadingSlash\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.349",
6
+ "version": "1.0.350",
7
7
  "main": "build/index.js",
8
8
  "types": "build/index.d.ts",
9
9
  "files": [