@plumile/filter-query 0.1.17
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/LICENSE +21 -0
- package/README.md +202 -0
- package/lib/errors.d.ts +39 -0
- package/lib/errors.d.ts.map +1 -0
- package/lib/errors.js +2 -0
- package/lib/esm/errors.d.ts +39 -0
- package/lib/esm/errors.d.ts.map +1 -0
- package/lib/esm/errors.js +2 -0
- package/lib/esm/index.d.ts +7 -0
- package/lib/esm/index.d.ts.map +1 -0
- package/lib/esm/index.js +5 -0
- package/lib/esm/mutate.d.ts +6 -0
- package/lib/esm/mutate.d.ts.map +1 -0
- package/lib/esm/mutate.js +88 -0
- package/lib/esm/parse.d.ts +3 -0
- package/lib/esm/parse.d.ts.map +1 -0
- package/lib/esm/parse.js +164 -0
- package/lib/esm/schema.d.ts +5 -0
- package/lib/esm/schema.d.ts.map +1 -0
- package/lib/esm/schema.js +55 -0
- package/lib/esm/stringify.d.ts +3 -0
- package/lib/esm/stringify.d.ts.map +1 -0
- package/lib/esm/stringify.js +50 -0
- package/lib/esm/types.d.ts +43 -0
- package/lib/esm/types.d.ts.map +1 -0
- package/lib/esm/types.js +2 -0
- package/lib/index.d.ts +7 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +5 -0
- package/lib/mutate.d.ts +6 -0
- package/lib/mutate.d.ts.map +1 -0
- package/lib/mutate.js +88 -0
- package/lib/parse.d.ts +3 -0
- package/lib/parse.d.ts.map +1 -0
- package/lib/parse.js +164 -0
- package/lib/schema.d.ts +5 -0
- package/lib/schema.d.ts.map +1 -0
- package/lib/schema.js +55 -0
- package/lib/stringify.d.ts +3 -0
- package/lib/stringify.d.ts.map +1 -0
- package/lib/stringify.js +50 -0
- package/lib/tsconfig.esm.tsbuildinfo +1 -0
- package/lib/types/errors.d.ts +39 -0
- package/lib/types/errors.d.ts.map +1 -0
- package/lib/types/index.d.ts +7 -0
- package/lib/types/index.d.ts.map +1 -0
- package/lib/types/mutate.d.ts +6 -0
- package/lib/types/mutate.d.ts.map +1 -0
- package/lib/types/parse.d.ts +3 -0
- package/lib/types/parse.d.ts.map +1 -0
- package/lib/types/schema.d.ts +5 -0
- package/lib/types/schema.d.ts.map +1 -0
- package/lib/types/stringify.d.ts +3 -0
- package/lib/types/stringify.d.ts.map +1 -0
- package/lib/types/types.d.ts +43 -0
- package/lib/types/types.d.ts.map +1 -0
- package/lib/types.d.ts +43 -0
- package/lib/types.d.ts.map +1 -0
- package/lib/types.js +2 -0
- package/package.json +43 -0
- package/src/__tests__/additional-coverage.test.ts +82 -0
- package/src/__tests__/list-edge.test.ts +26 -0
- package/src/__tests__/mutate.test.ts +33 -0
- package/src/__tests__/parse-stringify.test.ts +104 -0
- package/src/__tests__/remove-filter.test.ts +46 -0
- package/src/__tests__/schema-edge.test.ts +46 -0
- package/src/__tests__/schema-infer.test-d.ts +24 -0
- package/src/__tests__/stability-and-diagnostics.test.ts +40 -0
- package/src/errors.ts +46 -0
- package/src/index.ts +6 -0
- package/src/mutate.ts +132 -0
- package/src/parse.ts +221 -0
- package/src/schema.ts +81 -0
- package/src/stringify.ts +60 -0
- package/src/types.ts +88 -0
- package/tools/build-package.sh +5 -0
- package/tools/test-build-package.sh +4 -0
- package/tsconfig.build.json +8 -0
- package/tsconfig.esm.json +7 -0
- package/tsconfig.json +8 -0
- package/tsconfig.types.json +9 -0
- package/vitest.config.ts +7 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) Plumile and its affiliates.
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
# @plumile/filter-query
|
|
2
|
+
|
|
3
|
+
Typed, schema‑driven filter query string parser / serializer with immutable helpers and strong TypeScript inference. Zero runtime dependencies.
|
|
4
|
+
|
|
5
|
+
## Why?
|
|
6
|
+
|
|
7
|
+
Applications often encode complex filter state (numbers, ranges, multi-selects, text searches) into a URL query string. Ad‑hoc solutions easily drift: inconsistent operator names, ambiguous serialization, lost type safety, brittle parsing, and noisy re-renders. `@plumile/filter-query` gives you:
|
|
8
|
+
|
|
9
|
+
- A mandatory schema (single source of truth) declaring fields and allowed operators.
|
|
10
|
+
- Deterministic, canonical string generation (stable ordering for cache keys & SSR).
|
|
11
|
+
- Strong TypeScript inference for the shape of `filters` (no manual typings).
|
|
12
|
+
- Non‑blocking diagnostics instead of exceptions (you decide how to surface issues).
|
|
13
|
+
- Immutable, reference‑stable mutation helpers (minimize React renders / memo churn).
|
|
14
|
+
- Simple, explicit operator semantics (predictable merging & precedence rules).
|
|
15
|
+
|
|
16
|
+
## Installation
|
|
17
|
+
|
|
18
|
+
```bash
|
|
19
|
+
npm install @plumile/filter-query
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
## Core Concepts
|
|
23
|
+
|
|
24
|
+
| Concept | Description |
|
|
25
|
+
| ---------------- | --------------------------------------------------------------------------------- |
|
|
26
|
+
| Schema | Object produced by `defineSchema({ field: numberField(), ... })`. Required. |
|
|
27
|
+
| Field Descriptor | `numberField()` or `stringField()` optionally with a custom operator whitelist. |
|
|
28
|
+
| Filters Object | Parsed result shape inferred from the schema (operators become optional keys). |
|
|
29
|
+
| Diagnostics | Array of non-blocking issues (unknown field/op, invalid value, etc.). |
|
|
30
|
+
| Mutation Helpers | Pure functions returning new filter objects (or the same reference if no change). |
|
|
31
|
+
|
|
32
|
+
## Supported Operators
|
|
33
|
+
|
|
34
|
+
| Category | Operators | Syntax | Notes |
|
|
35
|
+
| ------------------ | ------------------------------------- | -------------------------- | --------------------------------------------------------------------------------- |
|
|
36
|
+
| Numeric comparison | `gt`, `gte`, `lt`, `lte`, `eq`, `neq` | `price.gt=10` | Last write wins per operator. |
|
|
37
|
+
| Textual comparison | `eq`, `neq` | `title.eq=foo` | Works for strings too. |
|
|
38
|
+
| Text search | `contains`, `sw`, `ew` | `title.contains=foo%20bar` | Raw values URL-decoded. |
|
|
39
|
+
| Range | `between` | `price.between=10,100` | Only first valid occurrence kept; duplicates yield `DuplicateBetween` diagnostic. |
|
|
40
|
+
| Inclusion lists | `in` | `id.in=1,2,3` | Multi-occurrences merge: `id.in=1,2&id.in=3` → `[1,2,3]`. |
|
|
41
|
+
| Exclusion lists | `nin` | `id.nin=4,5&...` | Same merging logic as `in`. |
|
|
42
|
+
|
|
43
|
+
### Value Parsing Rules
|
|
44
|
+
|
|
45
|
+
- Number field: attempts `Number()`, rejects `NaN` / `Infinity`.
|
|
46
|
+
- String field: raw decoded string (empty string allowed, but not produced by numeric parse).
|
|
47
|
+
- Lists: split by comma; invalid members emit `InvalidValue` and are skipped; empty result discards whole operator.
|
|
48
|
+
- Between: must have exactly 2 comma‑separated values; otherwise `InvalidArity`.
|
|
49
|
+
- Duplicate `between`: first valid stored, later ones produce `DuplicateBetween`.
|
|
50
|
+
|
|
51
|
+
## Quick Start
|
|
52
|
+
|
|
53
|
+
```ts
|
|
54
|
+
import {
|
|
55
|
+
defineSchema,
|
|
56
|
+
numberField,
|
|
57
|
+
stringField,
|
|
58
|
+
parse,
|
|
59
|
+
stringify,
|
|
60
|
+
setFilter,
|
|
61
|
+
} from '@plumile/filter-query';
|
|
62
|
+
|
|
63
|
+
const schema = defineSchema({
|
|
64
|
+
price: numberField(),
|
|
65
|
+
title: stringField(),
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
// Parse URL search (leading '?' optional)
|
|
69
|
+
const { filters, diagnostics } = parse(
|
|
70
|
+
'price.gt=10&title.contains=foo%20bar',
|
|
71
|
+
schema,
|
|
72
|
+
);
|
|
73
|
+
// filters.price?.gt === 10, filters.title?.contains === 'foo bar'
|
|
74
|
+
// diagnostics: []
|
|
75
|
+
|
|
76
|
+
// Apply immutable mutation
|
|
77
|
+
const updated = setFilter(filters, schema, 'price', 'between', [10, 100]);
|
|
78
|
+
|
|
79
|
+
// Serialize back (canonical ordering: schema field order, then operator order)
|
|
80
|
+
const qs = stringify(updated, schema);
|
|
81
|
+
// => price.gt=10&price.between=10,100&title.contains=foo%20bar
|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
## Schema Definition
|
|
85
|
+
|
|
86
|
+
```ts
|
|
87
|
+
const schema = defineSchema({
|
|
88
|
+
price: numberField(), // full numeric operator set
|
|
89
|
+
title: stringField(['contains', 'sw']), // restrict to subset
|
|
90
|
+
});
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
Both helpers accept an optional operator list to _whitelist_ allowed operators (anything not listed becomes `UnknownOperator` if present in input).
|
|
94
|
+
|
|
95
|
+
### Operator Defaults
|
|
96
|
+
|
|
97
|
+
- `numberField()` default: `gt,gte,lt,lte,eq,neq,between,in,nin`
|
|
98
|
+
- `stringField()` default: `contains,sw,ew,eq,neq,in,nin`
|
|
99
|
+
|
|
100
|
+
## Diagnostics
|
|
101
|
+
|
|
102
|
+
Returned shape:
|
|
103
|
+
|
|
104
|
+
```ts
|
|
105
|
+
interface DiagnosticBase {
|
|
106
|
+
kind: string;
|
|
107
|
+
field?: string;
|
|
108
|
+
operator?: string;
|
|
109
|
+
detail?: string;
|
|
110
|
+
}
|
|
111
|
+
// Kinds: 'UnknownField' | 'UnknownOperator' | 'InvalidValue' | 'InvalidArity' | 'DuplicateBetween' | 'DecodeError'
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
Parsing never throws for content errors; all issues are accumulated. You decide how to surface them (log panel, dev overlay, UI badges, etc.).
|
|
115
|
+
|
|
116
|
+
## Filters Object & Type Inference
|
|
117
|
+
|
|
118
|
+
From a schema:
|
|
119
|
+
|
|
120
|
+
```ts
|
|
121
|
+
const schema = defineSchema({ price: numberField(), title: stringField() });
|
|
122
|
+
```
|
|
123
|
+
|
|
124
|
+
The inferred `filters` type is roughly:
|
|
125
|
+
|
|
126
|
+
```ts
|
|
127
|
+
{
|
|
128
|
+
price?: {
|
|
129
|
+
gt?: number; gte?: number; lt?: number; lte?: number; eq?: number; neq?: number;
|
|
130
|
+
between?: readonly [number, number];
|
|
131
|
+
in?: readonly number[]; nin?: readonly number[];
|
|
132
|
+
};
|
|
133
|
+
title?: {
|
|
134
|
+
contains?: string; sw?: string; ew?: string; eq?: string; neq?: string;
|
|
135
|
+
in?: readonly string[]; nin?: readonly string[];
|
|
136
|
+
};
|
|
137
|
+
}
|
|
138
|
+
```
|
|
139
|
+
|
|
140
|
+
Everything is optional so you can build partial filters progressively.
|
|
141
|
+
|
|
142
|
+
## Mutation Helpers
|
|
143
|
+
|
|
144
|
+
```ts
|
|
145
|
+
setFilter(filters, schema, 'price', 'gt', 10); // add/update value
|
|
146
|
+
setFilter(filters, schema, 'price', 'gt', undefined); // remove operator key
|
|
147
|
+
removeFilter(filters, 'price'); // drop entire field
|
|
148
|
+
removeFilter(filters, 'price', 'gt'); // drop one operator
|
|
149
|
+
mergeFilters(base, patch); // shallow merge per field
|
|
150
|
+
```
|
|
151
|
+
|
|
152
|
+
### Reference Stability
|
|
153
|
+
|
|
154
|
+
- If a mutation results in no semantic change → the _same_ object reference is returned.
|
|
155
|
+
- Enables cheap memoization (`useMemo`, React context selectors, etc.).
|
|
156
|
+
|
|
157
|
+
## Serialization Rules
|
|
158
|
+
|
|
159
|
+
1. Field iteration order = schema object key order (stable in modern JS for own string keys).
|
|
160
|
+
2. Operator order = `descriptor.operators` order.
|
|
161
|
+
3. Operators with no value / empty arrays are skipped.
|
|
162
|
+
4. List operators keep insertion order across multi-occurrences.
|
|
163
|
+
5. Output omits leading `?` (caller decides how to prefix).
|
|
164
|
+
|
|
165
|
+
## Edge Cases & Examples
|
|
166
|
+
|
|
167
|
+
| Input | Result | Diagnostics |
|
|
168
|
+
| ------------------------------------- | -------------------------- | ------------------------------------------- |
|
|
169
|
+
| `price.gt=abc` | `filters.price?.gt` absent | `InvalidValue` |
|
|
170
|
+
| `price.between=1,2,3` | none stored | `InvalidArity` |
|
|
171
|
+
| `price.between=1,5&price.between=2,6` | `[1,5]` | `DuplicateBetween` |
|
|
172
|
+
| `price.in=` | ignored | none (empty split produces no valid values) |
|
|
173
|
+
| `unknown.gt=5` | ignored | `UnknownField` |
|
|
174
|
+
| `price.xyz=5` | ignored | `UnknownOperator` |
|
|
175
|
+
| `title.contains=x%ZZ` | ignored | `DecodeError` + maybe others |
|
|
176
|
+
|
|
177
|
+
## Custom Operator Subsets
|
|
178
|
+
|
|
179
|
+
```ts
|
|
180
|
+
const minimal = defineSchema({ price: numberField(['gt', 'lt']) });
|
|
181
|
+
// parse('price.eq=5', minimal) → diagnostics: UnknownOperator
|
|
182
|
+
```
|
|
183
|
+
|
|
184
|
+
## Performance Notes
|
|
185
|
+
|
|
186
|
+
- Parsing is single pass over pairs; only allocates for: decoded strings, filter field objects when first used, diagnostics entries.
|
|
187
|
+
- Mutations avoid cloning unchanged branches (shallow one-level cloning only when something changes).
|
|
188
|
+
|
|
189
|
+
## FAQ
|
|
190
|
+
|
|
191
|
+
**Q: Why not support arbitrary operator names?**
|
|
192
|
+
To keep type inference precise and predictable. Add new core operators via a PR if they are broadly useful.
|
|
193
|
+
|
|
194
|
+
**Q: How do I clear everything?**
|
|
195
|
+
Just use an empty object `{}` or parse an empty string and replace your state reference.
|
|
196
|
+
|
|
197
|
+
**Q: Does order of repeated list operators matter?**
|
|
198
|
+
Yes, items are appended in encounter order, preserving user intent (e.g. prioritized IDs).
|
|
199
|
+
|
|
200
|
+
## License
|
|
201
|
+
|
|
202
|
+
MIT
|
package/lib/errors.d.ts
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
export type Diagnostic = UnknownFieldDiagnostic | UnknownOperatorDiagnostic | InvalidValueDiagnostic | InvalidArityDiagnostic | DuplicateBetweenDiagnostic | DecodeErrorDiagnostic;
|
|
2
|
+
export interface BaseDiagnostic {
|
|
3
|
+
readonly kind: string;
|
|
4
|
+
readonly field?: string;
|
|
5
|
+
readonly operator?: string;
|
|
6
|
+
readonly detail?: string;
|
|
7
|
+
}
|
|
8
|
+
export interface UnknownFieldDiagnostic extends BaseDiagnostic {
|
|
9
|
+
kind: 'UnknownField';
|
|
10
|
+
field: string;
|
|
11
|
+
}
|
|
12
|
+
export interface UnknownOperatorDiagnostic extends BaseDiagnostic {
|
|
13
|
+
kind: 'UnknownOperator';
|
|
14
|
+
field: string;
|
|
15
|
+
operator: string;
|
|
16
|
+
}
|
|
17
|
+
export interface InvalidValueDiagnostic extends BaseDiagnostic {
|
|
18
|
+
kind: 'InvalidValue';
|
|
19
|
+
field: string;
|
|
20
|
+
operator: string;
|
|
21
|
+
detail?: string;
|
|
22
|
+
}
|
|
23
|
+
export interface InvalidArityDiagnostic extends BaseDiagnostic {
|
|
24
|
+
kind: 'InvalidArity';
|
|
25
|
+
field: string;
|
|
26
|
+
operator: string;
|
|
27
|
+
detail: string;
|
|
28
|
+
}
|
|
29
|
+
export interface DuplicateBetweenDiagnostic extends BaseDiagnostic {
|
|
30
|
+
kind: 'DuplicateBetween';
|
|
31
|
+
field: string;
|
|
32
|
+
}
|
|
33
|
+
export interface DecodeErrorDiagnostic extends BaseDiagnostic {
|
|
34
|
+
kind: 'DecodeError';
|
|
35
|
+
field: string;
|
|
36
|
+
operator: string;
|
|
37
|
+
detail: string;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=errors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,UAAU,GAClB,sBAAsB,GACtB,yBAAyB,GACzB,sBAAsB,GACtB,sBAAsB,GACtB,0BAA0B,GAC1B,qBAAqB,CAAC;AAE1B,MAAM,WAAW,cAAc;IAC7B,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC;IACxB,QAAQ,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;CACf;AACD,MAAM,WAAW,yBAA0B,SAAQ,cAAc;IAC/D,IAAI,EAAE,iBAAiB,CAAC;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;CAClB;AACD,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AACD,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;CAChB;AACD,MAAM,WAAW,0BAA2B,SAAQ,cAAc;IAChE,IAAI,EAAE,kBAAkB,CAAC;IACzB,KAAK,EAAE,MAAM,CAAC;CACf;AACD,MAAM,WAAW,qBAAsB,SAAQ,cAAc;IAC3D,IAAI,EAAE,aAAa,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;CAChB"}
|
package/lib/errors.js
ADDED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgRGlhZ25vc3RpYyA9XG4gIHwgVW5rbm93bkZpZWxkRGlhZ25vc3RpY1xuICB8IFVua25vd25PcGVyYXRvckRpYWdub3N0aWNcbiAgfCBJbnZhbGlkVmFsdWVEaWFnbm9zdGljXG4gIHwgSW52YWxpZEFyaXR5RGlhZ25vc3RpY1xuICB8IER1cGxpY2F0ZUJldHdlZW5EaWFnbm9zdGljXG4gIHwgRGVjb2RlRXJyb3JEaWFnbm9zdGljO1xuXG5leHBvcnQgaW50ZXJmYWNlIEJhc2VEaWFnbm9zdGljIHtcbiAgcmVhZG9ubHkga2luZDogc3RyaW5nO1xuICByZWFkb25seSBmaWVsZD86IHN0cmluZztcbiAgcmVhZG9ubHkgb3BlcmF0b3I/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRldGFpbD86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVbmtub3duRmllbGREaWFnbm9zdGljIGV4dGVuZHMgQmFzZURpYWdub3N0aWMge1xuICBraW5kOiAnVW5rbm93bkZpZWxkJztcbiAgZmllbGQ6IHN0cmluZztcbn1cbmV4cG9ydCBpbnRlcmZhY2UgVW5rbm93bk9wZXJhdG9yRGlhZ25vc3RpYyBleHRlbmRzIEJhc2VEaWFnbm9zdGljIHtcbiAga2luZDogJ1Vua25vd25PcGVyYXRvcic7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIG9wZXJhdG9yOiBzdHJpbmc7XG59XG5leHBvcnQgaW50ZXJmYWNlIEludmFsaWRWYWx1ZURpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdJbnZhbGlkVmFsdWUnO1xuICBmaWVsZDogc3RyaW5nO1xuICBvcGVyYXRvcjogc3RyaW5nO1xuICBkZXRhaWw/OiBzdHJpbmc7XG59XG5leHBvcnQgaW50ZXJmYWNlIEludmFsaWRBcml0eURpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdJbnZhbGlkQXJpdHknO1xuICBmaWVsZDogc3RyaW5nO1xuICBvcGVyYXRvcjogc3RyaW5nO1xuICBkZXRhaWw6IHN0cmluZzsgLy8gZXhwZWN0ZWQgYXJpdHkgaW5mb1xufVxuZXhwb3J0IGludGVyZmFjZSBEdXBsaWNhdGVCZXR3ZWVuRGlhZ25vc3RpYyBleHRlbmRzIEJhc2VEaWFnbm9zdGljIHtcbiAga2luZDogJ0R1cGxpY2F0ZUJldHdlZW4nO1xuICBmaWVsZDogc3RyaW5nO1xufVxuZXhwb3J0IGludGVyZmFjZSBEZWNvZGVFcnJvckRpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdEZWNvZGVFcnJvcic7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIG9wZXJhdG9yOiBzdHJpbmc7XG4gIGRldGFpbDogc3RyaW5nO1xufVxuIl19
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
export type Diagnostic = UnknownFieldDiagnostic | UnknownOperatorDiagnostic | InvalidValueDiagnostic | InvalidArityDiagnostic | DuplicateBetweenDiagnostic | DecodeErrorDiagnostic;
|
|
2
|
+
export interface BaseDiagnostic {
|
|
3
|
+
readonly kind: string;
|
|
4
|
+
readonly field?: string;
|
|
5
|
+
readonly operator?: string;
|
|
6
|
+
readonly detail?: string;
|
|
7
|
+
}
|
|
8
|
+
export interface UnknownFieldDiagnostic extends BaseDiagnostic {
|
|
9
|
+
kind: 'UnknownField';
|
|
10
|
+
field: string;
|
|
11
|
+
}
|
|
12
|
+
export interface UnknownOperatorDiagnostic extends BaseDiagnostic {
|
|
13
|
+
kind: 'UnknownOperator';
|
|
14
|
+
field: string;
|
|
15
|
+
operator: string;
|
|
16
|
+
}
|
|
17
|
+
export interface InvalidValueDiagnostic extends BaseDiagnostic {
|
|
18
|
+
kind: 'InvalidValue';
|
|
19
|
+
field: string;
|
|
20
|
+
operator: string;
|
|
21
|
+
detail?: string;
|
|
22
|
+
}
|
|
23
|
+
export interface InvalidArityDiagnostic extends BaseDiagnostic {
|
|
24
|
+
kind: 'InvalidArity';
|
|
25
|
+
field: string;
|
|
26
|
+
operator: string;
|
|
27
|
+
detail: string;
|
|
28
|
+
}
|
|
29
|
+
export interface DuplicateBetweenDiagnostic extends BaseDiagnostic {
|
|
30
|
+
kind: 'DuplicateBetween';
|
|
31
|
+
field: string;
|
|
32
|
+
}
|
|
33
|
+
export interface DecodeErrorDiagnostic extends BaseDiagnostic {
|
|
34
|
+
kind: 'DecodeError';
|
|
35
|
+
field: string;
|
|
36
|
+
operator: string;
|
|
37
|
+
detail: string;
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=errors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../../src/errors.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,UAAU,GAClB,sBAAsB,GACtB,yBAAyB,GACzB,sBAAsB,GACtB,sBAAsB,GACtB,0BAA0B,GAC1B,qBAAqB,CAAC;AAE1B,MAAM,WAAW,cAAc;IAC7B,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC;IACxB,QAAQ,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAC3B,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;CACf;AACD,MAAM,WAAW,yBAA0B,SAAQ,cAAc;IAC/D,IAAI,EAAE,iBAAiB,CAAC;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;CAClB;AACD,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AACD,MAAM,WAAW,sBAAuB,SAAQ,cAAc;IAC5D,IAAI,EAAE,cAAc,CAAC;IACrB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;CAChB;AACD,MAAM,WAAW,0BAA2B,SAAQ,cAAc;IAChE,IAAI,EAAE,kBAAkB,CAAC;IACzB,KAAK,EAAE,MAAM,CAAC;CACf;AACD,MAAM,WAAW,qBAAsB,SAAQ,cAAc;IAC3D,IAAI,EAAE,aAAa,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,MAAM,CAAC;CAChB"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgRGlhZ25vc3RpYyA9XG4gIHwgVW5rbm93bkZpZWxkRGlhZ25vc3RpY1xuICB8IFVua25vd25PcGVyYXRvckRpYWdub3N0aWNcbiAgfCBJbnZhbGlkVmFsdWVEaWFnbm9zdGljXG4gIHwgSW52YWxpZEFyaXR5RGlhZ25vc3RpY1xuICB8IER1cGxpY2F0ZUJldHdlZW5EaWFnbm9zdGljXG4gIHwgRGVjb2RlRXJyb3JEaWFnbm9zdGljO1xuXG5leHBvcnQgaW50ZXJmYWNlIEJhc2VEaWFnbm9zdGljIHtcbiAgcmVhZG9ubHkga2luZDogc3RyaW5nO1xuICByZWFkb25seSBmaWVsZD86IHN0cmluZztcbiAgcmVhZG9ubHkgb3BlcmF0b3I/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRldGFpbD86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVbmtub3duRmllbGREaWFnbm9zdGljIGV4dGVuZHMgQmFzZURpYWdub3N0aWMge1xuICBraW5kOiAnVW5rbm93bkZpZWxkJztcbiAgZmllbGQ6IHN0cmluZztcbn1cbmV4cG9ydCBpbnRlcmZhY2UgVW5rbm93bk9wZXJhdG9yRGlhZ25vc3RpYyBleHRlbmRzIEJhc2VEaWFnbm9zdGljIHtcbiAga2luZDogJ1Vua25vd25PcGVyYXRvcic7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIG9wZXJhdG9yOiBzdHJpbmc7XG59XG5leHBvcnQgaW50ZXJmYWNlIEludmFsaWRWYWx1ZURpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdJbnZhbGlkVmFsdWUnO1xuICBmaWVsZDogc3RyaW5nO1xuICBvcGVyYXRvcjogc3RyaW5nO1xuICBkZXRhaWw/OiBzdHJpbmc7XG59XG5leHBvcnQgaW50ZXJmYWNlIEludmFsaWRBcml0eURpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdJbnZhbGlkQXJpdHknO1xuICBmaWVsZDogc3RyaW5nO1xuICBvcGVyYXRvcjogc3RyaW5nO1xuICBkZXRhaWw6IHN0cmluZzsgLy8gZXhwZWN0ZWQgYXJpdHkgaW5mb1xufVxuZXhwb3J0IGludGVyZmFjZSBEdXBsaWNhdGVCZXR3ZWVuRGlhZ25vc3RpYyBleHRlbmRzIEJhc2VEaWFnbm9zdGljIHtcbiAga2luZDogJ0R1cGxpY2F0ZUJldHdlZW4nO1xuICBmaWVsZDogc3RyaW5nO1xufVxuZXhwb3J0IGludGVyZmFjZSBEZWNvZGVFcnJvckRpYWdub3N0aWMgZXh0ZW5kcyBCYXNlRGlhZ25vc3RpYyB7XG4gIGtpbmQ6ICdEZWNvZGVFcnJvcic7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIG9wZXJhdG9yOiBzdHJpbmc7XG4gIGRldGFpbDogc3RyaW5nO1xufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,mBAAmB,YAAY,CAAC;AAChC,cAAc,aAAa,CAAC;AAC5B,mBAAmB,aAAa,CAAC;AACjC,cAAc,YAAY,CAAC;AAC3B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,aAAa,CAAC"}
|
package/lib/esm/index.js
ADDED
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export * from './schema.js';
|
|
2
|
+
export * from './parse.js';
|
|
3
|
+
export * from './stringify.js';
|
|
4
|
+
export * from './mutate.js';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsY0FBYyxhQUFhLENBQUM7QUFFNUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlICogZnJvbSAnLi90eXBlcy5qcyc7XG5leHBvcnQgKiBmcm9tICcuL3NjaGVtYS5qcyc7XG5leHBvcnQgdHlwZSAqIGZyb20gJy4vZXJyb3JzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vcGFyc2UuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdHJpbmdpZnkuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9tdXRhdGUuanMnO1xuIl19
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { InferFilters, Schema } from './types.js';
|
|
2
|
+
export declare function setFilter<S extends Schema, F extends keyof S, O extends keyof NonNullable<InferFilters<S>[F]>>(filters: Readonly<InferFilters<S>>, _schema: S, field: F, operator: O, value: NonNullable<InferFilters<S>[F]>[O] | undefined): InferFilters<S>;
|
|
3
|
+
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>;
|
|
4
|
+
export declare function mergeFilters<S extends Schema>(base: Readonly<InferFilters<S>>, patch: Readonly<InferFilters<S>>): InferFilters<S>;
|
|
5
|
+
export declare function isEmpty<S extends Schema>(filters: Readonly<InferFilters<S>>): boolean;
|
|
6
|
+
//# sourceMappingURL=mutate.d.ts.map
|
|
@@ -0,0 +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;AAmCvD,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;AAKD,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;AAKD,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;AAGD,wBAAgB,OAAO,CAAC,CAAC,SAAS,MAAM,EACtC,OAAO,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GACjC,OAAO,CAET"}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
function shallowEqual(a, b) {
|
|
2
|
+
if (a === b) {
|
|
3
|
+
return true;
|
|
4
|
+
}
|
|
5
|
+
if (typeof a !== 'object' ||
|
|
6
|
+
a === null ||
|
|
7
|
+
typeof b !== 'object' ||
|
|
8
|
+
b === null) {
|
|
9
|
+
return false;
|
|
10
|
+
}
|
|
11
|
+
const aObj = a;
|
|
12
|
+
const bObj = b;
|
|
13
|
+
const aKeys = Object.keys(aObj);
|
|
14
|
+
const bKeys = Object.keys(bObj);
|
|
15
|
+
if (aKeys.length !== bKeys.length) {
|
|
16
|
+
return false;
|
|
17
|
+
}
|
|
18
|
+
for (const k of aKeys) {
|
|
19
|
+
if (aObj[k] !== bObj[k]) {
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return true;
|
|
24
|
+
}
|
|
25
|
+
export function setFilter(filters, _schema, field, operator, value) {
|
|
26
|
+
const currentField = filters[field];
|
|
27
|
+
if (value === undefined) {
|
|
28
|
+
if (currentField === undefined) {
|
|
29
|
+
return filters;
|
|
30
|
+
}
|
|
31
|
+
const { [operator]: _removed, ...rest } = currentField;
|
|
32
|
+
if (Object.keys(rest).length === 0) {
|
|
33
|
+
const { [field]: _f, ...clone } = filters;
|
|
34
|
+
return clone;
|
|
35
|
+
}
|
|
36
|
+
return { ...filters, [field]: rest };
|
|
37
|
+
}
|
|
38
|
+
const newField = {
|
|
39
|
+
...currentField,
|
|
40
|
+
[operator]: value,
|
|
41
|
+
};
|
|
42
|
+
if (currentField !== undefined && shallowEqual(currentField, newField)) {
|
|
43
|
+
return filters;
|
|
44
|
+
}
|
|
45
|
+
return { ...filters, [field]: newField };
|
|
46
|
+
}
|
|
47
|
+
export function removeFilter(filters, field, operator) {
|
|
48
|
+
const currentField = filters[field];
|
|
49
|
+
if (currentField === undefined) {
|
|
50
|
+
return filters;
|
|
51
|
+
}
|
|
52
|
+
if (operator === undefined) {
|
|
53
|
+
const { [field]: _removed, ...clone } = filters;
|
|
54
|
+
return clone;
|
|
55
|
+
}
|
|
56
|
+
const { [operator]: _opRemoved, ...rest } = currentField;
|
|
57
|
+
if (Object.keys(rest).length === 0) {
|
|
58
|
+
const { [field]: _f, ...clone } = filters;
|
|
59
|
+
return clone;
|
|
60
|
+
}
|
|
61
|
+
return { ...filters, [field]: rest };
|
|
62
|
+
}
|
|
63
|
+
export function mergeFilters(base, patch) {
|
|
64
|
+
let changed = false;
|
|
65
|
+
const result = { ...base };
|
|
66
|
+
for (const key of Object.keys(patch)) {
|
|
67
|
+
const existing = base[key];
|
|
68
|
+
const incoming = patch[key];
|
|
69
|
+
if (incoming === undefined) {
|
|
70
|
+
}
|
|
71
|
+
else if (existing === undefined) {
|
|
72
|
+
result[key] = incoming;
|
|
73
|
+
changed = true;
|
|
74
|
+
}
|
|
75
|
+
else if (!shallowEqual(existing, incoming)) {
|
|
76
|
+
result[key] = incoming;
|
|
77
|
+
changed = true;
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
if (changed) {
|
|
81
|
+
return result;
|
|
82
|
+
}
|
|
83
|
+
return base;
|
|
84
|
+
}
|
|
85
|
+
export function isEmpty(filters) {
|
|
86
|
+
return Object.keys(filters).length === 0;
|
|
87
|
+
}
|
|
88
|
+
//# 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 * Shallow object equality (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\n/**\n * Set or unset a specific operator value for a field.\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 * Remove either an entire field or a specific operator inside a 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 * Merge two filters objects preserving reference identity for unchanged fields.\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/** Determine if the filters object is empty (no fields). */\nexport function isEmpty<S extends Schema>(\n  filters: Readonly<InferFilters<S>>,\n): boolean {\n  return Object.keys(filters).length === 0;\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parse.d.ts","sourceRoot":"","sources":["../../src/parse.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,MAAM,EAMN,WAAW,EACZ,MAAM,YAAY,CAAC;AAmKpB,wBAAgB,KAAK,CAAC,CAAC,SAAS,MAAM,EACpC,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,CAAC,GACR,WAAW,CAAC,CAAC,CAAC,CA8ChB"}
|