velocious 1.0.350 → 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.
@@ -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":"AAAA,YAAY;AAEZ,OAAO,KAAK,UAAU,MAAM,YAAY,CAAA;AACxC,OAAO,EAAC,yBAAyB,EAAC,MAAM,sCAAsC,CAAA;AAE9E;;GAEG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH,MAAM,mBAAmB,GAAG;IAC1B,QAAQ;IACR,QAAQ;IACR,MAAM;IACN,MAAM;IACN,OAAO;IACP,MAAM;IACN,MAAM;IACN,KAAK;IACL,IAAI;IACJ,IAAI;IACJ,IAAI;IACJ,IAAI;CACL,CAAA;AAED;;;;GAIG;AACH,MAAM,UAAU,sBAAsB,CAAC,UAAU,EAAE,MAAM;IACvD,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CAAC,+CAA+C,OAAO,MAAM,EAAE,CAAC,CAAA;IACjF,CAAC;IAED,iCAAiC;IACjC,MAAM,UAAU,GAAG,EAAE,CAAA;IAErB,KAAK,MAAM,CAAC,GAAG,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;QACrD,MAAM,SAAS,GAAG,eAAe,CAAC,GAAG,CAAC,CAAA;QAEtC,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,MAAM,IAAI,KAAK,CAAC,yCAAyC,GAAG,EAAE,CAAC,CAAA;QACjE,CAAC;QAED,MAAM,YAAY,GAAG,kBAAkB,CAAC,EAAC,UAAU,EAAE,KAAK,EAAE,SAAS,CAAC,SAAS,EAAC,CAAC,CAAA;QACjF,MAAM,gBAAgB,GAAG,gBAAgB,CAAC,EAAC,UAAU,EAAE,IAAI,EAAE,YAAY,CAAC,IAAI,EAAC,CAAC,CAAA;QAChF,MAAM,aAAa,GAAG,oBAAoB,CAAC,EAAC,UAAU,EAAE,gBAAgB,EAAE,KAAK,EAAE,YAAY,CAAC,cAAc,EAAC,CAAC,CAAA;QAE9G,IAAI,CAAC,aAAa,EAAE,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,8BAA8B,YAAY,CAAC,cAAc,SAAS,gBAAgB,CAAC,IAAI,EAAE,CAAC,CAAA;QAC5G,CAAC;QAED,MAAM,KAAK,GAAG,qBAAqB,CAAC;YAClC,SAAS,EAAE,SAAS,CAAC,SAAS;YAC9B,KAAK,EAAE,QAAQ;SAChB,CAAC,CAAA;QAEF,IAAI,KAAK,KAAK,sBAAsB;YAAE,SAAQ;QAE9C,UAAU,CAAC,IAAI,CAAC;YACd,aAAa;YACb,IAAI,EAAE,YAAY,CAAC,IAAI;YACvB,SAAS,EAAE,SAAS,CAAC,SAAS;YAC9B,KAAK;SACN,CAAC,CAAA;IACJ,CAAC;IAED,OAAO,UAAU,CAAA;AACnB,CAAC;AAED,MAAM,sBAAsB,GAAG,MAAM,CAAC,wBAAwB,CAAC,CAAA;AAE/D;;;;;GAKG;AACH,SAAS,gBAAgB,CAAC,EAAC,UAAU,EAAE,IAAI,EAAC;IAC1C,IAAI,iBAAiB,GAAG,UAAU,CAAA;IAElC,KAAK,MAAM,gBAAgB,IAAI,IAAI,EAAE,CAAC;QACpC,MAAM,YAAY,GAAG,mBAAmB,CAAC,iBAAiB,CAAC,CAAC,gBAAgB,CAAC,CAAA;QAE7E,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,IAAI,KAAK,CAAC,iCAAiC,gBAAgB,SAAS,iBAAiB,CAAC,IAAI,EAAE,CAAC,CAAA;QACrG,CAAC;QAED,iBAAiB,GAAG,YAAY,CAAC,gBAAgB,CAAA;IACnD,CAAC;IAED,OAAO,iBAAiB,CAAA;AAC1B,CAAC;AAED;;;;;GAKG;AACH,SAAS,kBAAkB,CAAC,EAAC,UAAU,EAAE,KAAK,EAAC;IAC7C,uBAAuB;IACvB,MAAM,IAAI,GAAG,EAAE,CAAA;IACf,IAAI,iBAAiB,GAAG,UAAU,CAAA;IAClC,IAAI,cAAc,GAAG,KAAK,CAAA;IAE1B,OAAO,IAAI,EAAE,CAAC;QACZ,IAAI,oBAAoB,CAAC,EAAC,UAAU,EAAE,iBAAiB,EAAE,KAAK,EAAE,cAAc,EAAC,CAAC,EAAE,CAAC;YACjF,MAAK;QACP,CAAC;QAED,MAAM,KAAK,GAAG,sBAAsB,CAAC;YACnC,UAAU,EAAE,iBAAiB;YAC7B,KAAK,EAAE,cAAc;SACtB,CAAC,CAAA;QAEF,IAAI,CAAC,KAAK;YAAE,MAAK;QAEjB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAA;QACjC,iBAAiB,GAAG,KAAK,CAAC,gBAAgB,CAAA;QAC1C,cAAc,GAAG,KAAK,CAAC,cAAc,CAAA;IACvC,CAAC;IAED,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CAAC,6BAA6B,KAAK,EAAE,CAAC,CAAA;IACvD,CAAC;IAED,OAAO;QACL,cAAc,EAAE,cAAc;QAC9B,IAAI;KACL,CAAA;AACH,CAAC;AAED;;;;;GAKG;AACH,SAAS,sBAAsB,CAAC,EAAC,UAAU,EAAE,KAAK,EAAC;IACjD,IAAI,SAAS,GAAG,IAAI,CAAA;IAEpB,KAAK,MAAM,gBAAgB,IAAI,MAAM,CAAC,IAAI,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAC,EAAE,CAAC;QAC5E,MAAM,YAAY,GAAG,mBAAmB,CAAC,UAAU,CAAC,CAAC,gBAAgB,CAAC,CAAA;QAEtE,KAAK,MAAM,SAAS,IAAI,sBAAsB,CAAC,gBAAgB,CAAC,EAAE,CAAC;YACjE,MAAM,cAAc,GAAG,0BAA0B,CAAC,KAAK,EAAE,SAAS,CAAC,CAAA;YAEnE,IAAI,cAAc,KAAK,IAAI;gBAAE,SAAQ;YACrC,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC;gBAAE,SAAQ;YACvC,IAAI,SAAS,IAAI,SAAS,CAAC,MAAM,IAAI,SAAS,CAAC,eAAe;gBAAE,SAAQ;YAExE,SAAS,GAAG;gBACV,eAAe,EAAE,SAAS,CAAC,MAAM;gBACjC,gBAAgB;gBAChB,cAAc;gBACd,gBAAgB,EAAE,YAAY,CAAC,gBAAgB;aAChD,CAAA;QACH,CAAC;IACH,CAAC;IAED,IAAI,CAAC,SAAS;QAAE,OAAO,IAAI,CAAA;IAE3B,OAAO;QACL,gBAAgB,EAAE,SAAS,CAAC,gBAAgB;QAC5C,cAAc,EAAE,SAAS,CAAC,cAAc;QACxC,gBAAgB,EAAE,SAAS,CAAC,gBAAgB;KAC7C,CAAA;AACH,CAAC;AAED;;;;;;;;;;;;;;GAcG;AACH,SAAS,0BAA0B,CAAC,KAAK,EAAE,SAAS;IAClD,IAAI,KAAK,CAAC,UAAU,CAAC,GAAG,SAAS,GAAG,CAAC,EAAE,CAAC;QACtC,OAAO,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC,CAAA;IAC1C,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,IAAI,SAAS,CAAC,MAAM;QAAE,OAAO,IAAI,CAAA;IACjD,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,SAAS,CAAC;QAAE,OAAO,IAAI,CAAA;IAE7C,MAAM,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAA;IAE/C,IAAI,QAAQ,GAAG,GAAG,IAAI,QAAQ,GAAG,GAAG;QAAE,OAAO,IAAI,CAAA;IAEjD,OAAO,QAAQ,CAAC,WAAW,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC,CAAA;AACnE,CAAC;AAED;;;GAGG;AACH,SAAS,sBAAsB,CAAC,gBAAgB;IAC9C,OAAO,SAAS,CAAC,CAAC,gBAAgB,EAAE,UAAU,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAA;AAC/E,CAAC;AAED;;;;;GAKG;AACH,SAAS,oBAAoB,CAAC,EAAC,UAAU,EAAE,KAAK,EAAC;IAC/C,KAAK,MAAM,CAAC,aAAa,EAAE,UAAU,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC,EAAE,CAAC;QACvF,IAAI,qBAAqB,CAAC,EAAC,aAAa,EAAE,UAAU,EAAE,KAAK,EAAC,CAAC,EAAE,CAAC;YAC9D,OAAO,aAAa,CAAA;QACtB,CAAC;IACH,CAAC;IAED,OAAO,SAAS,CAAA;AAClB,CAAC;AAED;;;GAGG;AACH,SAAS,mBAAmB,CAAC,UAAU;IACrC,IAAI,OAAO,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,mBAAmB,KAAK,UAAU,EAAE,CAAC;QAC9E,oEAAoE;QACpE,MAAM,OAAO,GAAG,EAAE,CAAA;QAClB,MAAM,gBAAgB,GAAG,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,mBAAmB,EAAE,CAAA;QAE9E,KAAK,MAAM,gBAAgB,IAAI,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC7D,MAAM,YAAY,GAAG,gBAAgB,CAAC,gBAAgB,CAAC,CAAA;YAEvD,IAAI,OAAO,YAAY,CAAC,aAAa,KAAK,UAAU,IAAI,YAAY,CAAC,aAAa,EAAE;gBAAE,SAAQ;YAE9F,MAAM,gBAAgB,GAAG,YAAY,CAAC,mBAAmB,EAAE,CAAA;YAE3D,IAAI,CAAC,gBAAgB;gBAAE,SAAQ;YAE/B,OAAO,CAAC,gBAAgB,CAAC,GAAG;gBAC1B,gBAAgB;aACjB,CAAA;QACH,CAAC;QAED,OAAO,OAAO,CAAA;IAChB,CAAC;IAED,IAAI,OAAO,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,uBAAuB,KAAK,UAAU;QAC/E,OAAO,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,wBAAwB,KAAK,UAAU,EAAE,CAAC;QACjF,oEAAoE;QACpE,MAAM,OAAO,GAAG,EAAE,CAAA;QAClB,MAAM,WAAW,GAAG,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,uBAAuB,EAAE,CAAA;QAC7E,MAAM,wBAAwB,GAAG,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,wBAAwB,EAAE,CAAA;QAE3F,KAAK,MAAM,gBAAgB,IAAI,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YACxD,MAAM,gBAAgB,GAAG,yBAAyB,CAAC,wBAAwB,CAAC,gBAAgB,CAAC,CAAC,CAAA;YAE9F,IAAI,CAAC,gBAAgB;gBAAE,SAAQ;YAE/B,OAAO,CAAC,gBAAgB,CAAC,GAAG,EAAC,gBAAgB,EAAC,CAAA;QAChD,CAAC;QAED,OAAO,OAAO,CAAA;IAChB,CAAC;IAED,OAAO,EAAE,CAAA;AACX,CAAC;AAED;;;GAGG;AACH,SAAS,gBAAgB,CAAC,UAAU;IAClC,IAAI,OAAO,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,+BAA+B,KAAK,UAAU,EAAE,CAAC;QAC1F,OAAO,qCAAqC,CAAC,CAAC,EAAC,kBAAmB,CAAC,UAAU,CAAC,CAAC,+BAA+B,EAAE,CAAC,CAAC,CAAA;IACpH,CAAC;IAED,MAAM,cAAc,GAAG,OAAO,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,cAAc,KAAK,UAAU;QACzF,CAAC,CAAC,kBAAkB,CAAC,CAAC,UAAU,CAAC,CAAC,cAAc,EAAE;QAClD,CAAC,CAAC,EAAE,CAAA;IACN,MAAM,UAAU,GAAG,cAAc,CAAC,UAAU,CAAA;IAC5C,qCAAqC;IACrC,MAAM,OAAO,GAAG,EAAE,CAAA;IAElB,IAAI,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;QAC9B,KAAK,MAAM,aAAa,IAAI,UAAU,EAAE,CAAC;YACvC,IAAI,OAAO,aAAa,KAAK,QAAQ;gBAAE,SAAQ;YAE/C,OAAO,CAAC,aAAa,CAAC,GAAG,aAAa,CAAA;QACxC,CAAC;IACH,CAAC;SAAM,IAAI,aAAa,CAAC,UAAU,CAAC,EAAE,CAAC;QACrC,KAAK,MAAM,aAAa,IAAI,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACpD,OAAO,CAAC,aAAa,CAAC,GAAG,aAAa,CAAA;QACxC,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAA;AAChB,CAAC;AAED;;;;;;GAMG;AACH,SAAS,qBAAqB,CAAC,EAAC,aAAa,EAAE,UAAU,EAAE,KAAK,EAAC;IAC/D,OAAO,SAAS,CAAC;QACf,aAAa;QACb,UAAU;QACV,UAAU,CAAC,UAAU,CAAC,aAAa,CAAC;QACpC,UAAU,CAAC,UAAU,CAAC,UAAU,CAAC;KAClC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAA;AACpB,CAAC;AAED;;;GAGG;AACH,SAAS,eAAe,CAAC,GAAG;IAC1B,KAAK,MAAM,SAAS,IAAI,mBAAmB,EAAE,CAAC;QAC5C,MAAM,MAAM,GAAG,IAAI,SAAS,EAAE,CAAA;QAC9B,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC;YAAE,SAAQ;QAEnC,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAA;QAE1D,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CAAC,wBAAwB,GAAG,EAAE,CAAC,CAAA;QAChD,CAAC;QAED,OAAO;YACL,SAAS;YACT,SAAS,EAAE,+BAA+B,CAAC,CAAC,SAAS,CAAC;SACvD,CAAA;IACH,CAAC;IAED,KAAK,MAAM,SAAS,IAAI,mBAAmB,EAAE,CAAC;QAC5C,MAAM,WAAW,GAAG,kBAAkB,CAAC,SAAS,CAAC,CAAA;QAEjD,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,WAAW,CAAC;YAAE,SAAQ;QAExC,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC,CAAA;QAE/D,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CAAC,wBAAwB,GAAG,EAAE,CAAC,CAAA;QAChD,CAAC;QAED,OAAO;YACL,SAAS;YACT,SAAS,EAAE,+BAA+B,CAAC,CAAC,SAAS,CAAC;SACvD,CAAA;IACH,CAAC;IAED,OAAO,IAAI,CAAA;AACb,CAAC;AAED;;;GAGG;AACH,SAAS,kBAAkB,CAAC,KAAK;IAC/B,MAAM,QAAQ,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;IAEjC,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;AAC/F,CAAC;AAED;;;;;GAKG;AACH,SAAS,qBAAqB,CAAC,EAAC,SAAS,EAAE,KAAK,EAAC;IAC/C,IAAI,SAAS,KAAK,MAAM,EAAE,CAAC;QACzB,MAAM,YAAY,GAAG,uBAAuB,CAAC,KAAK,CAAC,CAAA;QAEnD,IAAI,YAAY,KAAK,IAAI;YAAE,OAAO,sBAAsB,CAAA;QAExD,OAAO,YAAY,CAAA;IACrB,CAAC;IAED,IAAI,SAAS,KAAK,IAAI,IAAI,SAAS,KAAK,QAAQ,EAAE,CAAC;QACjD,MAAM,eAAe,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAA;QAEpD,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC;YAAE,OAAO,sBAAsB,CAAA;QAE7D,OAAO,eAAe,CAAA;IACxB,CAAC;IAED,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI;QAAE,OAAO,sBAAsB,CAAA;IACxE,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC;QAAE,OAAO,sBAAsB,CAAA;IAEhF,OAAO,KAAK,CAAA;AACd,CAAC;AAED;;;GAGG;AACH,SAAS,uBAAuB,CAAC,KAAK;IACpC,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,KAAK;QAAE,OAAO,KAAK,CAAA;IACnD,IAAI,KAAK,KAAK,CAAC,IAAI,KAAK,KAAK,GAAG,IAAI,KAAK,KAAK,MAAM;QAAE,OAAO,IAAI,CAAA;IACjE,IAAI,KAAK,KAAK,CAAC,IAAI,KAAK,KAAK,GAAG,IAAI,KAAK,KAAK,OAAO;QAAE,OAAO,KAAK,CAAA;IACnE,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE;QAAE,OAAO,IAAI,CAAA;IAEtE,MAAM,IAAI,KAAK,CAAC,kCAAkC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAA;AACpE,CAAC;AAED;;;GAGG;AACH,SAAS,qBAAqB,CAAC,KAAK;IAClC,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACzB,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC,CAAA;IACvF,CAAC;IAED,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,OAAO,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAA;IAC1F,CAAC;IAED,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,EAAE;QAAE,OAAO,EAAE,CAAA;IAEpE,OAAO,CAAC,KAAK,CAAC,CAAA;AAChB,CAAC;AAED;;;;GAIG;AAEH;;;;;;GAMG;AACH,MAAM,UAAU,gBAAgB,CAAC,UAAU,EAAE,UAAU;IACrD,MAAM,QAAQ,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAA;IAE/G,4BAA4B;IAC5B,MAAM,KAAK,GAAG,EAAE,CAAA;IAEhB,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;QAClC,MAAM,eAAe,GAAG,KAAK,CAAC,CAAC,CAAC,CAAA;QAChC,MAAM,kBAAkB,GAAG,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,KAAK,CAAA;QAE5E,IAAI,kBAAkB,KAAK,KAAK,IAAI,kBAAkB,KAAK,MAAM,EAAE,CAAC;YAClE,MAAM,IAAI,KAAK,CAAC,mCAAmC,kBAAkB,SAAS,OAAO,EAAE,CAAC,CAAA;QAC1F,CAAC;QAED,MAAM,iBAAiB,GAAG,oBAAoB,CAAC,EAAC,UAAU,EAAE,KAAK,EAAE,eAAe,EAAC,CAAC,CAAA;QAEpF,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACvB,MAAM,IAAI,KAAK,CAAC,mCAAmC,eAAe,SAAS,UAAU,CAAC,IAAI,EAAE,CAAC,CAAA;QAC/F,CAAC;QAED,KAAK,CAAC,IAAI,CAAC,EAAC,SAAS,EAAE,iBAAiB,EAAE,SAAS,EAAE,kBAAkB,EAAC,CAAC,CAAA;IAC3E,CAAC;IAED,OAAO,KAAK,CAAA;AACd,CAAC;AAED;;;GAGG;AACH,SAAS,aAAa,CAAC,KAAK;IAC1B,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,KAAK,CAAA;IAE7E,MAAM,SAAS,GAAG,MAAM,CAAC,cAAc,CAAC,KAAK,CAAC,CAAA;IAE9C,OAAO,SAAS,KAAK,MAAM,CAAC,SAAS,IAAI,SAAS,KAAK,IAAI,CAAA;AAC7D,CAAC;AAED;;;GAGG;AACH,SAAS,SAAS,CAAC,MAAM;IACvB,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,CAAA;AACpC,CAAC","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.350",
6
+ "version": "1.0.351",
7
7
  "main": "build/index.js",
8
8
  "types": "build/index.d.ts",
9
9
  "files": [