Type filters

This commit is contained in:
Adria Navarro 2024-07-09 14:01:44 +02:00
parent 9c6347f7fd
commit 38f7b88735
3 changed files with 70 additions and 57 deletions

View File

@ -11,6 +11,7 @@ import {
import { SqlStatements } from "./sqlStatements" import { SqlStatements } from "./sqlStatements"
import SqlTableQueryBuilder from "./sqlTable" import SqlTableQueryBuilder from "./sqlTable"
import { import {
AnySearchFilter,
BBReferenceFieldMetadata, BBReferenceFieldMetadata,
FieldSchema, FieldSchema,
FieldType, FieldType,
@ -41,7 +42,7 @@ const envLimit = environment.SQL_MAX_ROWS
: null : null
const BASE_LIMIT = envLimit || 5000 const BASE_LIMIT = envLimit || 5000
function likeKey(client: string, key: string): string { function likeKey(client: string | string[], key: string): string {
let start: string, end: string let start: string, end: string
switch (client) { switch (client) {
case SqlClient.MY_SQL: case SqlClient.MY_SQL:
@ -207,18 +208,27 @@ class InternalBuilder {
return alias || name return alias || name
} }
function iterate( function iterate(
structure: { [key: string]: any }, structure: AnySearchFilter,
fn: (key: string, value: any) => void fn: (key: string, value: any) => void,
complexKeyFn?: (key: string[], value: any) => void
) { ) {
for (let [key, value] of Object.entries(structure)) { for (const key in structure) {
const value = structure[key]
const updatedKey = dbCore.removeKeyNumbering(key) const updatedKey = dbCore.removeKeyNumbering(key)
const isRelationshipField = updatedKey.includes(".") const isRelationshipField = updatedKey.includes(".")
if (updatedKey === InternalSearchFilterOperator.COMPLEX_ID_OPERATOR) { let castedTypeValue
if (
key === InternalSearchFilterOperator.COMPLEX_ID_OPERATOR &&
(castedTypeValue = structure[key]) &&
complexKeyFn
) {
const alias = getTableAlias(tableName) const alias = getTableAlias(tableName)
fn( complexKeyFn(
value.id.map((x: string) => (alias ? `${alias}.${x}` : x)), castedTypeValue.id.map((x: string) =>
value.values alias ? `${alias}.${x}` : x
),
castedTypeValue.values
) )
} else if (!opts.relationship && !isRelationshipField) { } else if (!opts.relationship && !isRelationshipField) {
const alias = getTableAlias(tableName) const alias = getTableAlias(tableName)
@ -246,7 +256,7 @@ class InternalBuilder {
} }
} }
const contains = (mode: object, any: boolean = false) => { const contains = (mode: AnySearchFilter, any: boolean = false) => {
const rawFnc = allOr ? "orWhereRaw" : "whereRaw" const rawFnc = allOr ? "orWhereRaw" : "whereRaw"
const not = mode === filters?.notContains ? "NOT " : "" const not = mode === filters?.notContains ? "NOT " : ""
function stringifyArray(value: Array<any>, quoteStyle = '"'): string { function stringifyArray(value: Array<any>, quoteStyle = '"'): string {
@ -258,7 +268,7 @@ class InternalBuilder {
return `[${value.join(",")}]` return `[${value.join(",")}]`
} }
if (this.client === SqlClient.POSTGRES) { if (this.client === SqlClient.POSTGRES) {
iterate(mode, (key: string, value: Array<any>) => { iterate(mode, (key, value) => {
const wrap = any ? "" : "'" const wrap = any ? "" : "'"
const op = any ? "\\?| array" : "@>" const op = any ? "\\?| array" : "@>"
const fieldNames = key.split(/\./g) const fieldNames = key.split(/\./g)
@ -273,7 +283,7 @@ class InternalBuilder {
}) })
} else if (this.client === SqlClient.MY_SQL) { } else if (this.client === SqlClient.MY_SQL) {
const jsonFnc = any ? "JSON_OVERLAPS" : "JSON_CONTAINS" const jsonFnc = any ? "JSON_OVERLAPS" : "JSON_CONTAINS"
iterate(mode, (key: string, value: Array<any>) => { iterate(mode, (key, value) => {
query = query[rawFnc]( query = query[rawFnc](
`${not}COALESCE(${jsonFnc}(${key}, '${stringifyArray( `${not}COALESCE(${jsonFnc}(${key}, '${stringifyArray(
value value
@ -282,7 +292,7 @@ class InternalBuilder {
}) })
} else { } else {
const andOr = mode === filters?.containsAny ? " OR " : " AND " const andOr = mode === filters?.containsAny ? " OR " : " AND "
iterate(mode, (key: string, value: Array<any>) => { iterate(mode, (key, value) => {
let statement = "" let statement = ""
for (let i in value) { for (let i in value) {
if (typeof value[i] === "string") { if (typeof value[i] === "string") {
@ -306,10 +316,16 @@ class InternalBuilder {
} }
if (filters.oneOf) { if (filters.oneOf) {
iterate(filters.oneOf, (key, array) => {
const fnc = allOr ? "orWhereIn" : "whereIn" const fnc = allOr ? "orWhereIn" : "whereIn"
iterate(
filters.oneOf,
(key: string, array) => {
query = query[fnc](key, Array.isArray(array) ? array : [array]) query = query[fnc](key, Array.isArray(array) ? array : [array])
}) },
(key: string[], array) => {
query = query[fnc](key, Array.isArray(array) ? array : [array])
}
)
} }
if (filters.string) { if (filters.string) {
iterate(filters.string, (key, value) => { iterate(filters.string, (key, value) => {

View File

@ -190,8 +190,8 @@ export class ExternalRequest<T extends Operation> {
if (filters) { if (filters) {
// need to map over the filters and make sure the _id field isn't present // need to map over the filters and make sure the _id field isn't present
let prefix = 1 let prefix = 1
for (let operator of Object.values(filters)) { for (const operator of Object.values(filters)) {
for (let field of Object.keys(operator || {})) { for (const field of Object.keys(operator || {})) {
if (dbCore.removeKeyNumbering(field) === "_id") { if (dbCore.removeKeyNumbering(field) === "_id") {
if (primary) { if (primary) {
const parts = breakRowIdField(operator[field]) const parts = breakRowIdField(operator[field])

View File

@ -21,51 +21,48 @@ export enum InternalSearchFilterOperator {
COMPLEX_ID_OPERATOR = "_complexIdOperator", COMPLEX_ID_OPERATOR = "_complexIdOperator",
} }
export interface SearchFilters { type BasicFilter = Record<string, string> & {
allOr?: boolean [InternalSearchFilterOperator.COMPLEX_ID_OPERATOR]?: never
// TODO: this is just around for now - we need a better way to do or/and
// allows just fuzzy to be or - all the fuzzy/like parameters
fuzzyOr?: boolean
onEmptyFilter?: EmptyFilterOption
[SearchFilterOperator.STRING]?: {
[key: string]: string
} }
[SearchFilterOperator.FUZZY]?: {
[key: string]: string type ArrayFilter = Record<string, string[]> & {
[InternalSearchFilterOperator.COMPLEX_ID_OPERATOR]?: {
id: string[]
values: string[]
} }
[SearchFilterOperator.RANGE]?: { }
[key: string]:
type RangeFilter = Record<
string,
| { | {
high: number | string high: number | string
low: number | string low: number | string
} }
| { high: number | string } | { high: number | string }
| { low: number | string } | { low: number | string }
> & {
[InternalSearchFilterOperator.COMPLEX_ID_OPERATOR]?: never
} }
[SearchFilterOperator.EQUAL]?: {
[key: string]: any export type AnySearchFilter = BasicFilter | ArrayFilter | RangeFilter
}
[SearchFilterOperator.NOT_EQUAL]?: { export interface SearchFilters {
[key: string]: any allOr?: boolean
} // TODO: this is just around for now - we need a better way to do or/and
[SearchFilterOperator.EMPTY]?: { // allows just fuzzy to be or - all the fuzzy/like parameters
[key: string]: any fuzzyOr?: boolean
} onEmptyFilter?: EmptyFilterOption
[SearchFilterOperator.NOT_EMPTY]?: { [SearchFilterOperator.STRING]?: BasicFilter
[key: string]: any [SearchFilterOperator.FUZZY]?: BasicFilter
} [SearchFilterOperator.RANGE]?: RangeFilter
[SearchFilterOperator.ONE_OF]?: { [SearchFilterOperator.EQUAL]?: BasicFilter
[key: string]: any[] [SearchFilterOperator.NOT_EQUAL]?: BasicFilter
} [SearchFilterOperator.EMPTY]?: BasicFilter
[SearchFilterOperator.CONTAINS]?: { [SearchFilterOperator.NOT_EMPTY]?: BasicFilter
[key: string]: any[] [SearchFilterOperator.ONE_OF]?: ArrayFilter
} [SearchFilterOperator.CONTAINS]?: ArrayFilter
[SearchFilterOperator.NOT_CONTAINS]?: { [SearchFilterOperator.NOT_CONTAINS]?: ArrayFilter
[key: string]: any[] [SearchFilterOperator.CONTAINS_ANY]?: ArrayFilter
}
[SearchFilterOperator.CONTAINS_ANY]?: {
[key: string]: any[]
}
// specific to SQS/SQLite search on internal tables this can be used // specific to SQS/SQLite search on internal tables this can be used
// to make sure the documents returned are always filtered down to a // to make sure the documents returned are always filtered down to a
// specific document type (such as just rows) // specific document type (such as just rows)