Merge remote-tracking branch 'refs/remotes/origin/feat/automation-step-naming-updates' into feat/automation-step-naming-updates
This commit is contained in:
commit
2f881dd16f
Binary file not shown.
Binary file not shown.
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"$schema": "node_modules/lerna/schemas/lerna-schema.json",
|
||||
"version": "2.31.8",
|
||||
"version": "2.32.0",
|
||||
"npmClient": "yarn",
|
||||
"packages": [
|
||||
"packages/*",
|
||||
|
|
|
@ -1 +1 @@
|
|||
Subproject commit 048c37ecd921340614bf61a76a774aaa46176569
|
||||
Subproject commit 7899d07904d89d48954dd500da7b5dec32b781dd
|
|
@ -3,6 +3,7 @@ export * as processors from "./processors"
|
|||
export * as analytics from "./analytics"
|
||||
export { default as identification } from "./identification"
|
||||
export * as backfillCache from "./backfill"
|
||||
export { publishEvent } from "./events"
|
||||
|
||||
import { processors } from "./processors"
|
||||
|
||||
|
|
|
@ -2,6 +2,7 @@ import { it, expect, describe, vi } from "vitest"
|
|||
import AISettings from "./index.svelte"
|
||||
import { render } from "@testing-library/svelte"
|
||||
import { admin, licensing } from "stores/portal"
|
||||
import { notifications } from "@budibase/bbui"
|
||||
|
||||
vi.spyOn(notifications, "error").mockImplementation(vi.fn)
|
||||
vi.spyOn(notifications, "success").mockImplementation(vi.fn)
|
||||
|
|
|
@ -2907,6 +2907,28 @@ describe.each([
|
|||
'Invalid body - "query.$and.conditions[1].$and.conditions" is required'
|
||||
)
|
||||
})
|
||||
|
||||
it("returns no rows when onEmptyFilter set to none", async () => {
|
||||
await expectSearch({
|
||||
query: {
|
||||
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
|
||||
$and: {
|
||||
conditions: [{ equal: { name: "" } }],
|
||||
},
|
||||
},
|
||||
}).toFindNothing()
|
||||
})
|
||||
|
||||
it("returns all rows when onEmptyFilter set to all", async () => {
|
||||
await expectSearch({
|
||||
query: {
|
||||
onEmptyFilter: EmptyFilterOption.RETURN_ALL,
|
||||
$and: {
|
||||
conditions: [{ equal: { name: "" } }],
|
||||
},
|
||||
},
|
||||
}).toHaveLength(4)
|
||||
})
|
||||
})
|
||||
|
||||
!isLucene &&
|
||||
|
@ -3035,5 +3057,27 @@ describe.each([
|
|||
},
|
||||
}).toContainExactly([{ age: 1, name: "Jane" }])
|
||||
})
|
||||
|
||||
it("returns no rows when onEmptyFilter set to none", async () => {
|
||||
await expectSearch({
|
||||
query: {
|
||||
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
|
||||
$or: {
|
||||
conditions: [{ equal: { name: "" } }],
|
||||
},
|
||||
},
|
||||
}).toFindNothing()
|
||||
})
|
||||
|
||||
it("returns all rows when onEmptyFilter set to all", async () => {
|
||||
await expectSearch({
|
||||
query: {
|
||||
onEmptyFilter: EmptyFilterOption.RETURN_ALL,
|
||||
$or: {
|
||||
conditions: [{ equal: { name: "" } }],
|
||||
},
|
||||
},
|
||||
}).toHaveLength(4)
|
||||
})
|
||||
})
|
||||
})
|
||||
|
|
|
@ -27,6 +27,12 @@ import { isPlainObject, isEmpty } from "lodash"
|
|||
import { decodeNonAscii } from "./helpers/schema"
|
||||
|
||||
const HBS_REGEX = /{{([^{].*?)}}/g
|
||||
const LOGICAL_OPERATORS = Object.values(LogicalOperator)
|
||||
const SEARCH_OPERATORS = [
|
||||
...Object.values(BasicOperator),
|
||||
...Object.values(ArrayOperator),
|
||||
...Object.values(RangeOperator),
|
||||
]
|
||||
|
||||
/**
|
||||
* Returns the valid operator options for a certain data type
|
||||
|
@ -117,7 +123,7 @@ export function recurseLogicalOperators(
|
|||
filters: SearchFilters,
|
||||
fn: (f: SearchFilters) => SearchFilters
|
||||
) {
|
||||
for (const logical of Object.values(LogicalOperator)) {
|
||||
for (const logical of LOGICAL_OPERATORS) {
|
||||
if (filters[logical]) {
|
||||
filters[logical]!.conditions = filters[logical]!.conditions.map(
|
||||
condition => fn(condition)
|
||||
|
@ -135,7 +141,7 @@ export function recurseSearchFilters(
|
|||
filters = processFn(filters)
|
||||
|
||||
// Recurse through logical operators
|
||||
for (const logical of Object.values(LogicalOperator)) {
|
||||
for (const logical of LOGICAL_OPERATORS) {
|
||||
if (filters[logical]) {
|
||||
filters[logical]!.conditions = filters[logical]!.conditions.map(
|
||||
condition => recurseSearchFilters(condition, processFn)
|
||||
|
@ -773,12 +779,16 @@ export function runQuery<T extends Record<string, any>>(
|
|||
return filterFunctions[key as SearchFilterOperator]?.(doc) ?? false
|
||||
})
|
||||
|
||||
if (query.allOr) {
|
||||
// there are no filters - logical operators can cover this up
|
||||
if (!hasFilters(query)) {
|
||||
return true
|
||||
} else if (query.allOr) {
|
||||
return results.some(result => result === true)
|
||||
} else {
|
||||
return results.every(result => result === true)
|
||||
}
|
||||
}
|
||||
|
||||
return docs.filter(docMatch)
|
||||
}
|
||||
|
||||
|
@ -841,14 +851,33 @@ export const hasFilters = (query?: SearchFilters) => {
|
|||
if (!query) {
|
||||
return false
|
||||
}
|
||||
const skipped = ["allOr", "onEmptyFilter"]
|
||||
for (let [key, value] of Object.entries(query)) {
|
||||
if (skipped.includes(key) || typeof value !== "object") {
|
||||
const check = (filters: SearchFilters): boolean => {
|
||||
for (const logical of LOGICAL_OPERATORS) {
|
||||
if (filters[logical]) {
|
||||
for (const condition of filters[logical]?.conditions || []) {
|
||||
const result = check(condition)
|
||||
if (result) {
|
||||
return result
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
for (const search of SEARCH_OPERATORS) {
|
||||
const searchValue = filters[search]
|
||||
if (!searchValue || typeof searchValue !== "object") {
|
||||
continue
|
||||
}
|
||||
if (Object.keys(value || {}).length !== 0) {
|
||||
const filtered = Object.entries(searchValue).filter(entry => {
|
||||
const valueDefined =
|
||||
entry[1] !== undefined || entry[1] !== null || entry[1] !== ""
|
||||
// not empty is an edge case, null is allowed for it - this is covered by test cases
|
||||
return search === BasicOperator.NOT_EMPTY || valueDefined
|
||||
})
|
||||
if (filtered.length !== 0) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
return check(query)
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue