@overture-stack/sqon-builder
v1.1.0
Published
SQON creation and manipulation library
Downloads
390
Readme
SQON Builder
SQONBuilder is a utility for the creation and manipulation of SQON filter objects. SQON is an acronym for "Serialized Query Object Notation" and was defined as the filter syntax for Arranger.
Along with the SQONBuilder, this library exports TypeScript type definitions for a SQON and the matching schema validation objects that you can use to independently validate the structure of a SQON.
Arranger Compatibility
Important note: This utility is only compatible with Arranger v3+
Arranger 3 clarified the SQON property naming, changing field
to be fieldName
in every Filter. This builder and exported type definitions use this syntax.
Table of Contents
How to Use
Filters
To create a SQON that filters one property by value:
import SQONBuilder from '@overture-stack/sqon-builder';
SQONBuilder.in('name', ['Jim', 'Bob']);
Produces a SQON with the following content:
{
"op": "in",
"content": {
"fieldName": "name",
"value": ["Jim", "Bob"]
}
}
There are currently 3 filters available (to be expanded to match the full SQON specification):
| Filter | Value Type | Description |
| :--------: | :-----------------------------------------: | :----------------------------------------------------------------------------: |
| in
| Array<string | number> | string | number | In - field must match the provided value or be included in the array of values |
| gt
| number | Greater Than - field value must be greater than the given number |
| lt
| number | Lesser Than - field value must be lesser than the given number |
A SQON can chain multiple of these filters together into a single SQON that requires all the provided conditions. By default, these are grouped into an 'and' operator:
SQONBuilder.in('name', ['Jim', 'Bob']).gt('score', 9000).lt('age', 100);
Creates the SQON:
{
"op": "and",
"content": [
{ "op": "in", "content": { "fieldName": "name", "value": ["Jim", "Bob"] } },
{ "op": "gt", "content": { "fieldName": "score", "value": 9000 } },
{ "op": "lt", "content": { "fieldName": "age", "value": 100 } }
]
}
Combining Multiple Filters
Every SQON can be combined with other SQONs through the boolean combinations and
, or
, and not
:
const nameFilter = SQONBuilder.in('name', ['Jim', 'Bob']);
const scoreFilter = SQONBuilder.gt('score', 9000);
SQONBuilder.or([nameFilter, scoreFilter]);
Result:
{
"op": "or",
"content": [
{ "op": "in", "content": { "fieldName": "name", "value": ["Jim", "Bob"] } },
{ "op": "gt", "content": { "fieldName": "score", "value": 9000 } }
]
}
A SQON can also chain these operations like with filters to combine with other SQONs. Chaining another operator onto the right of a builder will wrap the existing SQON in that operator, combined with the provided content.
Each Combination Operator can accept a SQON or array of SQONs.
const score = SQONBuilder.gt('score', 9000);
const age = SQONBuilder.lt('age', 100);
const name = SQONBuilder.in('name', ['Jim', 'Bob']);
score.or(age).and(name);
This is equivalent to:
SQON.and([
SQON.or([
score,
age
])
name
]);
Result:
{
"op": "and",
"content": [
{
"op": "or",
"content": [
{ "op": "gt", "content": { "fieldName": "score", "value": 9000 } },
{ "op": "lt", "content": { "fieldName": "age", "value": 100 } }
]
},
{ "op": "in", "content": { "fieldName": "name", "value": ["Jim", "Bob"] } }
]
}
The .not()
combination will wrap the existing content in an and
operation, and then insert a not
operator surrounding the argument:
SQONBuilder.lt('age', 100).not(SQONBuilder.in('status', ['DENIED']) );
Results in:
{
"op": "and",
"content": [
{ "op": "lt", "content": { "fieldName": "age", "value": 100 } },
{
"op": "not",
"content": [
{
"op": "in",
"content": { "fieldName": "status", "value": ["DENIED"] }
}
]
}
]
}
SQON Output to String or Object
The SQONBuilder object can be passed directly to most network request libraries since it contains the properties of the SQON. In cases where a string is required or the object with builder functions cannot be used, the builder can be output as a string with .toString()
or as a plain object as .toValue()
.
const builder = SQONBuilder.in('name', ['Jim', 'Bob']);
builder.toString();
// '{"op":"in","content":{"fieldName":"name","value":["Jim","Bob"]}}'
builder.toValue();
// { op: 'in', content: { fieldName: 'name', value: [ 'Jim', 'Bob' ] } }
API
SQONBuilder default export
SQONBuilder(sqon: SQONBuilder | SQON | string)
The package default export is the SQONBuilder
. This is a function that will generate a SQONBuilder
object from another SQONBuilder
, a SQON
object, or JSON string
.
Note: This will attempt to parse the provided string as JSON and will then validate that the contents are a valid SQON. If the provided string cannot be parsed from JSON a
SyntaxError
will be thrown. If the parsed string is not a valid SQON, aZodError
will be thrown.
Example: const builder: SQONBuilder = SQONBuilder({op: 'in', content: {fieldName: 'name', value: ['Jim']}});
The SQONBuilder
function also provides many static methods that can be used to generate a new SQONBuilder without having a SQON to start with.
Example: const builder: SQONBuilder = SQONBuilder.in('name', 'Jim');
The returned SQONBuilder
object stores the value of the generated SQON which can be accessed as a string with .toString()
or as an object with .toValue()
. From the SQONBuilder
object you can now combine the stored SQON value with other filters or combination operations.
Example: builder.lt('age','30').gt('score',95);
Filter: In
SQONBuilder.in(fieldName: string, value: ArrayFilterValue) => SQONBuilder
Creates a filter requiring the given field to have one of the given values. The value matches the ArrayFilterValue
type which representes a string, a number, or an array of strings and numbers.
Example: SQONBuilder.in('name',['Jim','Bob'])
Filter: Greater Than
SQONBuilder.gt(fieldName: string, value: number) => SQONBuilder
Greater Than operator. Create a filter requiring the given field to be greater than the given value.
Example: SQONBuilder.gt('age',21)
Filter: Lesser Than
SQONBuilder.lt(fieldName: string, value: number) => SQONBuilder
Lesser Than operator. Create a filter requiring the given field to be lesser than the given value.
Example: SQONBuilder.lt('count', 100)
Combine: And
SQONBuilder.and(sqon: SQON | SQON[], pivot?: string) => SQONBuilder
Wrap the provided SQONs in an and
operator, creating a filter requiring all the nested filters to be true.
Example: SQONBuilder.and( [someSqon, anotherSqon] )
Combine: Or
SQONBuilder.or(sqon: SQON | SQON[], pivot?: string) => SQONBuilder
Wrap the provided SQONs in an and
operator, creating a filter requiring at least one of the nested filters to be true.
Example: SQONBuilder.or( [someSqon, anotherSqon] )
Combine: Not
SQONBuilder.not(sqon: SQON | SQON[], pivot?: string) => SQONBuilder
Wrap the provided SQONs in a not
operator, creating a filter requiring all the nested filters to be false.
Example: SQONBuilder.not( [someSqon] )
From
SQONBuilder.from(input: unknown) => SQONBuilder
Build a new SQON from a string or from a JSON object.
Example with string:
SQONBuilder.from(
'{"op":"and","content":[{"op":"in","content":{"fieldName":"name","value":"Tim"}},{"op":"gt","content":{"fieldName":"age","value":"19"}}]}',
);
This differs from the default function in that it will accept any input (i.e. type unknown
). The use case for this function is when you have a JS object that could potentially be a valid SQON, you can pass this uncheckedObject
to SQONBuilder.from(uncheckedObject)
and validation will be performed before the SQONBuilder
is returned, if possible.
If the provided string cannot be parsed from JSON a SyntaxError
will be thrown.
If the provided object or parsed string is not a valid SQON, a ZodError
will be thrown.
SQONBuilder Object
The SQONBuilder
type is an object that stores a SQON
object and exposes functions to modify that SQON
, returning a new SQONBuilder
object. This allows a user to chain function calls together to create complex filters one step at a time.
All filter and combination methods listed above can be called from the returned SQONBuilder
object:
- In:
.in(fieldName: string, value: ArrayFilterValue) => SQONBuilder
- GreaterThan:
.gt(fieldName: string, value: number) => SQONBuilder
- LesserThan:
.lt(fieldName: string, value: number) => SQONBuilder
- And:
.and(sqon: SQONBuilder | SQON) => SQONBuilder
- Or:
.in(sqon: SQONBuilder | SQON) => SQONBuilder
- Not:
.in(sqon: SQONBuilder | SQON) => SQONBuilder
Example: const mySqon: SQON = SQONBuilder.in('name', ['Jim']).or(SQONBuilder.gt('age', 25).lt('score', 50)).toValue();
All builder methods are side-effect free: modifications made with to a SQONBuilder
are additive on top of a cloned SQON from the previous builder, so that the original builder is not modified. This can let you, for example, create a builder as a base SQON to make variations on in your application:
const base = SQONBuilder.in('name', ['Jim']);
const ageRestricted = base.gt('age', 25);
// {"op":"and","content":[{"op":"in","content":{"fieldName":"name","value":["Jim"]}},{"op":"gt","content":{"fieldName":"age","value":25}}]}
const scoreRestricted = base.lt('score', 50);
// {"op":"and","content":[{"op":"in","content":{"fieldName":"name","value":["Jim"]}},{"op":"lt","content":{"fieldName":"score","value":50}}]}
As filters and combinations are added to a SQONBuilder
, the SQON
maintained will be reduced, if possible. This will result in matching filters and combinations to be combined, and empty combinations to be removed.
In addition to the filter and combination functions, there are some additional methods that can modify the filters stored in the existing SQON, and then two methods for outputing the stored SQON
as either a string with .toString()
or a value with .toValue()
ToString
builder.toString() => string
Return stored SQON
value as string.
ToValue
builder.toValue() => SQON
Return stored SQON
value as an object.
Remove Exact Filter
.removeExactFilter(filter: FilterOperator) => SQONBuilder
Find exact matching filter and remove it from the SQON.
For filters with an array of values, the order of the array will be ignored during matching.
Note: This only looks for filters at the root of the sqon or in the content of the top level combination operator. This will not search recursively through the SQON.
const initial = SQONBuilder.in('name', 'Jim').gt('score', 50);
initial.removeFilter({op: FilterKeys.In, content: {fieldName: 'name', value: ['Jim']}});
// {op: 'gt', content: {fieldName: 'score', value: 50}}
Remove Filter
.(fieldName: string, op?: FilterKey, value?: FilterValue) => SQONBuilder
Find partial matching filters based on optional arguments and remove them from the SQON.
If only the fieldName is provided, all filters on that field will be removed.
If the fieldName and operator are provided, then a filter matching that fieldName and op will be removed (shouldn't ever be more than one in an operator due to the SQON reducer).
If values are provided, then a filter exactly matching all the arguments will be removed.
If a filter is found that matches the fieldName and op, then all matching values form the provided array will be removed from the filter. This lets you remove select values from an array filter without removing the entire filter.
Note: This only looks for filters at the root of the sqon or in the content of the top level combination operator. This will not search recursively through the SQON.
Example removing all filters for a property:
const builder = SQONBuilder.in('name', ['Jim', 'Bob']).gt('age', 20).lt('age', 50);
// {"op":"and","content":[{"op":"in","content":{"fieldName":"name","value":["Jim","Bob"]}},{"op":"gt","content":{"fieldName":"age","value":20}},{"op":"lt","content":{"fieldName":"age","value":50}}]}
// Remove all filters on 'age'
builder.removeFilter('age');
// {"op":"in","content":{"fieldName":"name","value":["Jim","Bob"]}}
Example removing some values from an array filter:
const builder = SQONBuilder.in('name', ['Jim', 'Bob', 'May']);
builder.removeFilter('name', FilterKeys.In, ['Jim', 'Bob', 'Sue']);
// {"op":"in","content":{"fieldName":"name","value":["May"]}}
Set Filter
.setFilter(fieldName: string, op: FilterKey, value: FilterValueMap[FilterKey]) => SQONBuilder
Add a specific filter to the content of the top level operator, or replace a matching filter (same op
and fieldName
) with the new value specified.
If the current SQON is just a filter, then the existing filter and this new filter will be combined with an and
operator.
Example:
const builder = SQONBuilder.gt('age', 50).in('name', ['Jim', 'Bob']);
builder.setFilter('name', FilterKeys.In, ['Jim', 'Sue']);
// {"op":"and","content":[{"op":"gt","content":{"fieldName":"age","value":50}},{"op":"in","content":{"fieldName":"name","value":["Jim","Sue"]}}]}
Reduce SQON
reduceSQON(sqon: SQON) => SQON
The reduceSQON
function is used internally by the SQON builder to reduce the complexity of a SQON by removing redundant operators and collecting similar filters. For example, if there is an and
combination wrapping a single filter, the and
operator can be removed and replaced by the single filter.
The reducer function is exported so you are able to run the reduction algorithm on SQONs independent of the SQONBuilder.
import { reduceSQON } from '@overture-stack/sqon-builder`;
const sqon: SQON = { op: 'and', 'content': [ { op: 'lt', content: { fieldName: 'age', value: 100 } } ] };
const reduced = reduceSQON(sqon);
// { op: 'lt', content: { fieldName: 'age', value: 100 } }
Check Matching Filter
checkMatchingFilter(a: FilterOperator, b: FilterOperator) => boolean
The checkMatchingFilter
function will compare two FilterOperators
and return is they match in all properties. This ensures they have the same filter operation (op
value), same fieldName
and matching value
. Note that for array values, the match is performed independent of order and after removing duplicates - it is a match on the logical content not on the exact array.
import { checkMatchingFilter } from '@overture-stack/sqon-builder';
const filterA = { op: FilterKeys.In, content: { fieldName: 'name', value: ['Jim', 'Bob', 'May'] } };
const filterB = { op: FilterKeys.In, content: { fieldName: 'name', value: ['May', 'Jim', 'Bob'] } };
const matchResult = checkMatchingFilter(filterA, filterB); // true
Types and SQON Validation
SQON types are exported from the library, and a Zod schema that will provide validation of the type is exported as a variable with a matching name.
The base type is SQON
.
import { SQON } form '@overture-stack/sqon-builder';
// TypeScript checked SQON types
const mySqon: SQON = { op: 'in', content: { fieldName: 'name', value: [ 'Jim', 'Bob' ] } };
// Parsing an unknown variable
const maybeSqon = {'might':'not be valid'};// invalid sqon
const validationResult = SQON.safeParse(maybeSqon);
if(validationResult.success) {
// Successfully parsed
const validSqon = validationResult.data;
} else {
// Validation errors, can be read out here
const errors = validationResult.error;
}
SQON Type Composition
A SQON is composed from a series of nested Operators
that take the general structure:
{
"op": "<operation code>",
"content": <Value or array of values>
}
There are two types of Operators:
FilterOperators
- Define a rule used to filter data based on matches to a specific field. For example, thein
filter will find all data that has a field value matching one of the included values in the filter.CombinationOperators
- Combine multiple operators with boolean logic.
FilterOperators
The content
property of a FilterOperator
is a single object of the form:
{
"fieldName": "<Field name to filter on>",
"value": <Value or array of values>,
}
There are two categories of filters that differ based on the types of values they accept as an argument:
ArrayFilter
- accept a number, a string, or an array of numbers or strings.ScalarFilter
- accept only a single number Types are exported that represent these filters, their expected value typesArrayFilterValue
/ScalarFilterValue
, as well as types that define the keys available for each filter type and all filters together:ArrayFilterKeys
,ScalarFilterKeys
andFilterKeys
.
CombinationOperators
The content
property of a CombinationOperator
is an array of other operators - these can be FilterOperators
or additional nested CombinationOperators
There are three supported combinations representing the common boolean operations and
and or
, plus not
. not
indicates that all contained operations must be false, so is equivalent to a list of negated filters combined with an and
.
All keys used in combinations are available in the CombinationKeys
object.
Note: The SQONBuilder will not create a cyclical combination loop. All SQONs passed into the SQON builder are cloned to get their immediate value and are stored by reference.
The Operator
type is a union of the FilterOperator | CombinationOperator
. This is equivalent to a SQON
type - SQON is just an alias for this type.
Convenient Type Guards
To help identify which type of Operator
the top level of a given SQON
is, this package exports some functions that act as type guards:
isCombination()
will identify if the input is aCombinationOperator
isFilter()
will identify if the input is aFilterOperator
isArrayFilter()
will further narrow a filter down to see if it can accept an Array of values, or only a single number.isScalarFilter()
will further narrow a filter down to see if requires a number as a value.isArrayFilterKey()
will identify if the given input is one of the ArrayFilterKeys.isScalarFilter()
will identify if the given input is one of the ScalarFilterKeys.
Example:
import { isCombination, isArrayFilter } from '@overture-stack/sqon-builder';
const sqon: SQON = getSQON();
if(isCombination(sqon)) {
// can interact with sqon knowing it is a combination operator
} else {
// can interact with sqon knowing it is a filter
// Lets narrow down the filter type further
if(isArrayFilter(sqon)) {
// we can assign an array of values to sqon.content.value
} else {
// sqon.content.value is a single number
}
}