velocious 1.0.349 → 1.0.351

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":"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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 === 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"]}
@@ -1 +1 @@
1
- {"version":3,"file":"ransack.d.ts","sourceRoot":"","sources":["../../../src/utils/ransack.js"],"names":[],"mappings":"AAoCA;;;;GAIG;AACH,mDAJW,iBAAiB,UACjB,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,gBAAgB,EAAE,CAyC9B;AA6UD;;;;GAIG;AAEH;;;;;;GAMG;AACH,6CAJW,iBAAiB,cACjB,MAAM,GACJ,WAAW,EAAE,CA2BzB;;;;;eApCa,MAAM;;;;eACN,KAAK,GAAG,MAAM;;+BA1Zf,MAAM,GAAG,KAAK,GAAG,IAAI,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI,GAAG,IAAI,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,MAAM,GAAG,OAAO;gCAIrG,cAAc,6BAA6B,EAAE,OAAO,GAAG,cAAc,4BAA4B,EAAE,OAAO;;;;;mBAKzG,MAAM;;;;UACN,MAAM,EAAE;;;;eACR,gBAAgB;;;;WAChB,GAAG"}
1
+ {"version":3,"file":"ransack.d.ts","sourceRoot":"","sources":["../../../src/utils/ransack.js"],"names":[],"mappings":"AAoCA;;;;GAIG;AACH,mDAJW,iBAAiB,UACjB,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,gBAAgB,EAAE,CAyC9B;AA0WD;;;;GAIG;AAEH;;;;;;GAMG;AACH,6CAJW,iBAAiB,cACjB,MAAM,GACJ,WAAW,EAAE,CA2BzB;;;;;eApCa,MAAM;;;;eACN,KAAK,GAAG,MAAM;;+BAvbf,MAAM,GAAG,KAAK,GAAG,IAAI,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI,GAAG,IAAI,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,MAAM,GAAG,OAAO;gCAIrG,cAAc,6BAA6B,EAAE,OAAO,GAAG,cAAc,4BAA4B,EAAE,OAAO;;;;;mBAKzG,MAAM;;;;UACN,MAAM,EAAE;;;;eACR,gBAAgB;;;;WAChB,GAAG"}
@@ -127,9 +127,9 @@ function findRelationshipPrefix({ modelClass, value }) {
127
127
  for (const relationshipName of Object.keys(relationshipEntries(modelClass))) {
128
128
  const relationship = relationshipEntries(modelClass)[relationshipName];
129
129
  for (const candidate of relationshipCandidates(relationshipName)) {
130
- if (!value.startsWith(`${candidate}_`))
130
+ const remainingValue = stripRelationshipCandidate(value, candidate);
131
+ if (remainingValue === null)
131
132
  continue;
132
- const remainingValue = value.slice(candidate.length + 1);
133
133
  if (remainingValue.length < 1)
134
134
  continue;
135
135
  if (bestMatch && candidate.length <= bestMatch.candidateLength)
@@ -150,6 +150,34 @@ function findRelationshipPrefix({ modelClass, value }) {
150
150
  targetModelClass: bestMatch.targetModelClass
151
151
  };
152
152
  }
153
+ /**
154
+ * Returns the portion of `value` after `candidate` when `candidate`
155
+ * sits at a relationship-path boundary, or null when there's no
156
+ * boundary match. Two boundary forms are accepted:
157
+ * - snake: `<candidate>_` followed by the rest of the path (e.g.
158
+ * `task_project_id` against candidate `task` returns `project_id`).
159
+ * - camel: `<candidate>` immediately followed by an uppercase letter,
160
+ * which marks a new word in camelCase (e.g. `taskProjectId` against
161
+ * candidate `task` returns `projectId` with the leading `P`
162
+ * lowercased so the remainder stays in caller-form for the next
163
+ * attribute / relationship match).
164
+ * @param {string} value - Remaining ransack path.
165
+ * @param {string} candidate - Relationship name candidate.
166
+ * @returns {string | null} - Remainder after the candidate, or null.
167
+ */
168
+ function stripRelationshipCandidate(value, candidate) {
169
+ if (value.startsWith(`${candidate}_`)) {
170
+ return value.slice(candidate.length + 1);
171
+ }
172
+ if (value.length <= candidate.length)
173
+ return null;
174
+ if (!value.startsWith(candidate))
175
+ return null;
176
+ const nextChar = value.charAt(candidate.length);
177
+ if (nextChar < "A" || nextChar > "Z")
178
+ return null;
179
+ return nextChar.toLowerCase() + value.slice(candidate.length + 1);
180
+ }
153
181
  /**
154
182
  * @param {string} relationshipName - Relationship name.
155
183
  * @returns {string[]} - Candidate tokens for matching.
@@ -396,4 +424,4 @@ function isPlainObject(value) {
396
424
  function uniqunize(values) {
397
425
  return Array.from(new Set(values));
398
426
  }
399
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ransack.js","sourceRoot":"","sources":["../../../src/utils/ransack.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport * as inflection from \"inflection\"\nimport {resolveFrontendModelClass} from \"../frontend-models/model-registry.js\"\n\n/**\n * @typedef {\"cont\" | \"end\" | \"eq\" | \"gt\" | \"gteq\" | \"in\" | \"lt\" | \"lteq\" | \"not_eq\" | \"not_in\" | \"null\" | \"start\"} RansackPredicate\n */\n\n/**\n * @typedef {typeof import(\"../database/record/index.js\").default | typeof import(\"../frontend-models/base.js\").default} RansackModelClass\n */\n\n/**\n * @typedef {object} RansackCondition\n * @property {string} attributeName - Resolved attribute name.\n * @property {string[]} path - Resolved relationship path.\n * @property {RansackPredicate} predicate - Parsed Ransack predicate.\n * @property {any} value - Normalized value.\n */\n\nconst supportedPredicates = [\n  \"not_in\",\n  \"not_eq\",\n  \"gteq\",\n  \"lteq\",\n  \"start\",\n  \"cont\",\n  \"null\",\n  \"end\",\n  \"eq\",\n  \"gt\",\n  \"lt\",\n  \"in\"\n]\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @param {Record<string, any>} params - Ransack-style params hash.\n * @returns {RansackCondition[]} - Normalized conditions.\n */\nexport function normalizeRansackParams(modelClass, params) {\n  if (!isPlainObject(params)) {\n    throw new Error(`ransack params must be a plain object, got: ${typeof params}`)\n  }\n\n  /** @type {RansackCondition[]} */\n  const normalized = []\n\n  for (const [key, rawValue] of Object.entries(params)) {\n    const parsedKey = parseRansackKey(key)\n\n    if (!parsedKey) {\n      throw new Error(`Unsupported ransack predicate in key: ${key}`)\n    }\n\n    const resolvedPath = resolveRansackPath({modelClass, value: parsedKey.pathValue})\n    const targetModelClass = modelClassAtPath({modelClass, path: resolvedPath.path})\n    const attributeName = resolveAttributeName({modelClass: targetModelClass, value: resolvedPath.attributeValue})\n\n    if (!attributeName) {\n      throw new Error(`Unknown ransack attribute \"${resolvedPath.attributeValue}\" for ${targetModelClass.name}`)\n    }\n\n    const value = normalizeRansackValue({\n      predicate: parsedKey.predicate,\n      value: rawValue\n    })\n\n    if (value === SKIP_RANSACK_CONDITION) continue\n\n    normalized.push({\n      attributeName,\n      path: resolvedPath.path,\n      predicate: parsedKey.predicate,\n      value\n    })\n  }\n\n  return normalized\n}\n\nconst SKIP_RANSACK_CONDITION = Symbol(\"skip-ransack-condition\")\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Root model class.\n * @param {string[]} args.path - Relationship path.\n * @returns {RansackModelClass} - Target model class.\n */\nfunction modelClassAtPath({modelClass, path}) {\n  let currentModelClass = modelClass\n\n  for (const relationshipName of path) {\n    const relationship = relationshipEntries(currentModelClass)[relationshipName]\n\n    if (!relationship) {\n      throw new Error(`Unknown ransack relationship \"${relationshipName}\" for ${currentModelClass.name}`)\n    }\n\n    currentModelClass = relationship.targetModelClass\n  }\n\n  return currentModelClass\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Current model class.\n * @param {string} args.value - Remaining path value.\n * @returns {{attributeValue: string, path: string[]}} - Resolved relationship path and remaining attribute value.\n */\nfunction resolveRansackPath({modelClass, value}) {\n  /** @type {string[]} */\n  const path = []\n  let currentModelClass = modelClass\n  let remainingValue = value\n\n  while (true) {\n    if (resolveAttributeName({modelClass: currentModelClass, value: remainingValue})) {\n      break\n    }\n\n    const match = findRelationshipPrefix({\n      modelClass: currentModelClass,\n      value: remainingValue\n    })\n\n    if (!match) break\n\n    path.push(match.relationshipName)\n    currentModelClass = match.targetModelClass\n    remainingValue = match.remainingValue\n  }\n\n  if (remainingValue.length < 1) {\n    throw new Error(`Invalid ransack key path: ${value}`)\n  }\n\n  return {\n    attributeValue: remainingValue,\n    path\n  }\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Current model class.\n * @param {string} args.value - Remaining value to match.\n * @returns {{relationshipName: string, remainingValue: string, targetModelClass: RansackModelClass} | null} - Matching relationship prefix.\n */\nfunction findRelationshipPrefix({modelClass, value}) {\n  let bestMatch = null\n\n  for (const relationshipName of Object.keys(relationshipEntries(modelClass))) {\n    const relationship = relationshipEntries(modelClass)[relationshipName]\n\n    for (const candidate of relationshipCandidates(relationshipName)) {\n      if (!value.startsWith(`${candidate}_`)) continue\n\n      const remainingValue = value.slice(candidate.length + 1)\n\n      if (remainingValue.length < 1) continue\n      if (bestMatch && candidate.length <= bestMatch.candidateLength) continue\n\n      bestMatch = {\n        candidateLength: candidate.length,\n        relationshipName,\n        remainingValue,\n        targetModelClass: relationship.targetModelClass\n      }\n    }\n  }\n\n  if (!bestMatch) return null\n\n  return {\n    relationshipName: bestMatch.relationshipName,\n    remainingValue: bestMatch.remainingValue,\n    targetModelClass: bestMatch.targetModelClass\n  }\n}\n\n/**\n * @param {string} relationshipName - Relationship name.\n * @returns {string[]} - Candidate tokens for matching.\n */\nfunction relationshipCandidates(relationshipName) {\n  return uniqunize([relationshipName, inflection.underscore(relationshipName)])\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Model class.\n * @param {string} args.value - Attribute candidate.\n * @returns {string | undefined} - Resolved attribute name.\n */\nfunction resolveAttributeName({modelClass, value}) {\n  for (const [attributeName, columnName] of Object.entries(attributeEntries(modelClass))) {\n    if (matchesAttributeValue({attributeName, columnName, value})) {\n      return attributeName\n    }\n  }\n\n  return undefined\n}\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @returns {Record<string, {targetModelClass: RansackModelClass}>} - Relationship entries keyed by name.\n */\nfunction relationshipEntries(modelClass) {\n  if (typeof /** @type {any} */ (modelClass).getRelationshipsMap === \"function\") {\n    /** @type {Record<string, {targetModelClass: RansackModelClass}>} */\n    const entries = {}\n    const relationshipsMap = /** @type {any} */ (modelClass).getRelationshipsMap()\n\n    for (const relationshipName of Object.keys(relationshipsMap)) {\n      const relationship = relationshipsMap[relationshipName]\n\n      if (typeof relationship.isPolymorphic === \"function\" && relationship.isPolymorphic()) continue\n\n      const targetModelClass = relationship.getTargetModelClass()\n\n      if (!targetModelClass) continue\n\n      entries[relationshipName] = {\n        targetModelClass\n      }\n    }\n\n    return entries\n  }\n\n  if (typeof /** @type {any} */ (modelClass).relationshipDefinitions === \"function\" &&\n    typeof /** @type {any} */ (modelClass).relationshipModelClasses === \"function\") {\n    /** @type {Record<string, {targetModelClass: RansackModelClass}>} */\n    const entries = {}\n    const definitions = /** @type {any} */ (modelClass).relationshipDefinitions()\n    const relationshipModelClasses = /** @type {any} */ (modelClass).relationshipModelClasses()\n\n    for (const relationshipName of Object.keys(definitions)) {\n      const targetModelClass = resolveFrontendModelClass(relationshipModelClasses[relationshipName])\n\n      if (!targetModelClass) continue\n\n      entries[relationshipName] = {targetModelClass}\n    }\n\n    return entries\n  }\n\n  return {}\n}\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @returns {Record<string, string>} - Attribute-to-column entries keyed by attribute name.\n */\nfunction attributeEntries(modelClass) {\n  if (typeof /** @type {any} */ (modelClass).getAttributeNameToColumnNameMap === \"function\") {\n    return /** @type {Record<string, string>} */ ((/** @type {any} */ (modelClass).getAttributeNameToColumnNameMap()))\n  }\n\n  const resourceConfig = typeof /** @type {any} */ (modelClass).resourceConfig === \"function\"\n    ? /** @type {any} */ (modelClass).resourceConfig()\n    : {}\n  const attributes = resourceConfig.attributes\n  /** @type {Record<string, string>} */\n  const entries = {}\n\n  if (Array.isArray(attributes)) {\n    for (const attributeName of attributes) {\n      if (typeof attributeName !== \"string\") continue\n\n      entries[attributeName] = attributeName\n    }\n  } else if (isPlainObject(attributes)) {\n    for (const attributeName of Object.keys(attributes)) {\n      entries[attributeName] = attributeName\n    }\n  }\n\n  return entries\n}\n\n/**\n * @param {object} args - Options.\n * @param {string} args.attributeName - Attribute name.\n * @param {string} args.columnName - Column name.\n * @param {string} args.value - Candidate value.\n * @returns {boolean} - Whether the candidate resolves to the attribute.\n */\nfunction matchesAttributeValue({attributeName, columnName, value}) {\n  return uniqunize([\n    attributeName,\n    columnName,\n    inflection.underscore(attributeName),\n    inflection.underscore(columnName)\n  ]).includes(value)\n}\n\n/**\n * @param {string} key - Ransack key.\n * @returns {{pathValue: string, predicate: RansackPredicate} | null} - Parsed key.\n */\nfunction parseRansackKey(key) {\n  for (const predicate of supportedPredicates) {\n    const suffix = `_${predicate}`\n    if (!key.endsWith(suffix)) continue\n\n    const pathValue = key.slice(0, key.length - suffix.length)\n\n    if (pathValue.length < 1) {\n      throw new Error(`Invalid ransack key: ${key}`)\n    }\n\n    return {\n      pathValue,\n      predicate: /** @type {RansackPredicate} */ (predicate)\n    }\n  }\n\n  for (const predicate of supportedPredicates) {\n    const camelSuffix = snakeToCamelSuffix(predicate)\n\n    if (!key.endsWith(camelSuffix)) continue\n\n    const pathValue = key.slice(0, key.length - camelSuffix.length)\n\n    if (pathValue.length < 1) {\n      throw new Error(`Invalid ransack key: ${key}`)\n    }\n\n    return {\n      pathValue,\n      predicate: /** @type {RansackPredicate} */ (predicate)\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {string} value - Snake-case predicate.\n * @returns {string} - CamelCase predicate suffix used in ransack keys.\n */\nfunction snakeToCamelSuffix(value) {\n  const segments = value.split(\"_\")\n\n  return segments.map((segment) => segment.charAt(0).toUpperCase() + segment.slice(1)).join(\"\")\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackPredicate} args.predicate - Parsed predicate.\n * @param {any} args.value - Raw value.\n * @returns {any} - Normalized value.\n */\nfunction normalizeRansackValue({predicate, value}) {\n  if (predicate === \"null\") {\n    const booleanValue = normalizeRansackBoolean(value)\n\n    if (booleanValue === null) return SKIP_RANSACK_CONDITION\n\n    return booleanValue\n  }\n\n  if (predicate === \"in\" || predicate === \"not_in\") {\n    const normalizedArray = normalizeRansackArray(value)\n\n    if (normalizedArray.length < 1) return SKIP_RANSACK_CONDITION\n\n    return normalizedArray\n  }\n\n  if (value === undefined || value === null) return SKIP_RANSACK_CONDITION\n  if (typeof value === \"string\" && value.length < 1) return SKIP_RANSACK_CONDITION\n\n  return value\n}\n\n/**\n * @param {unknown} value - Candidate boolean.\n * @returns {boolean | null} - Normalized boolean or null when blank.\n */\nfunction normalizeRansackBoolean(value) {\n  if (value === true || value === false) return value\n  if (value === 1 || value === \"1\" || value === \"true\") return true\n  if (value === 0 || value === \"0\" || value === \"false\") return false\n  if (value === undefined || value === null || value === \"\") return null\n\n  throw new Error(`Invalid ransack boolean value: ${String(value)}`)\n}\n\n/**\n * @param {unknown} value - Candidate array-ish value.\n * @returns {any[]} - Normalized array values.\n */\nfunction normalizeRansackArray(value) {\n  if (Array.isArray(value)) {\n    return value.filter((entry) => entry !== undefined && entry !== null && entry !== \"\")\n  }\n\n  if (typeof value === \"string\") {\n    return value.split(\",\").map((entry) => entry.trim()).filter((entry) => entry.length > 0)\n  }\n\n  if (value === undefined || value === null || value === \"\") return []\n\n  return [value]\n}\n\n/**\n * @typedef {object} RansackSort\n * @property {string} attribute - Resolved attribute name.\n * @property {\"asc\" | \"desc\"} direction - Sort direction.\n */\n\n/**\n * Parses and validates a ransack `s` sort string against model attributes.\n *\n * @param {RansackModelClass} modelClass - Model class for attribute validation.\n * @param {string} sortString - Ransack sort string (e.g., \"name asc\" or \"name asc, createdAt desc\").\n * @returns {RansackSort[]} - Validated sort definitions.\n */\nexport function parseRansackSort(modelClass, sortString) {\n  const segments = sortString.split(\",\").map((segment) => segment.trim()).filter((segment) => segment.length > 0)\n\n  /** @type {RansackSort[]} */\n  const sorts = []\n\n  for (const segment of segments) {\n    const parts = segment.split(/\\s+/)\n    const columnCandidate = parts[0]\n    const directionCandidate = parts.length > 1 ? parts[1].toLowerCase() : \"asc\"\n\n    if (directionCandidate !== \"asc\" && directionCandidate !== \"desc\") {\n      throw new Error(`Invalid ransack sort direction \"${directionCandidate}\" in: ${segment}`)\n    }\n\n    const resolvedAttribute = resolveAttributeName({modelClass, value: columnCandidate})\n\n    if (!resolvedAttribute) {\n      throw new Error(`Unknown ransack sort attribute \"${columnCandidate}\" for ${modelClass.name}`)\n    }\n\n    sorts.push({attribute: resolvedAttribute, direction: directionCandidate})\n  }\n\n  return sorts\n}\n\n/**\n * @param {unknown} value - Candidate object.\n * @returns {value is Record<string, any>} - Whether this 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 * @param {string[]} values - Input values.\n * @returns {string[]} - Unique values in original order.\n */\nfunction uniqunize(values) {\n  return Array.from(new Set(values))\n}\n"]}
427
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ransack.js","sourceRoot":"","sources":["../../../src/utils/ransack.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport * as inflection from \"inflection\"\nimport {resolveFrontendModelClass} from \"../frontend-models/model-registry.js\"\n\n/**\n * @typedef {\"cont\" | \"end\" | \"eq\" | \"gt\" | \"gteq\" | \"in\" | \"lt\" | \"lteq\" | \"not_eq\" | \"not_in\" | \"null\" | \"start\"} RansackPredicate\n */\n\n/**\n * @typedef {typeof import(\"../database/record/index.js\").default | typeof import(\"../frontend-models/base.js\").default} RansackModelClass\n */\n\n/**\n * @typedef {object} RansackCondition\n * @property {string} attributeName - Resolved attribute name.\n * @property {string[]} path - Resolved relationship path.\n * @property {RansackPredicate} predicate - Parsed Ransack predicate.\n * @property {any} value - Normalized value.\n */\n\nconst supportedPredicates = [\n  \"not_in\",\n  \"not_eq\",\n  \"gteq\",\n  \"lteq\",\n  \"start\",\n  \"cont\",\n  \"null\",\n  \"end\",\n  \"eq\",\n  \"gt\",\n  \"lt\",\n  \"in\"\n]\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @param {Record<string, any>} params - Ransack-style params hash.\n * @returns {RansackCondition[]} - Normalized conditions.\n */\nexport function normalizeRansackParams(modelClass, params) {\n  if (!isPlainObject(params)) {\n    throw new Error(`ransack params must be a plain object, got: ${typeof params}`)\n  }\n\n  /** @type {RansackCondition[]} */\n  const normalized = []\n\n  for (const [key, rawValue] of Object.entries(params)) {\n    const parsedKey = parseRansackKey(key)\n\n    if (!parsedKey) {\n      throw new Error(`Unsupported ransack predicate in key: ${key}`)\n    }\n\n    const resolvedPath = resolveRansackPath({modelClass, value: parsedKey.pathValue})\n    const targetModelClass = modelClassAtPath({modelClass, path: resolvedPath.path})\n    const attributeName = resolveAttributeName({modelClass: targetModelClass, value: resolvedPath.attributeValue})\n\n    if (!attributeName) {\n      throw new Error(`Unknown ransack attribute \"${resolvedPath.attributeValue}\" for ${targetModelClass.name}`)\n    }\n\n    const value = normalizeRansackValue({\n      predicate: parsedKey.predicate,\n      value: rawValue\n    })\n\n    if (value === SKIP_RANSACK_CONDITION) continue\n\n    normalized.push({\n      attributeName,\n      path: resolvedPath.path,\n      predicate: parsedKey.predicate,\n      value\n    })\n  }\n\n  return normalized\n}\n\nconst SKIP_RANSACK_CONDITION = Symbol(\"skip-ransack-condition\")\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Root model class.\n * @param {string[]} args.path - Relationship path.\n * @returns {RansackModelClass} - Target model class.\n */\nfunction modelClassAtPath({modelClass, path}) {\n  let currentModelClass = modelClass\n\n  for (const relationshipName of path) {\n    const relationship = relationshipEntries(currentModelClass)[relationshipName]\n\n    if (!relationship) {\n      throw new Error(`Unknown ransack relationship \"${relationshipName}\" for ${currentModelClass.name}`)\n    }\n\n    currentModelClass = relationship.targetModelClass\n  }\n\n  return currentModelClass\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Current model class.\n * @param {string} args.value - Remaining path value.\n * @returns {{attributeValue: string, path: string[]}} - Resolved relationship path and remaining attribute value.\n */\nfunction resolveRansackPath({modelClass, value}) {\n  /** @type {string[]} */\n  const path = []\n  let currentModelClass = modelClass\n  let remainingValue = value\n\n  while (true) {\n    if (resolveAttributeName({modelClass: currentModelClass, value: remainingValue})) {\n      break\n    }\n\n    const match = findRelationshipPrefix({\n      modelClass: currentModelClass,\n      value: remainingValue\n    })\n\n    if (!match) break\n\n    path.push(match.relationshipName)\n    currentModelClass = match.targetModelClass\n    remainingValue = match.remainingValue\n  }\n\n  if (remainingValue.length < 1) {\n    throw new Error(`Invalid ransack key path: ${value}`)\n  }\n\n  return {\n    attributeValue: remainingValue,\n    path\n  }\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Current model class.\n * @param {string} args.value - Remaining value to match.\n * @returns {{relationshipName: string, remainingValue: string, targetModelClass: RansackModelClass} | null} - Matching relationship prefix.\n */\nfunction findRelationshipPrefix({modelClass, value}) {\n  let bestMatch = null\n\n  for (const relationshipName of Object.keys(relationshipEntries(modelClass))) {\n    const relationship = relationshipEntries(modelClass)[relationshipName]\n\n    for (const candidate of relationshipCandidates(relationshipName)) {\n      const remainingValue = stripRelationshipCandidate(value, candidate)\n\n      if (remainingValue === null) continue\n      if (remainingValue.length < 1) continue\n      if (bestMatch && candidate.length <= bestMatch.candidateLength) continue\n\n      bestMatch = {\n        candidateLength: candidate.length,\n        relationshipName,\n        remainingValue,\n        targetModelClass: relationship.targetModelClass\n      }\n    }\n  }\n\n  if (!bestMatch) return null\n\n  return {\n    relationshipName: bestMatch.relationshipName,\n    remainingValue: bestMatch.remainingValue,\n    targetModelClass: bestMatch.targetModelClass\n  }\n}\n\n/**\n * Returns the portion of `value` after `candidate` when `candidate`\n * sits at a relationship-path boundary, or null when there's no\n * boundary match. Two boundary forms are accepted:\n * - snake: `<candidate>_` followed by the rest of the path (e.g.\n *   `task_project_id` against candidate `task` returns `project_id`).\n * - camel: `<candidate>` immediately followed by an uppercase letter,\n *   which marks a new word in camelCase (e.g. `taskProjectId` against\n *   candidate `task` returns `projectId` with the leading `P`\n *   lowercased so the remainder stays in caller-form for the next\n *   attribute / relationship match).\n * @param {string} value - Remaining ransack path.\n * @param {string} candidate - Relationship name candidate.\n * @returns {string | null} - Remainder after the candidate, or null.\n */\nfunction stripRelationshipCandidate(value, candidate) {\n  if (value.startsWith(`${candidate}_`)) {\n    return value.slice(candidate.length + 1)\n  }\n\n  if (value.length <= candidate.length) return null\n  if (!value.startsWith(candidate)) return null\n\n  const nextChar = value.charAt(candidate.length)\n\n  if (nextChar < \"A\" || nextChar > \"Z\") return null\n\n  return nextChar.toLowerCase() + value.slice(candidate.length + 1)\n}\n\n/**\n * @param {string} relationshipName - Relationship name.\n * @returns {string[]} - Candidate tokens for matching.\n */\nfunction relationshipCandidates(relationshipName) {\n  return uniqunize([relationshipName, inflection.underscore(relationshipName)])\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackModelClass} args.modelClass - Model class.\n * @param {string} args.value - Attribute candidate.\n * @returns {string | undefined} - Resolved attribute name.\n */\nfunction resolveAttributeName({modelClass, value}) {\n  for (const [attributeName, columnName] of Object.entries(attributeEntries(modelClass))) {\n    if (matchesAttributeValue({attributeName, columnName, value})) {\n      return attributeName\n    }\n  }\n\n  return undefined\n}\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @returns {Record<string, {targetModelClass: RansackModelClass}>} - Relationship entries keyed by name.\n */\nfunction relationshipEntries(modelClass) {\n  if (typeof /** @type {any} */ (modelClass).getRelationshipsMap === \"function\") {\n    /** @type {Record<string, {targetModelClass: RansackModelClass}>} */\n    const entries = {}\n    const relationshipsMap = /** @type {any} */ (modelClass).getRelationshipsMap()\n\n    for (const relationshipName of Object.keys(relationshipsMap)) {\n      const relationship = relationshipsMap[relationshipName]\n\n      if (typeof relationship.isPolymorphic === \"function\" && relationship.isPolymorphic()) continue\n\n      const targetModelClass = relationship.getTargetModelClass()\n\n      if (!targetModelClass) continue\n\n      entries[relationshipName] = {\n        targetModelClass\n      }\n    }\n\n    return entries\n  }\n\n  if (typeof /** @type {any} */ (modelClass).relationshipDefinitions === \"function\" &&\n    typeof /** @type {any} */ (modelClass).relationshipModelClasses === \"function\") {\n    /** @type {Record<string, {targetModelClass: RansackModelClass}>} */\n    const entries = {}\n    const definitions = /** @type {any} */ (modelClass).relationshipDefinitions()\n    const relationshipModelClasses = /** @type {any} */ (modelClass).relationshipModelClasses()\n\n    for (const relationshipName of Object.keys(definitions)) {\n      const targetModelClass = resolveFrontendModelClass(relationshipModelClasses[relationshipName])\n\n      if (!targetModelClass) continue\n\n      entries[relationshipName] = {targetModelClass}\n    }\n\n    return entries\n  }\n\n  return {}\n}\n\n/**\n * @param {RansackModelClass} modelClass - Model class.\n * @returns {Record<string, string>} - Attribute-to-column entries keyed by attribute name.\n */\nfunction attributeEntries(modelClass) {\n  if (typeof /** @type {any} */ (modelClass).getAttributeNameToColumnNameMap === \"function\") {\n    return /** @type {Record<string, string>} */ ((/** @type {any} */ (modelClass).getAttributeNameToColumnNameMap()))\n  }\n\n  const resourceConfig = typeof /** @type {any} */ (modelClass).resourceConfig === \"function\"\n    ? /** @type {any} */ (modelClass).resourceConfig()\n    : {}\n  const attributes = resourceConfig.attributes\n  /** @type {Record<string, string>} */\n  const entries = {}\n\n  if (Array.isArray(attributes)) {\n    for (const attributeName of attributes) {\n      if (typeof attributeName !== \"string\") continue\n\n      entries[attributeName] = attributeName\n    }\n  } else if (isPlainObject(attributes)) {\n    for (const attributeName of Object.keys(attributes)) {\n      entries[attributeName] = attributeName\n    }\n  }\n\n  return entries\n}\n\n/**\n * @param {object} args - Options.\n * @param {string} args.attributeName - Attribute name.\n * @param {string} args.columnName - Column name.\n * @param {string} args.value - Candidate value.\n * @returns {boolean} - Whether the candidate resolves to the attribute.\n */\nfunction matchesAttributeValue({attributeName, columnName, value}) {\n  return uniqunize([\n    attributeName,\n    columnName,\n    inflection.underscore(attributeName),\n    inflection.underscore(columnName)\n  ]).includes(value)\n}\n\n/**\n * @param {string} key - Ransack key.\n * @returns {{pathValue: string, predicate: RansackPredicate} | null} - Parsed key.\n */\nfunction parseRansackKey(key) {\n  for (const predicate of supportedPredicates) {\n    const suffix = `_${predicate}`\n    if (!key.endsWith(suffix)) continue\n\n    const pathValue = key.slice(0, key.length - suffix.length)\n\n    if (pathValue.length < 1) {\n      throw new Error(`Invalid ransack key: ${key}`)\n    }\n\n    return {\n      pathValue,\n      predicate: /** @type {RansackPredicate} */ (predicate)\n    }\n  }\n\n  for (const predicate of supportedPredicates) {\n    const camelSuffix = snakeToCamelSuffix(predicate)\n\n    if (!key.endsWith(camelSuffix)) continue\n\n    const pathValue = key.slice(0, key.length - camelSuffix.length)\n\n    if (pathValue.length < 1) {\n      throw new Error(`Invalid ransack key: ${key}`)\n    }\n\n    return {\n      pathValue,\n      predicate: /** @type {RansackPredicate} */ (predicate)\n    }\n  }\n\n  return null\n}\n\n/**\n * @param {string} value - Snake-case predicate.\n * @returns {string} - CamelCase predicate suffix used in ransack keys.\n */\nfunction snakeToCamelSuffix(value) {\n  const segments = value.split(\"_\")\n\n  return segments.map((segment) => segment.charAt(0).toUpperCase() + segment.slice(1)).join(\"\")\n}\n\n/**\n * @param {object} args - Options.\n * @param {RansackPredicate} args.predicate - Parsed predicate.\n * @param {any} args.value - Raw value.\n * @returns {any} - Normalized value.\n */\nfunction normalizeRansackValue({predicate, value}) {\n  if (predicate === \"null\") {\n    const booleanValue = normalizeRansackBoolean(value)\n\n    if (booleanValue === null) return SKIP_RANSACK_CONDITION\n\n    return booleanValue\n  }\n\n  if (predicate === \"in\" || predicate === \"not_in\") {\n    const normalizedArray = normalizeRansackArray(value)\n\n    if (normalizedArray.length < 1) return SKIP_RANSACK_CONDITION\n\n    return normalizedArray\n  }\n\n  if (value === undefined || value === null) return SKIP_RANSACK_CONDITION\n  if (typeof value === \"string\" && value.length < 1) return SKIP_RANSACK_CONDITION\n\n  return value\n}\n\n/**\n * @param {unknown} value - Candidate boolean.\n * @returns {boolean | null} - Normalized boolean or null when blank.\n */\nfunction normalizeRansackBoolean(value) {\n  if (value === true || value === false) return value\n  if (value === 1 || value === \"1\" || value === \"true\") return true\n  if (value === 0 || value === \"0\" || value === \"false\") return false\n  if (value === undefined || value === null || value === \"\") return null\n\n  throw new Error(`Invalid ransack boolean value: ${String(value)}`)\n}\n\n/**\n * @param {unknown} value - Candidate array-ish value.\n * @returns {any[]} - Normalized array values.\n */\nfunction normalizeRansackArray(value) {\n  if (Array.isArray(value)) {\n    return value.filter((entry) => entry !== undefined && entry !== null && entry !== \"\")\n  }\n\n  if (typeof value === \"string\") {\n    return value.split(\",\").map((entry) => entry.trim()).filter((entry) => entry.length > 0)\n  }\n\n  if (value === undefined || value === null || value === \"\") return []\n\n  return [value]\n}\n\n/**\n * @typedef {object} RansackSort\n * @property {string} attribute - Resolved attribute name.\n * @property {\"asc\" | \"desc\"} direction - Sort direction.\n */\n\n/**\n * Parses and validates a ransack `s` sort string against model attributes.\n *\n * @param {RansackModelClass} modelClass - Model class for attribute validation.\n * @param {string} sortString - Ransack sort string (e.g., \"name asc\" or \"name asc, createdAt desc\").\n * @returns {RansackSort[]} - Validated sort definitions.\n */\nexport function parseRansackSort(modelClass, sortString) {\n  const segments = sortString.split(\",\").map((segment) => segment.trim()).filter((segment) => segment.length > 0)\n\n  /** @type {RansackSort[]} */\n  const sorts = []\n\n  for (const segment of segments) {\n    const parts = segment.split(/\\s+/)\n    const columnCandidate = parts[0]\n    const directionCandidate = parts.length > 1 ? parts[1].toLowerCase() : \"asc\"\n\n    if (directionCandidate !== \"asc\" && directionCandidate !== \"desc\") {\n      throw new Error(`Invalid ransack sort direction \"${directionCandidate}\" in: ${segment}`)\n    }\n\n    const resolvedAttribute = resolveAttributeName({modelClass, value: columnCandidate})\n\n    if (!resolvedAttribute) {\n      throw new Error(`Unknown ransack sort attribute \"${columnCandidate}\" for ${modelClass.name}`)\n    }\n\n    sorts.push({attribute: resolvedAttribute, direction: directionCandidate})\n  }\n\n  return sorts\n}\n\n/**\n * @param {unknown} value - Candidate object.\n * @returns {value is Record<string, any>} - Whether this 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 * @param {string[]} values - Input values.\n * @returns {string[]} - Unique values in original order.\n */\nfunction uniqunize(values) {\n  return Array.from(new Set(values))\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.351",
7
7
  "main": "build/index.js",
8
8
  "types": "build/index.d.ts",
9
9
  "files": [