Assert length on search tests, fixes bug in SQS around on empty return none.

This commit is contained in:
Sam Rose 2024-04-11 09:53:54 +01:00
parent e6c3fd2951
commit 229bbc0d10
No known key found for this signature in database
2 changed files with 56 additions and 29 deletions

View File

@ -6,6 +6,7 @@ import {
Datasource, Datasource,
EmptyFilterOption, EmptyFilterOption,
FieldType, FieldType,
Row,
SearchFilters, SearchFilters,
Table, Table,
} from "@budibase/types" } from "@budibase/types"
@ -47,7 +48,7 @@ describe.each([
}) })
describe("strings", () => { describe("strings", () => {
beforeEach(async () => { beforeAll(async () => {
table = await config.api.table.save( table = await config.api.table.save(
tableForDatasource(datasource, { tableForDatasource(datasource, {
schema: { schema: {
@ -61,6 +62,13 @@ describe.each([
}) })
const rows = [{ name: "foo" }, { name: "bar" }] const rows = [{ name: "foo" }, { name: "bar" }]
let savedRows: Row[]
beforeAll(async () => {
savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
})
interface StringSearchTest { interface StringSearchTest {
query: SearchFilters query: SearchFilters
@ -68,6 +76,8 @@ describe.each([
} }
const stringSearchTests: StringSearchTest[] = [ const stringSearchTests: StringSearchTest[] = [
// These three test cases are generic and don't really need
// to be repeated for all data types, so we just do them here.
{ query: {}, expected: rows }, { query: {}, expected: rows },
{ {
query: { onEmptyFilter: EmptyFilterOption.RETURN_ALL }, query: { onEmptyFilter: EmptyFilterOption.RETURN_ALL },
@ -77,6 +87,7 @@ describe.each([
query: { onEmptyFilter: EmptyFilterOption.RETURN_NONE }, query: { onEmptyFilter: EmptyFilterOption.RETURN_NONE },
expected: [], expected: [],
}, },
// The rest of these tests are specific to strings.
{ query: { string: { name: "foo" } }, expected: [rows[0]] }, { query: { string: { name: "foo" } }, expected: [rows[0]] },
{ query: { string: { name: "none" } }, expected: [] }, { query: { string: { name: "none" } }, expected: [] },
{ query: { fuzzy: { name: "oo" } }, expected: [rows[0]] }, { query: { fuzzy: { name: "oo" } }, expected: [rows[0]] },
@ -88,13 +99,11 @@ describe.each([
it.each(stringSearchTests)( it.each(stringSearchTests)(
`should be able to run query: $query`, `should be able to run query: $query`,
async ({ query, expected }) => { async ({ query, expected }) => {
const savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
const { rows: foundRows } = await config.api.row.search(table._id!, { const { rows: foundRows } = await config.api.row.search(table._id!, {
tableId: table._id!, tableId: table._id!,
query, query,
}) })
expect(foundRows).toHaveLength(expected.length)
expect(foundRows).toEqual( expect(foundRows).toEqual(
expect.arrayContaining( expect.arrayContaining(
expected.map(r => expected.map(r =>
@ -107,7 +116,7 @@ describe.each([
}) })
describe("number", () => { describe("number", () => {
beforeEach(async () => { beforeAll(async () => {
table = await config.api.table.save( table = await config.api.table.save(
tableForDatasource(datasource, { tableForDatasource(datasource, {
schema: { schema: {
@ -121,6 +130,13 @@ describe.each([
}) })
const rows = [{ age: 1 }, { age: 10 }] const rows = [{ age: 1 }, { age: 10 }]
let savedRows: Row[]
beforeAll(async () => {
savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
})
interface NumberSearchTest { interface NumberSearchTest {
query: SearchFilters query: SearchFilters
@ -128,15 +144,6 @@ describe.each([
} }
const numberSearchTests: NumberSearchTest[] = [ const numberSearchTests: NumberSearchTest[] = [
{ query: {}, expected: rows },
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_ALL },
expected: rows,
},
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_NONE },
expected: [],
},
{ query: { equal: { age: 1 } }, expected: [rows[0]] }, { query: { equal: { age: 1 } }, expected: [rows[0]] },
{ query: { equal: { age: 2 } }, expected: [] }, { query: { equal: { age: 2 } }, expected: [] },
{ query: { notEqual: { age: 1 } }, expected: [rows[1]] }, { query: { notEqual: { age: 1 } }, expected: [rows[1]] },
@ -150,13 +157,11 @@ describe.each([
it.each(numberSearchTests)( it.each(numberSearchTests)(
`should be able to run query: $query`, `should be able to run query: $query`,
async ({ query, expected }) => { async ({ query, expected }) => {
const savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
const { rows: foundRows } = await config.api.row.search(table._id!, { const { rows: foundRows } = await config.api.row.search(table._id!, {
tableId: table._id!, tableId: table._id!,
query, query,
}) })
expect(foundRows).toHaveLength(expected.length)
expect(foundRows).toEqual( expect(foundRows).toEqual(
expect.arrayContaining( expect.arrayContaining(
expected.map(r => expected.map(r =>
@ -186,6 +191,13 @@ describe.each([
{ dob: new Date("2020-01-01").toISOString() }, { dob: new Date("2020-01-01").toISOString() },
{ dob: new Date("2020-01-10").toISOString() }, { dob: new Date("2020-01-10").toISOString() },
] ]
let savedRows: Row[]
beforeEach(async () => {
savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
})
interface DateSearchTest { interface DateSearchTest {
query: SearchFilters query: SearchFilters
@ -193,15 +205,6 @@ describe.each([
} }
const dateSearchTests: DateSearchTest[] = [ const dateSearchTests: DateSearchTest[] = [
{ query: {}, expected: rows },
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_ALL },
expected: rows,
},
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_NONE },
expected: [],
},
{ {
query: { equal: { dob: new Date("2020-01-01").toISOString() } }, query: { equal: { dob: new Date("2020-01-01").toISOString() } },
expected: [rows[0]], expected: [rows[0]],
@ -256,13 +259,11 @@ describe.each([
it.each(dateSearchTests)( it.each(dateSearchTests)(
`should be able to run query: $query`, `should be able to run query: $query`,
async ({ query, expected }) => { async ({ query, expected }) => {
const savedRows = await Promise.all(
rows.map(r => config.api.row.save(table._id!, r))
)
const { rows: foundRows } = await config.api.row.search(table._id!, { const { rows: foundRows } = await config.api.row.search(table._id!, {
tableId: table._id!, tableId: table._id!,
query, query,
}) })
expect(foundRows).toHaveLength(expected.length)
expect(foundRows).toEqual( expect(foundRows).toEqual(
expect.arrayContaining( expect.arrayContaining(
expected.map(r => expected.map(r =>

View File

@ -22,6 +22,7 @@ import {
SortDirection, SortDirection,
SqlQueryBinding, SqlQueryBinding,
Table, Table,
EmptyFilterOption,
} from "@budibase/types" } from "@budibase/types"
import environment from "../../environment" import environment from "../../environment"
@ -243,6 +244,7 @@ class InternalBuilder {
return query return query
} }
filters = parseFilters(filters) filters = parseFilters(filters)
let noFilters = true
// if all or specified in filters, then everything is an or // if all or specified in filters, then everything is an or
const allOr = filters.allOr const allOr = filters.allOr
if (filters.oneOf) { if (filters.oneOf) {
@ -250,6 +252,7 @@ class InternalBuilder {
const fnc = allOr ? "orWhereIn" : "whereIn" const fnc = allOr ? "orWhereIn" : "whereIn"
query = query[fnc](key, Array.isArray(array) ? array : [array]) query = query[fnc](key, Array.isArray(array) ? array : [array])
}) })
noFilters = false
} }
if (filters.string) { if (filters.string) {
iterate(filters.string, (key, value) => { iterate(filters.string, (key, value) => {
@ -265,9 +268,11 @@ class InternalBuilder {
]) ])
} }
}) })
noFilters = false
} }
if (filters.fuzzy) { if (filters.fuzzy) {
iterate(filters.fuzzy, like) iterate(filters.fuzzy, like)
noFilters = false
} }
if (filters.range) { if (filters.range) {
iterate(filters.range, (key, value) => { iterate(filters.range, (key, value) => {
@ -300,40 +305,61 @@ class InternalBuilder {
query = query[fnc](key, "<", value.high) query = query[fnc](key, "<", value.high)
} }
}) })
noFilters = false
} }
if (filters.equal) { if (filters.equal) {
iterate(filters.equal, (key, value) => { iterate(filters.equal, (key, value) => {
const fnc = allOr ? "orWhere" : "where" const fnc = allOr ? "orWhere" : "where"
query = query[fnc]({ [key]: value }) query = query[fnc]({ [key]: value })
}) })
// Somewhere above us in the stack adds `{ type: "row" }` to the `equal`
// key before we get here, so we need to still consider it empty when
// that's the case.
const equalEmpty =
Object.keys(filters.equal).length === 1 && filters.equal.type === "row"
if (!equalEmpty) {
noFilters = false
}
} }
if (filters.notEqual) { if (filters.notEqual) {
iterate(filters.notEqual, (key, value) => { iterate(filters.notEqual, (key, value) => {
const fnc = allOr ? "orWhereNot" : "whereNot" const fnc = allOr ? "orWhereNot" : "whereNot"
query = query[fnc]({ [key]: value }) query = query[fnc]({ [key]: value })
}) })
noFilters = false
} }
if (filters.empty) { if (filters.empty) {
iterate(filters.empty, key => { iterate(filters.empty, key => {
const fnc = allOr ? "orWhereNull" : "whereNull" const fnc = allOr ? "orWhereNull" : "whereNull"
query = query[fnc](key) query = query[fnc](key)
}) })
noFilters = false
} }
if (filters.notEmpty) { if (filters.notEmpty) {
iterate(filters.notEmpty, key => { iterate(filters.notEmpty, key => {
const fnc = allOr ? "orWhereNotNull" : "whereNotNull" const fnc = allOr ? "orWhereNotNull" : "whereNotNull"
query = query[fnc](key) query = query[fnc](key)
}) })
noFilters = false
} }
if (filters.contains) { if (filters.contains) {
contains(filters.contains) contains(filters.contains)
noFilters = false
} }
if (filters.notContains) { if (filters.notContains) {
contains(filters.notContains) contains(filters.notContains)
noFilters = false
} }
if (filters.containsAny) { if (filters.containsAny) {
contains(filters.containsAny, true) contains(filters.containsAny, true)
noFilters = false
} }
if (noFilters && filters.onEmptyFilter === EmptyFilterOption.RETURN_NONE) {
query = query.whereRaw("1=0")
}
return query return query
} }