@plumile/filter-query 0.1.42 → 0.1.43

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
@@ -52,6 +52,7 @@ The package is ESM-only; ensure your bundler supports native ESM (Vite, Next.js,
52
52
 
53
53
  - Number field: attempts `Number()`, rejects `NaN` / `Infinity`.
54
54
  - String field: raw decoded string (empty string allowed, but not produced by numeric parse).
55
+ - Boolean field: accepts `true`, `false`, `1`, `0` (case-insensitive).
55
56
  - Lists: split by comma; invalid members emit `InvalidValue` and are skipped; empty result discards whole operator.
56
57
  - Between: must have exactly 2 comma-separated values; otherwise `InvalidArity`.
57
58
  - Duplicate `between`: first valid stored, later ones produce `DuplicateBetween`.
@@ -60,7 +61,9 @@ The package is ESM-only; ensure your bundler supports native ESM (Vite, Next.js,
60
61
 
61
62
  ```ts
62
63
  import {
64
+ booleanField,
63
65
  defineSchema,
66
+ enumField,
64
67
  numberField,
65
68
  stringField,
66
69
  parse,
@@ -71,6 +74,8 @@ import {
71
74
  const schema = defineSchema({
72
75
  price: numberField(),
73
76
  title: stringField(),
77
+ active: booleanField(),
78
+ status: enumField(['OPEN', 'CLOSED']),
74
79
  });
75
80
 
76
81
  // Parse URL search (leading '?' optional)
@@ -133,6 +138,8 @@ function ProductsList() {
133
138
  const schema = defineSchema({
134
139
  price: numberField(), // full numeric operator set
135
140
  title: stringField(['contains', 'sw']), // restrict to subset
141
+ active: booleanField(['eq', 'neq']),
142
+ status: enumField(['OPEN', 'CLOSED'], ['eq', 'in']),
136
143
  });
137
144
  ```
138
145
 
@@ -142,6 +149,8 @@ Both helpers accept an optional operator list to _whitelist_ allowed operators (
142
149
 
143
150
  - `numberField()` default: `gt,gte,lt,lte,eq,neq,between,in,nin`
144
151
  - `stringField()` default: `contains,sw,ew,eq,neq,in,nin`
152
+ - `booleanField()` default: `eq,neq,in,nin`
153
+ - `enumField()` default: `eq,neq,in,nin`
145
154
 
146
155
  ## Diagnostics
147
156
 
@@ -185,6 +194,30 @@ The inferred `filters` type is roughly:
185
194
 
186
195
  Everything is optional so you can build partial filters progressively.
187
196
 
197
+ ## Custom Field Parsing
198
+
199
+ You can enforce custom parsing/serialisation rules via `customField`:
200
+
201
+ ```ts
202
+ import { customField, defineSchema } from '@plumile/filter-query';
203
+
204
+ const schema = defineSchema({
205
+ since: customField<Date>({
206
+ operators: ['eq'],
207
+ parse(raw) {
208
+ const parsed = new Date(raw);
209
+ if (Number.isNaN(parsed.getTime())) {
210
+ return undefined;
211
+ }
212
+ return parsed;
213
+ },
214
+ serialize(value) {
215
+ return value.toISOString();
216
+ },
217
+ }),
218
+ });
219
+ ```
220
+
188
221
  ## Mutation Helpers
189
222
 
190
223
  ```ts
@@ -1,9 +1,8 @@
1
1
  import type { InferFilters, Schema } from './types.js';
2
- type EqFieldKeys<S extends Schema> = {
3
- [K in keyof S]: 'eq' extends keyof NonNullable<InferFilters<S>[K]> ? K : never;
4
- }[keyof S];
5
2
  export type EqInput<S extends Schema> = Partial<{
6
- [K in EqFieldKeys<S>]: NonNullable<InferFilters<S>[K]>['eq'];
3
+ [K in keyof S as 'eq' extends keyof NonNullable<InferFilters<S>[K]> ? K : never]: NonNullable<InferFilters<S>[K]> extends {
4
+ eq?: infer V;
5
+ } ? V : never;
7
6
  }>;
8
7
  export declare function eq<S extends Schema>(input: EqInput<S>): InferFilters<S>;
9
8
  export declare const buildEqFilters: typeof eq;
@@ -11,5 +10,4 @@ export declare function setFilter<S extends Schema, F extends keyof S, O extends
11
10
  export declare function removeFilter<S extends Schema, F extends keyof S>(filters: Readonly<InferFilters<S>>, field: F, operator?: keyof NonNullable<InferFilters<S>[F]>): InferFilters<S>;
12
11
  export declare function mergeFilters<S extends Schema>(base: Readonly<InferFilters<S>>, patch: Readonly<InferFilters<S>>): InferFilters<S>;
13
12
  export declare function isEmpty<S extends Schema>(filters: Readonly<InferFilters<S>>): boolean;
14
- export {};
15
13
  //# sourceMappingURL=mutate.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"mutate.d.ts","sourceRoot":"","sources":["../../src/mutate.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AAgCvD,KAAK,WAAW,CAAC,CAAC,SAAS,MAAM,IAAI;KAClC,CAAC,IAAI,MAAM,CAAC,GAAG,IAAI,SAAS,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAC9D,CAAC,GACD,KAAK;CACV,CAAC,MAAM,CAAC,CAAC,CAAC;AAEX,MAAM,MAAM,OAAO,CAAC,CAAC,SAAS,MAAM,IAAI,OAAO,CAAC;KAC7C,CAAC,IAAI,WAAW,CAAC,CAAC,CAAC,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;CAC7D,CAAC,CAAC;AAOH,wBAAgB,EAAE,CAAC,CAAC,SAAS,MAAM,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC,GAAG,YAAY,CAAC,CAAC,CAAC,CAQvE;AAKD,eAAO,MAAM,cAAc,WAAK,CAAC;AAYjC,wBAAgB,SAAS,CACvB,CAAC,SAAS,MAAM,EAChB,CAAC,SAAS,MAAM,CAAC,EACjB,CAAC,SAAS,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAE/C,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAClC,OAAO,EAAE,CAAC,EACV,KAAK,EAAE,CAAC,EACR,QAAQ,EAAE,CAAC,EACX,KAAK,EAAE,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,SAAS,GACpD,YAAY,CAAC,CAAC,CAAC,CAwBjB;AASD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,EAAE,CAAC,SAAS,MAAM,CAAC,EAC9D,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAClC,KAAK,EAAE,CAAC,EACR,QAAQ,CAAC,EAAE,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAC/C,YAAY,CAAC,CAAC,CAAC,CAkBjB;AASD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,EAC3C,IAAI,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAC/B,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GAC/B,YAAY,CAAC,CAAC,CAAC,CAoBjB;AAOD,wBAAgB,OAAO,CAAC,CAAC,SAAS,MAAM,EACtC,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GACjC,OAAO,CAET"}
1
+ {"version":3,"file":"mutate.d.ts","sourceRoot":"","sources":["../../src/mutate.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AAgCvD,MAAM,MAAM,OAAO,CAAC,CAAC,SAAS,MAAM,IAAI,OAAO,CAAC;KAC7C,CAAC,IAAI,MAAM,CAAC,IAAI,IAAI,SAAS,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAC/D,CAAC,GACD,KAAK,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS;QAAE,EAAE,CAAC,EAAE,MAAM,CAAC,CAAA;KAAE,GAChE,CAAC,GACD,KAAK;CACV,CAAC,CAAC;AAOH,wBAAgB,EAAE,CAAC,CAAC,SAAS,MAAM,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC,GAAG,YAAY,CAAC,CAAC,CAAC,CAQvE;AAKD,eAAO,MAAM,cAAc,WAAK,CAAC;AAYjC,wBAAgB,SAAS,CACvB,CAAC,SAAS,MAAM,EAChB,CAAC,SAAS,MAAM,CAAC,EACjB,CAAC,SAAS,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAE/C,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAClC,OAAO,EAAE,CAAC,EACV,KAAK,EAAE,CAAC,EACR,QAAQ,EAAE,CAAC,EACX,KAAK,EAAE,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,SAAS,GACpD,YAAY,CAAC,CAAC,CAAC,CAwBjB;AASD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,EAAE,CAAC,SAAS,MAAM,CAAC,EAC9D,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAClC,KAAK,EAAE,CAAC,EACR,QAAQ,CAAC,EAAE,MAAM,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAC/C,YAAY,CAAC,CAAC,CAAC,CAkBjB;AASD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,EAC3C,IAAI,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,EAC/B,KAAK,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GAC/B,YAAY,CAAC,CAAC,CAAC,CAoBjB;AAOD,wBAAgB,OAAO,CAAC,CAAC,SAAS,MAAM,EACtC,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GACjC,OAAO,CAET"}
package/lib/esm/mutate.js CHANGED
@@ -95,4 +95,4 @@ export function mergeFilters(base, patch) {
95
95
  export function isEmpty(filters) {
96
96
  return Object.keys(filters).length === 0;
97
97
  }
98
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mutate.js","sourceRoot":"","sources":["../../src/mutate.ts"],"names":[],"mappings":"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","sourcesContent":["import type { InferFilters, Schema } from './types.js';\n\n/**\n * Checks shallow equality between two plain objects (own enumerable string keys only).\n */\nfunction shallowEqual(a: unknown, b: unknown): boolean {\n  if (a === b) {\n    return true;\n  }\n  if (\n    typeof a !== 'object' ||\n    a === null ||\n    typeof b !== 'object' ||\n    b === null\n  ) {\n    return false;\n  }\n  const aObj = a as Record<string, unknown>;\n  const bObj = b as Record<string, unknown>;\n  const aKeys = Object.keys(aObj);\n  const bKeys = Object.keys(bObj);\n  if (aKeys.length !== bKeys.length) {\n    return false;\n  }\n  for (const k of aKeys) {\n    if (aObj[k] !== bObj[k]) {\n      return false;\n    }\n  }\n  return true;\n}\n\ntype EqFieldKeys<S extends Schema> = {\n  [K in keyof S]: 'eq' extends keyof NonNullable<InferFilters<S>[K]>\n    ? K\n    : never;\n}[keyof S];\n\nexport type EqInput<S extends Schema> = Partial<{\n  [K in EqFieldKeys<S>]: NonNullable<InferFilters<S>[K]>['eq'];\n}>;\n\n/**\n * Builds a filters object using implicit equality for each provided field.\n *\n * @param input - Record of field values to encode as `{ eq: value }`.\n */\nexport function eq<S extends Schema>(input: EqInput<S>): InferFilters<S> {\n  const filters: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(input)) {\n    if (value !== undefined) {\n      filters[key] = { eq: value };\n    }\n  }\n  return filters as InferFilters<S>;\n}\n\n/**\n * Alias for {@link eq} for callers who prefer a more descriptive name.\n */\nexport const buildEqFilters = eq;\n\n/**\n * Sets or unsets a specific operator value on a field while preserving reference\n * equality when nothing changes.\n *\n * @param filters - Current filters object that will be treated as immutable input.\n * @param _schema - Schema reference (reserved for future validation hooks).\n * @param field - Field key to update.\n * @param operator - Operator name to set or remove.\n * @param value - New operator value; pass `undefined` to remove it.\n */\nexport function setFilter<\n  S extends Schema,\n  F extends keyof S,\n  O extends keyof NonNullable<InferFilters<S>[F]>,\n>(\n  filters: Readonly<InferFilters<S>>,\n  _schema: S, // reserved for future validation hooks\n  field: F,\n  operator: O,\n  value: NonNullable<InferFilters<S>[F]>[O] | undefined,\n): InferFilters<S> {\n  const currentField = filters[field];\n  if (value === undefined) {\n    if (currentField === undefined) {\n      return filters as InferFilters<S>;\n    }\n    const { [operator]: _removed, ...rest } = currentField as Record<\n      string,\n      unknown\n    >;\n    if (Object.keys(rest).length === 0) {\n      const { [field]: _f, ...clone } = filters as Record<string, unknown>;\n      return clone as InferFilters<S>;\n    }\n    return { ...filters, [field]: rest } as InferFilters<S>;\n  }\n  const newField = {\n    ...(currentField as Record<string, unknown> | undefined),\n    [operator]: value,\n  };\n  if (currentField !== undefined && shallowEqual(currentField, newField)) {\n    return filters as InferFilters<S>;\n  }\n  return { ...filters, [field]: newField } as InferFilters<S>;\n}\n\n/**\n * Removes either an entire field or a specific operator from the filters object.\n *\n * @param filters - Current filters object (treated immutably).\n * @param field - Field name to alter.\n * @param operator - Optional operator to remove; omit to drop the whole field.\n */\nexport function removeFilter<S extends Schema, F extends keyof S>(\n  filters: Readonly<InferFilters<S>>,\n  field: F,\n  operator?: keyof NonNullable<InferFilters<S>[F]>,\n): InferFilters<S> {\n  const currentField = filters[field];\n  if (currentField === undefined) {\n    return filters as InferFilters<S>;\n  }\n  if (operator === undefined) {\n    const { [field]: _removed, ...clone } = filters as Record<string, unknown>;\n    return clone as InferFilters<S>;\n  }\n  const { [operator]: _opRemoved, ...rest } = currentField as Record<\n    string,\n    unknown\n  >;\n  if (Object.keys(rest).length === 0) {\n    const { [field]: _f, ...clone } = filters as Record<string, unknown>;\n    return clone as InferFilters<S>;\n  }\n  return { ...filters, [field]: rest } as InferFilters<S>;\n}\n\n/**\n * Merges a patch of filters into an existing filters object while preserving reference\n * identity for unchanged fields.\n *\n * @param base - Original filters.\n * @param patch - Filters to overlay.\n */\nexport function mergeFilters<S extends Schema>(\n  base: Readonly<InferFilters<S>>,\n  patch: Readonly<InferFilters<S>>,\n): InferFilters<S> {\n  let changed = false;\n  const result: Partial<InferFilters<S>> = { ...base };\n  for (const key of Object.keys(patch) as (keyof S)[]) {\n    const existing = base[key];\n    const incoming = patch[key];\n    if (incoming === undefined) {\n      // skip undefined incoming\n    } else if (existing === undefined) {\n      result[key] = incoming as InferFilters<S>[typeof key];\n      changed = true;\n    } else if (!shallowEqual(existing, incoming)) {\n      result[key] = incoming as InferFilters<S>[typeof key];\n      changed = true;\n    }\n  }\n  if (changed) {\n    return result as InferFilters<S>;\n  }\n  return base as InferFilters<S>;\n}\n\n/**\n * Determines if the filters object is empty (no fields present).\n *\n * @param filters - Filters to inspect.\n */\nexport function isEmpty<S extends Schema>(\n  filters: Readonly<InferFilters<S>>,\n): boolean {\n  return Object.keys(filters).length === 0;\n}\n"]}
98
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mutate.js","sourceRoot":"","sources":["../../src/mutate.ts"],"names":[],"mappings":"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","sourcesContent":["import type { InferFilters, Schema } from './types.js';\n\n/**\n * Checks shallow equality between two plain objects (own enumerable string keys only).\n */\nfunction shallowEqual(a: unknown, b: unknown): boolean {\n  if (a === b) {\n    return true;\n  }\n  if (\n    typeof a !== 'object' ||\n    a === null ||\n    typeof b !== 'object' ||\n    b === null\n  ) {\n    return false;\n  }\n  const aObj = a as Record<string, unknown>;\n  const bObj = b as Record<string, unknown>;\n  const aKeys = Object.keys(aObj);\n  const bKeys = Object.keys(bObj);\n  if (aKeys.length !== bKeys.length) {\n    return false;\n  }\n  for (const k of aKeys) {\n    if (aObj[k] !== bObj[k]) {\n      return false;\n    }\n  }\n  return true;\n}\n\nexport type EqInput<S extends Schema> = Partial<{\n  [K in keyof S as 'eq' extends keyof NonNullable<InferFilters<S>[K]>\n    ? K\n    : never]: NonNullable<InferFilters<S>[K]> extends { eq?: infer V }\n    ? V\n    : never;\n}>;\n\n/**\n * Builds a filters object using implicit equality for each provided field.\n *\n * @param input - Record of field values to encode as `{ eq: value }`.\n */\nexport function eq<S extends Schema>(input: EqInput<S>): InferFilters<S> {\n  const filters: Record<string, unknown> = {};\n  for (const [key, value] of Object.entries(input)) {\n    if (value !== undefined) {\n      filters[key] = { eq: value };\n    }\n  }\n  return filters as InferFilters<S>;\n}\n\n/**\n * Alias for {@link eq} for callers who prefer a more descriptive name.\n */\nexport const buildEqFilters = eq;\n\n/**\n * Sets or unsets a specific operator value on a field while preserving reference\n * equality when nothing changes.\n *\n * @param filters - Current filters object that will be treated as immutable input.\n * @param _schema - Schema reference (reserved for future validation hooks).\n * @param field - Field key to update.\n * @param operator - Operator name to set or remove.\n * @param value - New operator value; pass `undefined` to remove it.\n */\nexport function setFilter<\n  S extends Schema,\n  F extends keyof S,\n  O extends keyof NonNullable<InferFilters<S>[F]>,\n>(\n  filters: Readonly<InferFilters<S>>,\n  _schema: S, // reserved for future validation hooks\n  field: F,\n  operator: O,\n  value: NonNullable<InferFilters<S>[F]>[O] | undefined,\n): InferFilters<S> {\n  const currentField = filters[field];\n  if (value === undefined) {\n    if (currentField === undefined) {\n      return filters as InferFilters<S>;\n    }\n    const { [operator]: _removed, ...rest } = currentField as Record<\n      string,\n      unknown\n    >;\n    if (Object.keys(rest).length === 0) {\n      const { [field]: _f, ...clone } = filters as Record<string, unknown>;\n      return clone as InferFilters<S>;\n    }\n    return { ...filters, [field]: rest } as InferFilters<S>;\n  }\n  const newField = {\n    ...(currentField as Record<string, unknown> | undefined),\n    [operator]: value,\n  };\n  if (currentField !== undefined && shallowEqual(currentField, newField)) {\n    return filters as InferFilters<S>;\n  }\n  return { ...filters, [field]: newField } as InferFilters<S>;\n}\n\n/**\n * Removes either an entire field or a specific operator from the filters object.\n *\n * @param filters - Current filters object (treated immutably).\n * @param field - Field name to alter.\n * @param operator - Optional operator to remove; omit to drop the whole field.\n */\nexport function removeFilter<S extends Schema, F extends keyof S>(\n  filters: Readonly<InferFilters<S>>,\n  field: F,\n  operator?: keyof NonNullable<InferFilters<S>[F]>,\n): InferFilters<S> {\n  const currentField = filters[field];\n  if (currentField === undefined) {\n    return filters as InferFilters<S>;\n  }\n  if (operator === undefined) {\n    const { [field]: _removed, ...clone } = filters as Record<string, unknown>;\n    return clone as InferFilters<S>;\n  }\n  const { [operator]: _opRemoved, ...rest } = currentField as Record<\n    string,\n    unknown\n  >;\n  if (Object.keys(rest).length === 0) {\n    const { [field]: _f, ...clone } = filters as Record<string, unknown>;\n    return clone as InferFilters<S>;\n  }\n  return { ...filters, [field]: rest } as InferFilters<S>;\n}\n\n/**\n * Merges a patch of filters into an existing filters object while preserving reference\n * identity for unchanged fields.\n *\n * @param base - Original filters.\n * @param patch - Filters to overlay.\n */\nexport function mergeFilters<S extends Schema>(\n  base: Readonly<InferFilters<S>>,\n  patch: Readonly<InferFilters<S>>,\n): InferFilters<S> {\n  let changed = false;\n  const result: Partial<InferFilters<S>> = { ...base };\n  for (const key of Object.keys(patch) as (keyof S)[]) {\n    const existing = base[key];\n    const incoming = patch[key];\n    if (incoming === undefined) {\n      // skip undefined incoming\n    } else if (existing === undefined) {\n      result[key] = incoming as InferFilters<S>[typeof key];\n      changed = true;\n    } else if (!shallowEqual(existing, incoming)) {\n      result[key] = incoming as InferFilters<S>[typeof key];\n      changed = true;\n    }\n  }\n  if (changed) {\n    return result as InferFilters<S>;\n  }\n  return base as InferFilters<S>;\n}\n\n/**\n * Determines if the filters object is empty (no fields present).\n *\n * @param filters - Filters to inspect.\n */\nexport function isEmpty<S extends Schema>(\n  filters: Readonly<InferFilters<S>>,\n): boolean {\n  return Object.keys(filters).length === 0;\n}\n"]}
package/lib/esm/parse.js CHANGED
@@ -25,6 +25,9 @@ function decodeValue(value, field, operator, diagnostics) {
25
25
  }
26
26
  }
27
27
  function parseSingleValue(descriptor, raw) {
28
+ if (descriptor.parse != null) {
29
+ return descriptor.parse(raw);
30
+ }
28
31
  if (descriptor.kind === 'number') {
29
32
  if (raw === '') {
30
33
  return undefined;
@@ -178,4 +181,4 @@ export function parse(rawSearch, schema) {
178
181
  }
179
182
  return acc;
180
183
  }
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parse.js","sourceRoot":"","sources":["../../src/parse.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Schema,\n  Operator,\n  ListOperator,\n  BetweenOperator,\n  InferFilters,\n  FieldDescriptor,\n  ParseResult,\n} from './types.js';\nimport type { Diagnostic } from './errors.js';\n\nconst BETWEEN: BetweenOperator = 'between';\nconst LIST_OPS: readonly ListOperator[] = ['in', 'nin'];\n\ninterface InternalAccumulator<S extends Schema> {\n  filters: InferFilters<S>;\n  diagnostics: Diagnostic[];\n}\n\n/** Shallow clone plus ensure field object helper (immutable update). */\nfunction ensureFieldObject<S extends Schema>(\n  filters: InferFilters<S>,\n  field: string,\n): { container: InferFilters<S>; target: Record<string, unknown> } {\n  const existing = filters[field as keyof S] as\n    | Record<string, unknown>\n    | undefined;\n  if (existing !== undefined) {\n    return { container: filters, target: existing };\n  }\n  const clone: InferFilters<S> = { ...filters };\n  const created: Record<string, unknown> = {};\n  (clone as Record<string, unknown>)[field] = created;\n  return { container: clone, target: created };\n}\n\n/** Decode component, recording a diagnostic on failure. */\nfunction decodeValue(\n  value: string,\n  field: string,\n  operator: string,\n  diagnostics: Diagnostic[],\n): string | undefined {\n  try {\n    return decodeURIComponent(value);\n  } catch {\n    diagnostics.push({\n      kind: 'DecodeError',\n      field,\n      operator,\n      detail: 'Failed to decode',\n    });\n    return undefined;\n  }\n}\n\n/** Parse a single raw value according to field kind. */\nfunction parseSingleValue(\n  descriptor: FieldDescriptor,\n  raw: string,\n): string | number | undefined {\n  if (descriptor.kind === 'number') {\n    if (raw === '') {\n      // Treat empty string as invalid for numeric fields (reject instead of coercing to 0)\n      return undefined;\n    }\n    const n = Number(raw);\n    if (!Number.isFinite(n)) {\n      return undefined;\n    }\n    return n;\n  }\n  return raw;\n}\n\n/** Handle a between operator value. */\nfunction handleBetween<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const parts: string[] = decoded.split(',');\n  if (parts.length !== 2) {\n    acc.diagnostics.push({\n      kind: 'InvalidArity',\n      field,\n      operator,\n      detail: 'Expected exactly 2 values',\n    });\n    return;\n  }\n  const [aRaw, bRaw] = parts as [string, string];\n  const a = parseSingleValue(descriptor, aRaw);\n  const b = parseSingleValue(descriptor, bRaw);\n  if (a === undefined || b === undefined) {\n    acc.diagnostics.push({\n      kind: 'InvalidValue',\n      field,\n      operator,\n      detail: 'One of the between values invalid',\n    });\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  if (Object.prototype.hasOwnProperty.call(ensured.target, 'between')) {\n    acc.diagnostics.push({ kind: 'DuplicateBetween', field });\n    return;\n  }\n  ensured.target.between = [a, b] as unknown;\n}\n\n/** Handle list operators (in, nin). */\nfunction handleList<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const listRaw = decoded.split(',');\n  const parsedList: (string | number)[] = [];\n  for (const item of listRaw) {\n    const parsed = parseSingleValue(descriptor, item);\n    if (parsed === undefined) {\n      acc.diagnostics.push({\n        kind: 'InvalidValue',\n        field,\n        operator,\n        detail: 'List item invalid',\n      });\n    } else {\n      parsedList.push(parsed);\n    }\n  }\n  if (parsedList.length === 0) {\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  const targetObj: Record<string, unknown> = ensured.target;\n  const existing = targetObj[operator];\n  if (Array.isArray(existing)) {\n    targetObj[operator] = [...existing, ...parsedList];\n  } else {\n    targetObj[operator] = parsedList;\n  }\n}\n\n/** Handle scalar operator values. */\nfunction handleScalar<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const scalar = parseSingleValue(descriptor, decoded);\n  if (scalar === undefined) {\n    acc.diagnostics.push({ kind: 'InvalidValue', field, operator });\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  const scalarTarget: Record<string, unknown> = ensured.target;\n  scalarTarget[operator] = scalar;\n}\n\n/**\n * Parses a raw search string (leading `?` optional) according to the provided schema\n * and returns the filters plus diagnostics.\n *\n * @param rawSearch - Query string to parse (with or without the leading question mark).\n * @param schema - Schema describing allowed fields and operators.\n * @returns Parsed filters alongside non-blocking diagnostics.\n */\nexport function parse<S extends Schema>(\n  rawSearch: string,\n  schema: S,\n): ParseResult<S> {\n  const acc: InternalAccumulator<S> = { filters: {}, diagnostics: [] };\n  let search = rawSearch;\n  if (search.startsWith('?')) {\n    search = search.slice(1);\n  }\n  if (search.trim() === '') {\n    return acc;\n  }\n  const segments = search.split('&');\n  for (const seg of segments) {\n    const eqIdx = seg.indexOf('=');\n    const validBasic = seg !== '' && eqIdx !== -1;\n    if (validBasic) {\n      const rawKey = seg.slice(0, eqIdx);\n      const rawVal = seg.slice(eqIdx + 1);\n      const lastDot = rawKey.lastIndexOf('.');\n      // Case 1: key with explicit operator (field.op)\n      const explicitShape = !(lastDot <= 0 || lastDot === rawKey.length - 1);\n      if (explicitShape) {\n        const field = rawKey.slice(0, lastDot);\n        const operator = rawKey.slice(lastDot + 1) as Operator;\n        const descriptor = schema[field];\n        const descOk = descriptor?.operators.includes(operator) === true;\n        if (!descOk) {\n          if (descriptor == null) {\n            acc.diagnostics.push({ kind: 'UnknownField', field });\n          } else {\n            acc.diagnostics.push({ kind: 'UnknownOperator', field, operator });\n          }\n        } else {\n          const decoded = decodeValue(rawVal, field, operator, acc.diagnostics);\n          if (decoded !== undefined) {\n            const isList = (LIST_OPS as readonly string[]).includes(operator);\n            if (operator === BETWEEN) {\n              handleBetween(descriptor, decoded, field, operator, acc);\n            } else if (isList) {\n              handleList(descriptor, decoded, field, operator, acc);\n            } else {\n              handleScalar(descriptor, decoded, field, operator, acc);\n            }\n          }\n        }\n      } else if (lastDot === -1) {\n        // implicit equality (no dot) => operator 'eq'\n        const field = rawKey;\n        const operator: Operator = 'eq';\n        const descriptor = schema[field];\n        if (descriptor == null) {\n          acc.diagnostics.push({ kind: 'UnknownField', field });\n        } else if (!descriptor.operators.includes(operator)) {\n          acc.diagnostics.push({ kind: 'UnknownOperator', field, operator });\n        } else {\n          const decoded = decodeValue(rawVal, field, operator, acc.diagnostics);\n          if (decoded !== undefined) {\n            handleScalar(descriptor, decoded, field, operator, acc);\n          }\n        }\n      }\n    }\n  }\n  return acc;\n}\n"]}
184
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parse.js","sourceRoot":"","sources":["../../src/parse.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Schema,\n  Operator,\n  ListOperator,\n  BetweenOperator,\n  InferFilters,\n  FieldDescriptor,\n  ParseResult,\n} from './types.js';\nimport type { Diagnostic } from './errors.js';\n\nconst BETWEEN: BetweenOperator = 'between';\nconst LIST_OPS: readonly ListOperator[] = ['in', 'nin'];\n\ninterface InternalAccumulator<S extends Schema> {\n  filters: InferFilters<S>;\n  diagnostics: Diagnostic[];\n}\n\n/** Shallow clone plus ensure field object helper (immutable update). */\nfunction ensureFieldObject<S extends Schema>(\n  filters: InferFilters<S>,\n  field: string,\n): { container: InferFilters<S>; target: Record<string, unknown> } {\n  const existing = filters[field as keyof S] as\n    | Record<string, unknown>\n    | undefined;\n  if (existing !== undefined) {\n    return { container: filters, target: existing };\n  }\n  const clone: InferFilters<S> = { ...filters };\n  const created: Record<string, unknown> = {};\n  (clone as Record<string, unknown>)[field] = created;\n  return { container: clone, target: created };\n}\n\n/** Decode component, recording a diagnostic on failure. */\nfunction decodeValue(\n  value: string,\n  field: string,\n  operator: string,\n  diagnostics: Diagnostic[],\n): string | undefined {\n  try {\n    return decodeURIComponent(value);\n  } catch {\n    diagnostics.push({\n      kind: 'DecodeError',\n      field,\n      operator,\n      detail: 'Failed to decode',\n    });\n    return undefined;\n  }\n}\n\n/** Parse a single raw value according to field kind. */\nfunction parseSingleValue(descriptor: FieldDescriptor, raw: string): unknown {\n  if (descriptor.parse != null) {\n    return descriptor.parse(raw);\n  }\n  if (descriptor.kind === 'number') {\n    if (raw === '') {\n      // Treat empty string as invalid for numeric fields (reject instead of coercing to 0)\n      return undefined;\n    }\n    const n = Number(raw);\n    if (!Number.isFinite(n)) {\n      return undefined;\n    }\n    return n;\n  }\n  return raw;\n}\n\n/** Handle a between operator value. */\nfunction handleBetween<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const parts: string[] = decoded.split(',');\n  if (parts.length !== 2) {\n    acc.diagnostics.push({\n      kind: 'InvalidArity',\n      field,\n      operator,\n      detail: 'Expected exactly 2 values',\n    });\n    return;\n  }\n  const [aRaw, bRaw] = parts as [string, string];\n  const a = parseSingleValue(descriptor, aRaw);\n  const b = parseSingleValue(descriptor, bRaw);\n  if (a === undefined || b === undefined) {\n    acc.diagnostics.push({\n      kind: 'InvalidValue',\n      field,\n      operator,\n      detail: 'One of the between values invalid',\n    });\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  if (Object.prototype.hasOwnProperty.call(ensured.target, 'between')) {\n    acc.diagnostics.push({ kind: 'DuplicateBetween', field });\n    return;\n  }\n  ensured.target.between = [a, b] as unknown;\n}\n\n/** Handle list operators (in, nin). */\nfunction handleList<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const listRaw = decoded.split(',');\n  const parsedList: unknown[] = [];\n  for (const item of listRaw) {\n    const parsed = parseSingleValue(descriptor, item);\n    if (parsed === undefined) {\n      acc.diagnostics.push({\n        kind: 'InvalidValue',\n        field,\n        operator,\n        detail: 'List item invalid',\n      });\n    } else {\n      parsedList.push(parsed);\n    }\n  }\n  if (parsedList.length === 0) {\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  const targetObj: Record<string, unknown> = ensured.target;\n  const existing = targetObj[operator];\n  if (Array.isArray(existing)) {\n    targetObj[operator] = [...existing, ...parsedList];\n  } else {\n    targetObj[operator] = parsedList;\n  }\n}\n\n/** Handle scalar operator values. */\nfunction handleScalar<S extends Schema>(\n  descriptor: FieldDescriptor,\n  decoded: string,\n  field: string,\n  operator: string,\n  acc: InternalAccumulator<S>,\n): void {\n  const scalar = parseSingleValue(descriptor, decoded);\n  if (scalar === undefined) {\n    acc.diagnostics.push({ kind: 'InvalidValue', field, operator });\n    return;\n  }\n  const ensured = ensureFieldObject(acc.filters, field);\n  acc.filters = ensured.container;\n  const scalarTarget: Record<string, unknown> = ensured.target;\n  scalarTarget[operator] = scalar;\n}\n\n/**\n * Parses a raw search string (leading `?` optional) according to the provided schema\n * and returns the filters plus diagnostics.\n *\n * @param rawSearch - Query string to parse (with or without the leading question mark).\n * @param schema - Schema describing allowed fields and operators.\n * @returns Parsed filters alongside non-blocking diagnostics.\n */\nexport function parse<S extends Schema>(\n  rawSearch: string,\n  schema: S,\n): ParseResult<S> {\n  const acc: InternalAccumulator<S> = { filters: {}, diagnostics: [] };\n  let search = rawSearch;\n  if (search.startsWith('?')) {\n    search = search.slice(1);\n  }\n  if (search.trim() === '') {\n    return acc;\n  }\n  const segments = search.split('&');\n  for (const seg of segments) {\n    const eqIdx = seg.indexOf('=');\n    const validBasic = seg !== '' && eqIdx !== -1;\n    if (validBasic) {\n      const rawKey = seg.slice(0, eqIdx);\n      const rawVal = seg.slice(eqIdx + 1);\n      const lastDot = rawKey.lastIndexOf('.');\n      // Case 1: key with explicit operator (field.op)\n      const explicitShape = !(lastDot <= 0 || lastDot === rawKey.length - 1);\n      if (explicitShape) {\n        const field = rawKey.slice(0, lastDot);\n        const operator = rawKey.slice(lastDot + 1) as Operator;\n        const descriptor = schema[field];\n        const descOk = descriptor?.operators.includes(operator) === true;\n        if (!descOk) {\n          if (descriptor == null) {\n            acc.diagnostics.push({ kind: 'UnknownField', field });\n          } else {\n            acc.diagnostics.push({ kind: 'UnknownOperator', field, operator });\n          }\n        } else {\n          const decoded = decodeValue(rawVal, field, operator, acc.diagnostics);\n          if (decoded !== undefined) {\n            const isList = (LIST_OPS as readonly string[]).includes(operator);\n            if (operator === BETWEEN) {\n              handleBetween(descriptor, decoded, field, operator, acc);\n            } else if (isList) {\n              handleList(descriptor, decoded, field, operator, acc);\n            } else {\n              handleScalar(descriptor, decoded, field, operator, acc);\n            }\n          }\n        }\n      } else if (lastDot === -1) {\n        // implicit equality (no dot) => operator 'eq'\n        const field = rawKey;\n        const operator: Operator = 'eq';\n        const descriptor = schema[field];\n        if (descriptor == null) {\n          acc.diagnostics.push({ kind: 'UnknownField', field });\n        } else if (!descriptor.operators.includes(operator)) {\n          acc.diagnostics.push({ kind: 'UnknownOperator', field, operator });\n        } else {\n          const decoded = decodeValue(rawVal, field, operator, acc.diagnostics);\n          if (decoded !== undefined) {\n            handleScalar(descriptor, decoded, field, operator, acc);\n          }\n        }\n      }\n    }\n  }\n  return acc;\n}\n"]}
@@ -1,5 +1,14 @@
1
- import type { FieldDescriptor, NumberFieldDescriptor, StringFieldDescriptor, Operator } from './types.js';
1
+ import type { FieldDescriptor, NumberFieldDescriptor, StringFieldDescriptor, BooleanFieldDescriptor, EnumFieldDescriptor, CustomFieldDescriptor, Operator } from './types.js';
2
2
  export declare function numberField(ops?: readonly Operator[]): NumberFieldDescriptor;
3
3
  export declare function stringField(ops?: readonly Operator[]): StringFieldDescriptor;
4
+ export declare function booleanField(ops?: readonly Operator[]): BooleanFieldDescriptor;
5
+ export declare function enumField<const T extends string>(values: readonly T[], ops?: readonly Operator[]): EnumFieldDescriptor<T>;
6
+ type CustomFieldOptions<T> = {
7
+ operators: readonly Operator[];
8
+ parse: (raw: string) => T | undefined;
9
+ serialize?: (value: T) => string;
10
+ };
11
+ export declare function customField<T>(options: CustomFieldOptions<T>): CustomFieldDescriptor<T>;
4
12
  export declare function defineSchema<S extends Record<string, FieldDescriptor>>(schema: S): S;
13
+ export {};
5
14
  //# sourceMappingURL=schema.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/schema.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,eAAe,EACf,qBAAqB,EACrB,qBAAqB,EACrB,QAAQ,EACT,MAAM,YAAY,CAAC;AA0BpB,wBAAgB,WAAW,CACzB,GAAG,GAAE,SAAS,QAAQ,EAA6B,GAClD,qBAAqB,CAmBvB;AAKD,wBAAgB,WAAW,CACzB,GAAG,GAAE,SAAS,QAAQ,EAA6B,GAClD,qBAAqB,CAWvB;AAKD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,EACpE,MAAM,EAAE,CAAC,GACR,CAAC,CAEH"}
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/schema.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,eAAe,EACf,qBAAqB,EACrB,qBAAqB,EACrB,sBAAsB,EACtB,mBAAmB,EACnB,qBAAqB,EACrB,QAAQ,EACT,MAAM,YAAY,CAAC;AAiCpB,wBAAgB,WAAW,CACzB,GAAG,GAAE,SAAS,QAAQ,EAA6B,GAClD,qBAAqB,CAmBvB;AAKD,wBAAgB,WAAW,CACzB,GAAG,GAAE,SAAS,QAAQ,EAA6B,GAClD,qBAAqB,CAWvB;AAKD,wBAAgB,YAAY,CAC1B,GAAG,GAAE,SAAS,QAAQ,EAA8B,GACnD,sBAAsB,CAqBxB;AAKD,wBAAgB,SAAS,CAAC,KAAK,CAAC,CAAC,SAAS,MAAM,EAC9C,MAAM,EAAE,SAAS,CAAC,EAAE,EACpB,GAAG,GAAE,SAAS,QAAQ,EAA2B,GAChD,mBAAmB,CAAC,CAAC,CAAC,CAexB;AAED,KAAK,kBAAkB,CAAC,CAAC,IAAI;IAC3B,SAAS,EAAE,SAAS,QAAQ,EAAE,CAAC;IAC/B,KAAK,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,CAAC,GAAG,SAAS,CAAC;IACtC,SAAS,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK,MAAM,CAAC;CAClC,CAAC;AAKF,wBAAgB,WAAW,CAAC,CAAC,EAC3B,OAAO,EAAE,kBAAkB,CAAC,CAAC,CAAC,GAC7B,qBAAqB,CAAC,CAAC,CAAC,CAO1B;AAKD,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,EACpE,MAAM,EAAE,CAAC,GACR,CAAC,CAEH"}
package/lib/esm/schema.js CHANGED
@@ -18,6 +18,13 @@ const DEFAULT_STRING_OPERATORS = [
18
18
  'in',
19
19
  'nin',
20
20
  ];
21
+ const DEFAULT_BOOLEAN_OPERATORS = [
22
+ 'eq',
23
+ 'neq',
24
+ 'in',
25
+ 'nin',
26
+ ];
27
+ const DEFAULT_ENUM_OPERATORS = ['eq', 'neq', 'in', 'nin'];
21
28
  export function numberField(ops = DEFAULT_NUMBER_OPERATORS) {
22
29
  return {
23
30
  kind: 'number',
@@ -49,7 +56,53 @@ export function stringField(ops = DEFAULT_STRING_OPERATORS) {
49
56
  },
50
57
  };
51
58
  }
59
+ export function booleanField(ops = DEFAULT_BOOLEAN_OPERATORS) {
60
+ return {
61
+ kind: 'boolean',
62
+ operators: ops,
63
+ parse(raw) {
64
+ const normalized = raw.trim().toLowerCase();
65
+ if (normalized === 'true' || normalized === '1') {
66
+ return true;
67
+ }
68
+ if (normalized === 'false' || normalized === '0') {
69
+ return false;
70
+ }
71
+ return undefined;
72
+ },
73
+ serialize(value) {
74
+ if (value === true) {
75
+ return 'true';
76
+ }
77
+ return 'false';
78
+ },
79
+ };
80
+ }
81
+ export function enumField(values, ops = DEFAULT_ENUM_OPERATORS) {
82
+ const valueSet = new Set(values);
83
+ return {
84
+ kind: 'enum',
85
+ operators: ops,
86
+ parse(raw) {
87
+ if (valueSet.has(raw)) {
88
+ return raw;
89
+ }
90
+ return undefined;
91
+ },
92
+ serialize(value) {
93
+ return String(value);
94
+ },
95
+ };
96
+ }
97
+ export function customField(options) {
98
+ return {
99
+ kind: 'custom',
100
+ operators: options.operators,
101
+ parse: options.parse,
102
+ serialize: options.serialize,
103
+ };
104
+ }
52
105
  export function defineSchema(schema) {
53
106
  return Object.freeze({ ...schema });
54
107
  }
55
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NoZW1hLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjaGVtYS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFPQSxNQUFNLHdCQUF3QixHQUF3QjtJQUNwRCxJQUFJO0lBQ0osS0FBSztJQUNMLElBQUk7SUFDSixLQUFLO0lBQ0wsSUFBSTtJQUNKLEtBQUs7SUFDTCxTQUFTO0lBQ1QsSUFBSTtJQUNKLEtBQUs7Q0FDTixDQUFDO0FBQ0YsTUFBTSx3QkFBd0IsR0FBd0I7SUFDcEQsVUFBVTtJQUNWLElBQUk7SUFDSixJQUFJO0lBQ0osSUFBSTtJQUNKLEtBQUs7SUFDTCxJQUFJO0lBQ0osS0FBSztDQUNOLENBQUM7QUFLRixNQUFNLFVBQVUsV0FBVyxDQUN6QixNQUEyQix3QkFBd0I7SUFFbkQsT0FBTztRQUNMLElBQUksRUFBRSxRQUFRO1FBQ2QsU0FBUyxFQUFFLEdBQUc7UUFDZCxLQUFLLENBQUMsR0FBVztZQUNmLElBQUksR0FBRyxLQUFLLEVBQUUsRUFBRSxDQUFDO2dCQUNmLE9BQU8sU0FBUyxDQUFDO1lBQ25CLENBQUM7WUFDRCxNQUFNLENBQUMsR0FBRyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDdEIsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7Z0JBQ3ZCLE9BQU8sQ0FBQyxDQUFDO1lBQ1gsQ0FBQztZQUNELE9BQU8sU0FBUyxDQUFDO1FBQ25CLENBQUM7UUFFRCxTQUFTLENBQUMsS0FBYztZQUN0QixPQUFPLE1BQU0sQ0FBQyxLQUFlLENBQUMsQ0FBQztRQUNqQyxDQUFDO0tBQ3VCLENBQUM7QUFDN0IsQ0FBQztBQUtELE1BQU0sVUFBVSxXQUFXLENBQ3pCLE1BQTJCLHdCQUF3QjtJQUVuRCxPQUFPO1FBQ0wsSUFBSSxFQUFFLFFBQVE7UUFDZCxTQUFTLEVBQUUsR0FBRztRQUNkLEtBQUssQ0FBQyxHQUFXO1lBQ2YsT0FBTyxHQUFHLENBQUM7UUFDYixDQUFDO1FBQ0QsU0FBUyxDQUFDLEtBQWM7WUFDdEIsT0FBTyxNQUFNLENBQUMsS0FBZSxDQUFDLENBQUM7UUFDakMsQ0FBQztLQUN1QixDQUFDO0FBQzdCLENBQUM7QUFLRCxNQUFNLFVBQVUsWUFBWSxDQUMxQixNQUFTO0lBRVQsT0FBTyxNQUFNLENBQUMsTUFBTSxDQUFDLEVBQUUsR0FBRyxNQUFNLEVBQUUsQ0FBQyxDQUFDO0FBQ3RDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7XG4gIEZpZWxkRGVzY3JpcHRvcixcbiAgTnVtYmVyRmllbGREZXNjcmlwdG9yLFxuICBTdHJpbmdGaWVsZERlc2NyaXB0b3IsXG4gIE9wZXJhdG9yLFxufSBmcm9tICcuL3R5cGVzLmpzJztcblxuY29uc3QgREVGQVVMVF9OVU1CRVJfT1BFUkFUT1JTOiByZWFkb25seSBPcGVyYXRvcltdID0gW1xuICAnZ3QnLFxuICAnZ3RlJyxcbiAgJ2x0JyxcbiAgJ2x0ZScsXG4gICdlcScsXG4gICduZXEnLFxuICAnYmV0d2VlbicsXG4gICdpbicsXG4gICduaW4nLFxuXTtcbmNvbnN0IERFRkFVTFRfU1RSSU5HX09QRVJBVE9SUzogcmVhZG9ubHkgT3BlcmF0b3JbXSA9IFtcbiAgJ2NvbnRhaW5zJyxcbiAgJ3N3JyxcbiAgJ2V3JyxcbiAgJ2VxJyxcbiAgJ25lcScsXG4gICdpbicsXG4gICduaW4nLFxuXTtcblxuLyoqXG4gKiBDcmVhdGUgYSBudW1lcmljIGZpZWxkIGRlc2NyaXB0b3Igd2l0aCBhIGdpdmVuIG9wZXJhdG9yIHdoaXRlbGlzdC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIG51bWJlckZpZWxkKFxuICBvcHM6IHJlYWRvbmx5IE9wZXJhdG9yW10gPSBERUZBVUxUX05VTUJFUl9PUEVSQVRPUlMsXG4pOiBOdW1iZXJGaWVsZERlc2NyaXB0b3Ige1xuICByZXR1cm4ge1xuICAgIGtpbmQ6ICdudW1iZXInLFxuICAgIG9wZXJhdG9yczogb3BzLFxuICAgIHBhcnNlKHJhdzogc3RyaW5nKTogbnVtYmVyIHwgdW5kZWZpbmVkIHtcbiAgICAgIGlmIChyYXcgPT09ICcnKSB7XG4gICAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgICB9XG4gICAgICBjb25zdCBuID0gTnVtYmVyKHJhdyk7XG4gICAgICBpZiAoTnVtYmVyLmlzRmluaXRlKG4pKSB7XG4gICAgICAgIHJldHVybiBuO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICB9LFxuICAgIC8vIEFjY2VwdCB1bmtub3duIHRvIHNhdGlzZnkgYmFzZSBpbnRlcmZhY2UgaW50ZXJzZWN0aW9uOyBjYWxsZXIgZW5zdXJlcyB0eXBlIGNvcnJlY3RuZXNzXG4gICAgc2VyaWFsaXplKHZhbHVlOiB1bmtub3duKTogc3RyaW5nIHtcbiAgICAgIHJldHVybiBTdHJpbmcodmFsdWUgYXMgbnVtYmVyKTtcbiAgICB9LFxuICB9IGFzIE51bWJlckZpZWxkRGVzY3JpcHRvcjtcbn1cblxuLyoqXG4gKiBDcmVhdGUgYSBzdHJpbmcgZmllbGQgZGVzY3JpcHRvciB3aXRoIGEgZ2l2ZW4gb3BlcmF0b3Igd2hpdGVsaXN0LlxuICovXG5leHBvcnQgZnVuY3Rpb24gc3RyaW5nRmllbGQoXG4gIG9wczogcmVhZG9ubHkgT3BlcmF0b3JbXSA9IERFRkFVTFRfU1RSSU5HX09QRVJBVE9SUyxcbik6IFN0cmluZ0ZpZWxkRGVzY3JpcHRvciB7XG4gIHJldHVybiB7XG4gICAga2luZDogJ3N0cmluZycsXG4gICAgb3BlcmF0b3JzOiBvcHMsXG4gICAgcGFyc2UocmF3OiBzdHJpbmcpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICAgICAgcmV0dXJuIHJhdztcbiAgICB9LFxuICAgIHNlcmlhbGl6ZSh2YWx1ZTogdW5rbm93bik6IHN0cmluZyB7XG4gICAgICByZXR1cm4gU3RyaW5nKHZhbHVlIGFzIHN0cmluZyk7XG4gICAgfSxcbiAgfSBhcyBTdHJpbmdGaWVsZERlc2NyaXB0b3I7XG59XG5cbi8qKlxuICogRnJlZXplIGFuZCByZXR1cm4gdGhlIHNjaGVtYSAoaW1tdXRhYmlsaXR5IGd1YXJkIGZvciBjb25zdW1lcnMpLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVmaW5lU2NoZW1hPFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBGaWVsZERlc2NyaXB0b3I+PihcbiAgc2NoZW1hOiBTLFxuKTogUyB7XG4gIHJldHVybiBPYmplY3QuZnJlZXplKHsgLi4uc2NoZW1hIH0pO1xufVxuIl19
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/schema.ts"],"names":[],"mappings":"AAUA,MAAM,wBAAwB,GAAwB;IACpD,IAAI;IACJ,KAAK;IACL,IAAI;IACJ,KAAK;IACL,IAAI;IACJ,KAAK;IACL,SAAS;IACT,IAAI;IACJ,KAAK;CACN,CAAC;AACF,MAAM,wBAAwB,GAAwB;IACpD,UAAU;IACV,IAAI;IACJ,IAAI;IACJ,IAAI;IACJ,KAAK;IACL,IAAI;IACJ,KAAK;CACN,CAAC;AACF,MAAM,yBAAyB,GAAwB;IACrD,IAAI;IACJ,KAAK;IACL,IAAI;IACJ,KAAK;CACN,CAAC;AACF,MAAM,sBAAsB,GAAwB,CAAC,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC;AAK/E,MAAM,UAAU,WAAW,CACzB,MAA2B,wBAAwB;IAEnD,OAAO;QACL,IAAI,EAAE,QAAQ;QACd,SAAS,EAAE,GAAG;QACd,KAAK,CAAC,GAAW;YACf,IAAI,GAAG,KAAK,EAAE,EAAE,CAAC;gBACf,OAAO,SAAS,CAAC;YACnB,CAAC;YACD,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;YACtB,IAAI,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;gBACvB,OAAO,CAAC,CAAC;YACX,CAAC;YACD,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,SAAS,CAAC,KAAc;YACtB,OAAO,MAAM,CAAC,KAAe,CAAC,CAAC;QACjC,CAAC;KACuB,CAAC;AAC7B,CAAC;AAKD,MAAM,UAAU,WAAW,CACzB,MAA2B,wBAAwB;IAEnD,OAAO;QACL,IAAI,EAAE,QAAQ;QACd,SAAS,EAAE,GAAG;QACd,KAAK,CAAC,GAAW;YACf,OAAO,GAAG,CAAC;QACb,CAAC;QACD,SAAS,CAAC,KAAc;YACtB,OAAO,MAAM,CAAC,KAAe,CAAC,CAAC;QACjC,CAAC;KACuB,CAAC;AAC7B,CAAC;AAKD,MAAM,UAAU,YAAY,CAC1B,MAA2B,yBAAyB;IAEpD,OAAO;QACL,IAAI,EAAE,SAAS;QACf,SAAS,EAAE,GAAG;QACd,KAAK,CAAC,GAAW;YACf,MAAM,UAAU,GAAG,GAAG,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YAC5C,IAAI,UAAU,KAAK,MAAM,IAAI,UAAU,KAAK,GAAG,EAAE,CAAC;gBAChD,OAAO,IAAI,CAAC;YACd,CAAC;YACD,IAAI,UAAU,KAAK,OAAO,IAAI,UAAU,KAAK,GAAG,EAAE,CAAC;gBACjD,OAAO,KAAK,CAAC;YACf,CAAC;YACD,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,SAAS,CAAC,KAAc;YACtB,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;gBACnB,OAAO,MAAM,CAAC;YAChB,CAAC;YACD,OAAO,OAAO,CAAC;QACjB,CAAC;KACwB,CAAC;AAC9B,CAAC;AAKD,MAAM,UAAU,SAAS,CACvB,MAAoB,EACpB,MAA2B,sBAAsB;IAEjD,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC;IACjC,OAAO;QACL,IAAI,EAAE,MAAM;QACZ,SAAS,EAAE,GAAG;QACd,KAAK,CAAC,GAAW;YACf,IAAI,QAAQ,CAAC,GAAG,CAAC,GAAQ,CAAC,EAAE,CAAC;gBAC3B,OAAO,GAAQ,CAAC;YAClB,CAAC;YACD,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,SAAS,CAAC,KAAc;YACtB,OAAO,MAAM,CAAC,KAAU,CAAC,CAAC;QAC5B,CAAC;KACwB,CAAC;AAC9B,CAAC;AAWD,MAAM,UAAU,WAAW,CACzB,OAA8B;IAE9B,OAAO;QACL,IAAI,EAAE,QAAQ;QACd,SAAS,EAAE,OAAO,CAAC,SAAS;QAC5B,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,SAAS,EAAE,OAAO,CAAC,SAAS;KACD,CAAC;AAChC,CAAC;AAKD,MAAM,UAAU,YAAY,CAC1B,MAAS;IAET,OAAO,MAAM,CAAC,MAAM,CAAC,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC;AACtC,CAAC","sourcesContent":["import type {\n  FieldDescriptor,\n  NumberFieldDescriptor,\n  StringFieldDescriptor,\n  BooleanFieldDescriptor,\n  EnumFieldDescriptor,\n  CustomFieldDescriptor,\n  Operator,\n} from './types.js';\n\nconst DEFAULT_NUMBER_OPERATORS: readonly Operator[] = [\n  'gt',\n  'gte',\n  'lt',\n  'lte',\n  'eq',\n  'neq',\n  'between',\n  'in',\n  'nin',\n];\nconst DEFAULT_STRING_OPERATORS: readonly Operator[] = [\n  'contains',\n  'sw',\n  'ew',\n  'eq',\n  'neq',\n  'in',\n  'nin',\n];\nconst DEFAULT_BOOLEAN_OPERATORS: readonly Operator[] = [\n  'eq',\n  'neq',\n  'in',\n  'nin',\n];\nconst DEFAULT_ENUM_OPERATORS: readonly Operator[] = ['eq', 'neq', 'in', 'nin'];\n\n/**\n * Create a numeric field descriptor with a given operator whitelist.\n */\nexport function numberField(\n  ops: readonly Operator[] = DEFAULT_NUMBER_OPERATORS,\n): NumberFieldDescriptor {\n  return {\n    kind: 'number',\n    operators: ops,\n    parse(raw: string): number | undefined {\n      if (raw === '') {\n        return undefined;\n      }\n      const n = Number(raw);\n      if (Number.isFinite(n)) {\n        return n;\n      }\n      return undefined;\n    },\n    // Accept unknown to satisfy base interface intersection; caller ensures type correctness\n    serialize(value: unknown): string {\n      return String(value as number);\n    },\n  } as NumberFieldDescriptor;\n}\n\n/**\n * Create a string field descriptor with a given operator whitelist.\n */\nexport function stringField(\n  ops: readonly Operator[] = DEFAULT_STRING_OPERATORS,\n): StringFieldDescriptor {\n  return {\n    kind: 'string',\n    operators: ops,\n    parse(raw: string): string | undefined {\n      return raw;\n    },\n    serialize(value: unknown): string {\n      return String(value as string);\n    },\n  } as StringFieldDescriptor;\n}\n\n/**\n * Create a boolean field descriptor with a given operator whitelist.\n */\nexport function booleanField(\n  ops: readonly Operator[] = DEFAULT_BOOLEAN_OPERATORS,\n): BooleanFieldDescriptor {\n  return {\n    kind: 'boolean',\n    operators: ops,\n    parse(raw: string): boolean | undefined {\n      const normalized = raw.trim().toLowerCase();\n      if (normalized === 'true' || normalized === '1') {\n        return true;\n      }\n      if (normalized === 'false' || normalized === '0') {\n        return false;\n      }\n      return undefined;\n    },\n    serialize(value: unknown): string {\n      if (value === true) {\n        return 'true';\n      }\n      return 'false';\n    },\n  } as BooleanFieldDescriptor;\n}\n\n/**\n * Create an enum field descriptor with a given operator whitelist.\n */\nexport function enumField<const T extends string>(\n  values: readonly T[],\n  ops: readonly Operator[] = DEFAULT_ENUM_OPERATORS,\n): EnumFieldDescriptor<T> {\n  const valueSet = new Set(values);\n  return {\n    kind: 'enum',\n    operators: ops,\n    parse(raw: string): T | undefined {\n      if (valueSet.has(raw as T)) {\n        return raw as T;\n      }\n      return undefined;\n    },\n    serialize(value: unknown): string {\n      return String(value as T);\n    },\n  } as EnumFieldDescriptor<T>;\n}\n\ntype CustomFieldOptions<T> = {\n  operators: readonly Operator[];\n  parse: (raw: string) => T | undefined;\n  serialize?: (value: T) => string;\n};\n\n/**\n * Create a custom field descriptor with explicit parse/serialize functions.\n */\nexport function customField<T>(\n  options: CustomFieldOptions<T>,\n): CustomFieldDescriptor<T> {\n  return {\n    kind: 'custom',\n    operators: options.operators,\n    parse: options.parse,\n    serialize: options.serialize,\n  } as CustomFieldDescriptor<T>;\n}\n\n/**\n * Freeze and return the schema (immutability guard for consumers).\n */\nexport function defineSchema<S extends Record<string, FieldDescriptor>>(\n  schema: S,\n): S {\n  return Object.freeze({ ...schema });\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"stringify.d.ts","sourceRoot":"","sources":["../../src/stringify.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,EAAY,MAAM,YAAY,CAAC;AAiBjE,wBAAgB,SAAS,CAAC,CAAC,SAAS,MAAM,EACxC,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC,EACxB,MAAM,EAAE,CAAC,GACR,MAAM,CA8DR"}
1
+ {"version":3,"file":"stringify.d.ts","sourceRoot":"","sources":["../../src/stringify.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,YAAY,EACZ,MAAM,EAGP,MAAM,YAAY,CAAC;AAmDpB,wBAAgB,SAAS,CAAC,CAAC,SAAS,MAAM,EACxC,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC,EACxB,MAAM,EAAE,CAAC,GACR,MAAM,CAkER"}
@@ -3,6 +3,34 @@ function encodeKey(field, op) {
3
3
  return field;
4
4
  return `${field}.${op}`;
5
5
  }
6
+ function serializeScalar(descriptor, value) {
7
+ if (value === undefined) {
8
+ return undefined;
9
+ }
10
+ const isPrimitive = typeof value === 'string' ||
11
+ typeof value === 'number' ||
12
+ typeof value === 'boolean';
13
+ if (isPrimitive) {
14
+ if (descriptor.serialize != null) {
15
+ try {
16
+ return descriptor.serialize(value);
17
+ }
18
+ catch {
19
+ return undefined;
20
+ }
21
+ }
22
+ return String(value);
23
+ }
24
+ if (descriptor.kind === 'custom' && descriptor.serialize != null) {
25
+ try {
26
+ return descriptor.serialize(value);
27
+ }
28
+ catch {
29
+ return undefined;
30
+ }
31
+ }
32
+ return JSON.stringify(value);
33
+ }
6
34
  export function stringify(filters, schema) {
7
35
  const parts = [];
8
36
  for (const field of Object.keys(schema)) {
@@ -14,21 +42,19 @@ export function stringify(filters, schema) {
14
42
  for (const op of descriptor.operators) {
15
43
  if (op === 'eq') {
16
44
  const scalar = value[op];
17
- if (scalar !== undefined) {
18
- let printable = scalar;
19
- if (typeof printable !== 'string' &&
20
- typeof printable !== 'number') {
21
- printable = JSON.stringify(printable);
22
- }
23
- parts.push(`${encodeKey(field, op)}=${encodeURIComponent(String(printable))}`);
45
+ const printable = serializeScalar(descriptor, scalar);
46
+ if (printable !== undefined) {
47
+ parts.push(`${encodeKey(field, op)}=${encodeURIComponent(printable)}`);
24
48
  }
25
49
  }
26
50
  else if (op === 'between') {
27
51
  const tuple = value.between;
28
52
  if (Array.isArray(tuple) && tuple.length === 2) {
29
- const first = encodeURIComponent(String(tuple[0]));
30
- const second = encodeURIComponent(String(tuple[1]));
31
- parts.push(`${encodeKey(field, op)}=${first},${second}`);
53
+ const first = serializeScalar(descriptor, tuple[0]);
54
+ const second = serializeScalar(descriptor, tuple[1]);
55
+ if (first !== undefined && second !== undefined) {
56
+ parts.push(`${encodeKey(field, op)}=${encodeURIComponent(first)},${encodeURIComponent(second)}`);
57
+ }
32
58
  }
33
59
  }
34
60
  else if (op === 'in' || op === 'nin') {
@@ -36,23 +62,26 @@ export function stringify(filters, schema) {
36
62
  if (Array.isArray(list) && list.length > 0) {
37
63
  const encodedItems = list
38
64
  .map((v) => {
39
- return encodeURIComponent(String(v));
65
+ const printable = serializeScalar(descriptor, v);
66
+ if (printable === undefined) {
67
+ return undefined;
68
+ }
69
+ return encodeURIComponent(printable);
70
+ })
71
+ .filter((item) => {
72
+ return item !== undefined;
40
73
  })
41
74
  .join(',');
42
- parts.push(`${encodeKey(field, op)}=${encodedItems}`);
75
+ if (encodedItems !== '') {
76
+ parts.push(`${encodeKey(field, op)}=${encodedItems}`);
77
+ }
43
78
  }
44
79
  }
45
80
  else {
46
81
  const scalar = value[op];
47
- if (scalar !== undefined) {
48
- let printable;
49
- if (typeof scalar === 'string' || typeof scalar === 'number') {
50
- printable = scalar;
51
- }
52
- else {
53
- printable = JSON.stringify(scalar);
54
- }
55
- parts.push(`${encodeKey(field, op)}=${encodeURIComponent(String(printable))}`);
82
+ const printable = serializeScalar(descriptor, scalar);
83
+ if (printable !== undefined) {
84
+ parts.push(`${encodeKey(field, op)}=${encodeURIComponent(printable)}`);
56
85
  }
57
86
  }
58
87
  }
@@ -60,4 +89,4 @@ export function stringify(filters, schema) {
60
89
  }
61
90
  return parts.join('&');
62
91
  }
63
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyaW5naWZ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3N0cmluZ2lmeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLQSxTQUFTLFNBQVMsQ0FBQyxLQUFhLEVBQUUsRUFBWTtJQUM1QyxJQUFJLEVBQUUsS0FBSyxJQUFJO1FBQUUsT0FBTyxLQUFLLENBQUM7SUFDOUIsT0FBTyxHQUFHLEtBQUssSUFBSSxFQUFFLEVBQUUsQ0FBQztBQUMxQixDQUFDO0FBU0QsTUFBTSxVQUFVLFNBQVMsQ0FDdkIsT0FBd0IsRUFDeEIsTUFBUztJQUVULE1BQU0sS0FBSyxHQUFhLEVBQUUsQ0FBQztJQUMzQixLQUFLLE1BQU0sS0FBSyxJQUFJLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztRQUN4QyxNQUFNLEtBQUssR0FBRyxPQUFPLENBQUMsS0FBZ0IsQ0FBQyxDQUFDO1FBQ3hDLE1BQU0sVUFBVSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNqQyxJQUFJLEtBQUssS0FBSyxTQUFTLElBQUksVUFBVSxLQUFLLFNBQVMsRUFBRSxDQUFDO1FBRXRELENBQUM7YUFBTSxDQUFDO1lBQ04sS0FBSyxNQUFNLEVBQUUsSUFBSSxVQUFVLENBQUMsU0FBUyxFQUFFLENBQUM7Z0JBQ3RDLElBQUksRUFBRSxLQUFLLElBQUksRUFBRSxDQUFDO29CQUNoQixNQUFNLE1BQU0sR0FBSSxLQUFpQyxDQUFDLEVBQUUsQ0FBQyxDQUFDO29CQUN0RCxJQUFJLE1BQU0sS0FBSyxTQUFTLEVBQUUsQ0FBQzt3QkFDekIsSUFBSSxTQUFTLEdBQVksTUFBTSxDQUFDO3dCQUNoQyxJQUNFLE9BQU8sU0FBUyxLQUFLLFFBQVE7NEJBQzdCLE9BQU8sU0FBUyxLQUFLLFFBQVEsRUFDN0IsQ0FBQzs0QkFDRCxTQUFTLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQzt3QkFDeEMsQ0FBQzt3QkFDRCxLQUFLLENBQUMsSUFBSSxDQUNSLEdBQUcsU0FBUyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsSUFBSSxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxDQUNuRSxDQUFDO29CQUNKLENBQUM7Z0JBQ0gsQ0FBQztxQkFBTSxJQUFJLEVBQUUsS0FBSyxTQUFTLEVBQUUsQ0FBQztvQkFDNUIsTUFBTSxLQUFLLEdBQUksS0FBaUMsQ0FBQyxPQUVwQyxDQUFDO29CQUNkLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsSUFBSSxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO3dCQUMvQyxNQUFNLEtBQUssR0FBRyxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQzt3QkFDbkQsTUFBTSxNQUFNLEdBQUcsa0JBQWtCLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7d0JBQ3BELEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxTQUFTLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxJQUFJLEtBQUssSUFBSSxNQUFNLEVBQUUsQ0FBQyxDQUFDO29CQUMzRCxDQUFDO2dCQUNILENBQUM7cUJBQU0sSUFBSSxFQUFFLEtBQUssSUFBSSxJQUFJLEVBQUUsS0FBSyxLQUFLLEVBQUUsQ0FBQztvQkFDdkMsTUFBTSxJQUFJLEdBQUksS0FBaUMsQ0FBQyxFQUFFLENBRXJDLENBQUM7b0JBQ2QsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7d0JBQzNDLE1BQU0sWUFBWSxHQUFHLElBQUk7NkJBQ3RCLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFOzRCQUNULE9BQU8sa0JBQWtCLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7d0JBQ3ZDLENBQUMsQ0FBQzs2QkFDRCxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7d0JBQ2IsS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLFNBQVMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLElBQUksWUFBWSxFQUFFLENBQUMsQ0FBQztvQkFDeEQsQ0FBQztnQkFDSCxDQUFDO3FCQUFNLENBQUM7b0JBQ04sTUFBTSxNQUFNLEdBQUksS0FBaUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztvQkFDdEQsSUFBSSxNQUFNLEtBQUssU0FBUyxFQUFFLENBQUM7d0JBQ3pCLElBQUksU0FBa0IsQ0FBQzt3QkFDdkIsSUFBSSxPQUFPLE1BQU0sS0FBSyxRQUFRLElBQUksT0FBTyxNQUFNLEtBQUssUUFBUSxFQUFFLENBQUM7NEJBQzdELFNBQVMsR0FBRyxNQUFNLENBQUM7d0JBQ3JCLENBQUM7NkJBQU0sQ0FBQzs0QkFDTixTQUFTLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQzt3QkFDckMsQ0FBQzt3QkFDRCxLQUFLLENBQUMsSUFBSSxDQUNSLEdBQUcsU0FBUyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsSUFBSSxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxDQUNuRSxDQUFDO29CQUNKLENBQUM7Z0JBQ0gsQ0FBQztZQUNILENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUN6QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBJbmZlckZpbHRlcnMsIFNjaGVtYSwgT3BlcmF0b3IgfSBmcm9tICcuL3R5cGVzLmpzJztcblxuLyoqXG4gKiBCdWlsZHMgdGhlIGVuY29kZWQga2V5IHBvcnRpb24gZm9yIGEgZmllbGQvb3BlcmF0b3IgcGFpciAoaW1wbGljaXQgYGVxYCBoYXMgbm8gZG90IHN1ZmZpeCkuXG4gKi9cbmZ1bmN0aW9uIGVuY29kZUtleShmaWVsZDogc3RyaW5nLCBvcDogT3BlcmF0b3IpOiBzdHJpbmcge1xuICBpZiAob3AgPT09ICdlcScpIHJldHVybiBmaWVsZDsgLy8gaW1wbGljaXQgZXF1YWxpdHkgZm9ybVxuICByZXR1cm4gYCR7ZmllbGR9LiR7b3B9YDtcbn1cblxuLyoqXG4gKiBTZXJpYWxpemVzIGEgZmlsdGVycyBvYmplY3QgaW50byBhIGNhbm9uaWNhbCBxdWVyeSBzdHJpbmcgcmVzcGVjdGluZyB0aGUgc2NoZW1hIG9wZXJhdG9yIG9yZGVyaW5nLlxuICpcbiAqIEBwYXJhbSBmaWx0ZXJzIC0gUGFyc2VkIGZpbHRlcnMgdG8gZW5jb2RlLlxuICogQHBhcmFtIHNjaGVtYSAtIFNjaGVtYSB1c2VkIHRvIGRldGVybWluZSBvcGVyYXRvciBvcmRlcmluZyBhbmQgZm9ybWF0dGluZy5cbiAqIEByZXR1cm5zIFF1ZXJ5IHN0cmluZyB3aXRob3V0IGxlYWRpbmcgcXVlc3Rpb24gbWFyay5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHN0cmluZ2lmeTxTIGV4dGVuZHMgU2NoZW1hPihcbiAgZmlsdGVyczogSW5mZXJGaWx0ZXJzPFM+LFxuICBzY2hlbWE6IFMsXG4pOiBzdHJpbmcge1xuICBjb25zdCBwYXJ0czogc3RyaW5nW10gPSBbXTtcbiAgZm9yIChjb25zdCBmaWVsZCBvZiBPYmplY3Qua2V5cyhzY2hlbWEpKSB7XG4gICAgY29uc3QgdmFsdWUgPSBmaWx0ZXJzW2ZpZWxkIGFzIGtleW9mIFNdO1xuICAgIGNvbnN0IGRlc2NyaXB0b3IgPSBzY2hlbWFbZmllbGRdO1xuICAgIGlmICh2YWx1ZSA9PT0gdW5kZWZpbmVkIHx8IGRlc2NyaXB0b3IgPT09IHVuZGVmaW5lZCkge1xuICAgICAgLy8gc2tpcCBhYnNlbnRcbiAgICB9IGVsc2Uge1xuICAgICAgZm9yIChjb25zdCBvcCBvZiBkZXNjcmlwdG9yLm9wZXJhdG9ycykge1xuICAgICAgICBpZiAob3AgPT09ICdlcScpIHtcbiAgICAgICAgICBjb25zdCBzY2FsYXIgPSAodmFsdWUgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW29wXTtcbiAgICAgICAgICBpZiAoc2NhbGFyICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGxldCBwcmludGFibGU6IHVua25vd24gPSBzY2FsYXI7XG4gICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgIHR5cGVvZiBwcmludGFibGUgIT09ICdzdHJpbmcnICYmXG4gICAgICAgICAgICAgIHR5cGVvZiBwcmludGFibGUgIT09ICdudW1iZXInXG4gICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgcHJpbnRhYmxlID0gSlNPTi5zdHJpbmdpZnkocHJpbnRhYmxlKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHBhcnRzLnB1c2goXG4gICAgICAgICAgICAgIGAke2VuY29kZUtleShmaWVsZCwgb3ApfT0ke2VuY29kZVVSSUNvbXBvbmVudChTdHJpbmcocHJpbnRhYmxlKSl9YCxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGVsc2UgaWYgKG9wID09PSAnYmV0d2VlbicpIHtcbiAgICAgICAgICBjb25zdCB0dXBsZSA9ICh2YWx1ZSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikuYmV0d2VlbiBhc1xuICAgICAgICAgICAgfCB1bmtub3duW11cbiAgICAgICAgICAgIHwgdW5kZWZpbmVkO1xuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KHR1cGxlKSAmJiB0dXBsZS5sZW5ndGggPT09IDIpIHtcbiAgICAgICAgICAgIGNvbnN0IGZpcnN0ID0gZW5jb2RlVVJJQ29tcG9uZW50KFN0cmluZyh0dXBsZVswXSkpO1xuICAgICAgICAgICAgY29uc3Qgc2Vjb25kID0gZW5jb2RlVVJJQ29tcG9uZW50KFN0cmluZyh0dXBsZVsxXSkpO1xuICAgICAgICAgICAgcGFydHMucHVzaChgJHtlbmNvZGVLZXkoZmllbGQsIG9wKX09JHtmaXJzdH0sJHtzZWNvbmR9YCk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGVsc2UgaWYgKG9wID09PSAnaW4nIHx8IG9wID09PSAnbmluJykge1xuICAgICAgICAgIGNvbnN0IGxpc3QgPSAodmFsdWUgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pW29wXSBhc1xuICAgICAgICAgICAgfCB1bmtub3duW11cbiAgICAgICAgICAgIHwgdW5kZWZpbmVkO1xuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KGxpc3QpICYmIGxpc3QubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgY29uc3QgZW5jb2RlZEl0ZW1zID0gbGlzdFxuICAgICAgICAgICAgICAubWFwKCh2KSA9PiB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGVuY29kZVVSSUNvbXBvbmVudChTdHJpbmcodikpO1xuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAuam9pbignLCcpO1xuICAgICAgICAgICAgcGFydHMucHVzaChgJHtlbmNvZGVLZXkoZmllbGQsIG9wKX09JHtlbmNvZGVkSXRlbXN9YCk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGNvbnN0IHNjYWxhciA9ICh2YWx1ZSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPilbb3BdO1xuICAgICAgICAgIGlmIChzY2FsYXIgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgbGV0IHByaW50YWJsZTogdW5rbm93bjtcbiAgICAgICAgICAgIGlmICh0eXBlb2Ygc2NhbGFyID09PSAnc3RyaW5nJyB8fCB0eXBlb2Ygc2NhbGFyID09PSAnbnVtYmVyJykge1xuICAgICAgICAgICAgICBwcmludGFibGUgPSBzY2FsYXI7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBwcmludGFibGUgPSBKU09OLnN0cmluZ2lmeShzY2FsYXIpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcGFydHMucHVzaChcbiAgICAgICAgICAgICAgYCR7ZW5jb2RlS2V5KGZpZWxkLCBvcCl9PSR7ZW5jb2RlVVJJQ29tcG9uZW50KFN0cmluZyhwcmludGFibGUpKX1gLFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcmV0dXJuIHBhcnRzLmpvaW4oJyYnKTtcbn1cbiJdfQ==
92
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stringify.js","sourceRoot":"","sources":["../../src/stringify.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  InferFilters,\n  Schema,\n  Operator,\n  FieldDescriptor,\n} from './types.js';\n\n/**\n * Builds the encoded key portion for a field/operator pair (implicit `eq` has no dot suffix).\n */\nfunction encodeKey(field: string, op: Operator): string {\n  if (op === 'eq') return field; // implicit equality form\n  return `${field}.${op}`;\n}\n\n/**\n * Serializes a scalar value using the field descriptor when possible.\n */\nfunction serializeScalar(\n  descriptor: FieldDescriptor,\n  value: unknown,\n): string | undefined {\n  if (value === undefined) {\n    return undefined;\n  }\n  const isPrimitive =\n    typeof value === 'string' ||\n    typeof value === 'number' ||\n    typeof value === 'boolean';\n  if (isPrimitive) {\n    if (descriptor.serialize != null) {\n      try {\n        return descriptor.serialize(value as never);\n      } catch {\n        return undefined;\n      }\n    }\n    return String(value);\n  }\n  if (descriptor.kind === 'custom' && descriptor.serialize != null) {\n    try {\n      return descriptor.serialize(value as never);\n    } catch {\n      return undefined;\n    }\n  }\n  return JSON.stringify(value);\n}\n\n/**\n * Serializes a filters object into a canonical query string respecting the schema operator ordering.\n *\n * @param filters - Parsed filters to encode.\n * @param schema - Schema used to determine operator ordering and formatting.\n * @returns Query string without leading question mark.\n */\nexport function stringify<S extends Schema>(\n  filters: InferFilters<S>,\n  schema: S,\n): string {\n  const parts: string[] = [];\n  for (const field of Object.keys(schema)) {\n    const value = filters[field as keyof S];\n    const descriptor = schema[field];\n    if (value === undefined || descriptor === undefined) {\n      // skip absent\n    } else {\n      for (const op of descriptor.operators) {\n        if (op === 'eq') {\n          const scalar = (value as Record<string, unknown>)[op];\n          const printable = serializeScalar(descriptor, scalar);\n          if (printable !== undefined) {\n            parts.push(\n              `${encodeKey(field, op)}=${encodeURIComponent(printable)}`,\n            );\n          }\n        } else if (op === 'between') {\n          const tuple = (value as Record<string, unknown>).between as\n            | unknown[]\n            | undefined;\n          if (Array.isArray(tuple) && tuple.length === 2) {\n            const first = serializeScalar(descriptor, tuple[0]);\n            const second = serializeScalar(descriptor, tuple[1]);\n            if (first !== undefined && second !== undefined) {\n              parts.push(\n                `${encodeKey(field, op)}=${encodeURIComponent(\n                  first,\n                )},${encodeURIComponent(second)}`,\n              );\n            }\n          }\n        } else if (op === 'in' || op === 'nin') {\n          const list = (value as Record<string, unknown>)[op] as\n            | unknown[]\n            | undefined;\n          if (Array.isArray(list) && list.length > 0) {\n            const encodedItems = list\n              .map((v) => {\n                const printable = serializeScalar(descriptor, v);\n                if (printable === undefined) {\n                  return undefined;\n                }\n                return encodeURIComponent(printable);\n              })\n              .filter((item): item is string => {\n                return item !== undefined;\n              })\n              .join(',');\n            if (encodedItems !== '') {\n              parts.push(`${encodeKey(field, op)}=${encodedItems}`);\n            }\n          }\n        } else {\n          const scalar = (value as Record<string, unknown>)[op];\n          const printable = serializeScalar(descriptor, scalar);\n          if (printable !== undefined) {\n            parts.push(\n              `${encodeKey(field, op)}=${encodeURIComponent(printable)}`,\n            );\n          }\n        }\n      }\n    }\n  }\n  return parts.join('&');\n}\n"]}
@@ -3,33 +3,31 @@ export type ScalarOperator = 'gt' | 'gte' | 'lt' | 'lte' | 'eq' | 'neq' | 'conta
3
3
  export type BetweenOperator = 'between';
4
4
  export type ListOperator = 'in' | 'nin';
5
5
  export type Operator = ScalarOperator | BetweenOperator | ListOperator;
6
- export type PrimitiveKind = 'number' | 'string';
7
- export interface FieldDescriptorBase<TKind extends PrimitiveKind> {
6
+ export type PrimitiveKind = 'number' | 'string' | 'boolean' | 'enum' | 'custom';
7
+ export interface FieldDescriptorBase<TKind extends PrimitiveKind, TValue> {
8
8
  readonly kind: TKind;
9
9
  readonly operators: readonly Operator[];
10
- readonly parse?: (raw: string) => unknown;
11
- readonly serialize?: (value: unknown) => string;
10
+ readonly parse?: (raw: string) => TValue | undefined;
11
+ readonly serialize?: (value: TValue) => string;
12
12
  }
13
- export type NumberFieldDescriptor = FieldDescriptorBase<'number'> & {
14
- readonly parse?: (raw: string) => number | undefined;
15
- readonly serialize?: (value: number) => string;
16
- };
17
- export type StringFieldDescriptor = FieldDescriptorBase<'string'> & {
18
- readonly parse?: (raw: string) => string | undefined;
19
- readonly serialize?: (value: string) => string;
20
- };
21
- export type FieldDescriptor = NumberFieldDescriptor | StringFieldDescriptor;
13
+ export type NumberFieldDescriptor = FieldDescriptorBase<'number', number>;
14
+ export type StringFieldDescriptor = FieldDescriptorBase<'string', string>;
15
+ export type BooleanFieldDescriptor = FieldDescriptorBase<'boolean', boolean>;
16
+ export type EnumFieldDescriptor<T extends string> = FieldDescriptorBase<'enum', T>;
17
+ export type CustomFieldDescriptor<T> = FieldDescriptorBase<'custom', T>;
18
+ export type FieldDescriptor = NumberFieldDescriptor | StringFieldDescriptor | BooleanFieldDescriptor | EnumFieldDescriptor<string> | CustomFieldDescriptor<unknown>;
22
19
  export type Schema = Readonly<Record<string, FieldDescriptor>>;
23
20
  type OperatorKeys<D extends FieldDescriptor> = Extract<D['operators'][number], ScalarOperator>;
24
- type ScalarMap<D extends FieldDescriptor> = D extends FieldDescriptorBase<'number'> ? Record<OperatorKeys<D>, number | undefined> : Record<OperatorKeys<D>, string | undefined>;
21
+ type FieldValue<D extends FieldDescriptor> = D extends FieldDescriptorBase<PrimitiveKind, infer TValue> ? TValue : never;
22
+ type ScalarMap<D extends FieldDescriptor> = Record<OperatorKeys<D>, FieldValue<D> | undefined>;
25
23
  type BetweenMap<D extends FieldDescriptor> = 'between' extends D['operators'][number] ? {
26
- between?: D extends FieldDescriptorBase<'number'> ? readonly [number, number] : readonly [string, string];
24
+ between?: readonly [FieldValue<D>, FieldValue<D>];
27
25
  } : Record<never, never>;
28
26
  type ListOps<D extends FieldDescriptor> = Extract<D['operators'][number], ListOperator>;
29
27
  type ListMap<D extends FieldDescriptor> = ('in' extends ListOps<D> ? {
30
- in?: D extends FieldDescriptorBase<'number'> ? readonly number[] : readonly string[];
28
+ in?: readonly FieldValue<D>[];
31
29
  } : Record<never, never>) & ('nin' extends ListOps<D> ? {
32
- nin?: D extends FieldDescriptorBase<'number'> ? readonly number[] : readonly string[];
30
+ nin?: readonly FieldValue<D>[];
33
31
  } : Record<never, never>);
34
32
  export type InferField<D extends FieldDescriptor> = Partial<ScalarMap<D> & BetweenMap<D> & ListMap<D>>;
35
33
  export type InferFilters<S extends Schema> = {
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"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"}