Restructure search.spec.ts to be much more readable.

This commit is contained in:
Sam Rose 2024-04-11 15:16:26 +01:00
parent eb56140ce2
commit c07882b452
No known key found for this signature in database
1 changed files with 200 additions and 206 deletions

View File

@ -7,15 +7,11 @@ import {
EmptyFilterOption, EmptyFilterOption,
FieldType, FieldType,
RowSearchParams, RowSearchParams,
SearchFilters,
Table, Table,
TableSchema,
} from "@budibase/types" } from "@budibase/types"
import _ from "lodash"
function leftContainsRight<
A extends Record<string, any>,
B extends Record<string, any>
>(left: A, right: B) {
return Object.entries(right).every(([k, v]) => left[k] === v)
}
jest.unmock("mssql") jest.unmock("mssql")
@ -32,6 +28,7 @@ describe.each([
let envCleanup: (() => void) | undefined let envCleanup: (() => void) | undefined
let datasource: Datasource | undefined let datasource: Datasource | undefined
let table: Table
beforeAll(async () => { beforeAll(async () => {
if (isSqs) { if (isSqs) {
@ -52,220 +49,217 @@ describe.each([
} }
}) })
async function testSearch<RowType extends Record<string, any>>( async function createTable(schema: TableSchema) {
test: SearchTest<RowType>, table = await config.api.table.save(
table: Table tableForDatasource(datasource, { schema })
) {
const expected = test.expectToFind
delete test.expectToFind
const { rows: foundRows } = await config.api.row.search(table._id!, {
...test,
tableId: table._id!,
})
if (!expected) {
return
}
expect(foundRows).toHaveLength(expected.length)
expect(foundRows).toEqual(
expect.arrayContaining(
expected.map(expectedRow =>
expect.objectContaining(
foundRows.find(foundRow => leftContainsRight(foundRow, expectedRow))
)
)
)
) )
} }
function searchTests<T extends Record<string, any>>( async function createRows(rows: Record<string, any>[]) {
name: string, await Promise.all(rows.map(r => config.api.row.save(table._id!, r)))
opts: { }
table: (ds?: Datasource) => Promise<Table>
rows: T[]
tests: SearchTest<T>[]
}
) {
let table: Table
for (const test of opts.tests) { class SearchAssertion {
test.toString = () => { constructor(private readonly query: RowSearchParams) {}
const queryStr = JSON.stringify({
query: test.query,
limit: test.limit,
sort: test.sort,
sortOrder: test.sortOrder,
})
const expectStr = JSON.stringify(test.expectToFind)
return `should run: ${queryStr} and find ${expectStr}`
}
}
// eslint-disable-next-line jest/valid-title async toFind(expectedRows: any[]) {
describe(name, () => { const { rows: foundRows } = await config.api.row.search(table._id!, {
beforeAll(async () => { ...this.query,
table = await opts.table(datasource) tableId: table._id!,
}) })
beforeAll(async () => { // eslint-disable-next-line jest/no-standalone-expect
await Promise.all( expect(foundRows).toHaveLength(expectedRows.length)
opts.rows.map(r => config.api.row.save(table._id!, r)) // eslint-disable-next-line jest/no-standalone-expect
expect(foundRows).toEqual(
expect.arrayContaining(
expectedRows.map((expectedRow: any) =>
expect.objectContaining(
foundRows.find(foundRow => _.isMatch(foundRow, expectedRow))
)
)
) )
)
}
async toFindNothing() {
await this.toFind([])
}
}
function expectSearch(query: Omit<RowSearchParams, "tableId">) {
return new SearchAssertion({ ...query, tableId: table._id! })
}
function expectQuery(query: SearchFilters) {
return expectSearch({ query })
}
describe("strings", () => {
beforeAll(async () => {
await createTable({
name: { name: "name", type: FieldType.STRING },
})
await createRows([{ name: "foo" }, { name: "bar" }])
})
describe("misc", () => {
it("should return all if no query is passed", () =>
expectSearch({} as RowSearchParams).toFind([
{ name: "foo" },
{ name: "bar" },
]))
it("should return all if empty query is passed", () =>
expectQuery({}).toFind([{ name: "foo" }, { name: "bar" }]))
it("should return all if onEmptyFilter is RETURN_ALL", () =>
expectQuery({
onEmptyFilter: EmptyFilterOption.RETURN_ALL,
}).toFind([{ name: "foo" }, { name: "bar" }]))
it("should return nothing if onEmptyFilter is RETURN_NONE", () =>
expectQuery({
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
}).toFindNothing())
})
describe("equal", () => {
it("successfully finds a row", () =>
expectQuery({ equal: { name: "foo" } }).toFind([{ name: "foo" }]))
it("fails to find nonexistent row", () =>
expectQuery({ equal: { name: "none" } }).toFindNothing())
})
describe("notEqual", () => {
it("successfully finds a row", () =>
expectQuery({ notEqual: { name: "foo" } }).toFind([{ name: "bar" }]))
it("fails to find nonexistent row", () =>
expectQuery({ notEqual: { name: "bar" } }).toFind([{ name: "foo" }]))
})
describe("oneOf", () => {
it("successfully finds a row", () =>
expectQuery({ oneOf: { name: ["foo"] } }).toFind([{ name: "foo" }]))
it("fails to find nonexistent row", () =>
expectQuery({ oneOf: { name: ["none"] } }).toFindNothing())
})
describe("fuzzy", () => {
it("successfully finds a row", () =>
expectQuery({ fuzzy: { name: "oo" } }).toFind([{ name: "foo" }]))
it("fails to find nonexistent row", () =>
expectQuery({ fuzzy: { name: "none" } }).toFindNothing())
})
})
describe("numbers", () => {
beforeAll(async () => {
await createTable({
age: { name: "age", type: FieldType.NUMBER },
})
await createRows([{ age: 1 }, { age: 10 }])
})
describe("equal", () => {
it("successfully finds a row", () =>
expectQuery({ equal: { age: 1 } }).toFind([{ age: 1 }]))
it("fails to find nonexistent row", () =>
expectQuery({ equal: { age: 2 } }).toFindNothing())
})
describe("notEqual", () => {
it("successfully finds a row", () =>
expectQuery({ notEqual: { age: 1 } }).toFind([{ age: 10 }]))
it("fails to find nonexistent row", () =>
expectQuery({ notEqual: { age: 10 } }).toFind([{ age: 1 }]))
})
describe("oneOf", () => {
it("successfully finds a row", () =>
expectQuery({ oneOf: { age: [1] } }).toFind([{ age: 1 }]))
it("fails to find nonexistent row", () =>
expectQuery({ oneOf: { age: [2] } }).toFindNothing())
})
describe("range", () => {
it("successfully finds a row", () =>
expectQuery({
range: { age: { low: 1, high: 5 } },
}).toFind([{ age: 1 }]))
it("successfully finds multiple rows", () =>
expectQuery({
range: { age: { low: 1, high: 10 } },
}).toFind([{ age: 1 }, { age: 10 }]))
it("successfully finds a row with a high bound", () =>
expectQuery({
range: { age: { low: 5, high: 10 } },
}).toFind([{ age: 10 }]))
})
})
describe("dates", () => {
const JAN_1ST = "2020-01-01T00:00:00.000Z"
const JAN_2ND = "2020-01-02T00:00:00.000Z"
const JAN_5TH = "2020-01-05T00:00:00.000Z"
const JAN_10TH = "2020-01-10T00:00:00.000Z"
beforeAll(async () => {
await createTable({
dob: { name: "dob", type: FieldType.DATETIME },
}) })
it.each(opts.tests)(`%s`, test => testSearch(test, table)) await createRows([{ dob: JAN_1ST }, { dob: JAN_10TH }])
}) })
}
interface SearchTest<RowType extends Record<string, any>> describe("equal", () => {
extends Omit<RowSearchParams, "tableId"> { it("successfully finds a row", () =>
expectToFind?: RowType[] expectQuery({ equal: { dob: JAN_1ST } }).toFind([{ dob: JAN_1ST }]))
}
searchTests("strings", { it("fails to find nonexistent row", () =>
table: async ds => { expectQuery({ equal: { dob: JAN_2ND } }).toFindNothing())
return await config.api.table.save( })
tableForDatasource(ds, {
schema: {
name: {
name: "name",
type: FieldType.STRING,
},
},
})
)
},
rows: [{ name: "foo" }, { name: "bar" }],
tests: [
// These test cases are generic and don't really need to be repeated for
// all data types, so we just do them here.
// @ts-expect-error - intentionally not passing a query to make sure the describe("notEqual", () => {
// API can handle it. it("successfully finds a row", () =>
{ expectToFind: [{ name: "foo" }, { name: "bar" }] }, expectQuery({ notEqual: { dob: JAN_1ST } }).toFind([{ dob: JAN_10TH }]))
{ query: {}, expectToFind: [{ name: "foo" }, { name: "bar" }] },
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_ALL },
expectToFind: [{ name: "foo" }, { name: "bar" }],
},
{
query: { onEmptyFilter: EmptyFilterOption.RETURN_NONE },
expectToFind: [],
},
// The rest of these tests are specific to strings.
{ query: { string: { name: "foo" } }, expectToFind: [{ name: "foo" }] },
{ query: { string: { name: "none" } }, expectToFind: [] },
{ query: { fuzzy: { name: "oo" } }, expectToFind: [{ name: "foo" }] },
{ query: { equal: { name: "foo" } }, expectToFind: [{ name: "foo" }] },
{ query: { notEqual: { name: "foo" } }, expectToFind: [{ name: "bar" }] },
{ query: { oneOf: { name: ["foo"] } }, expectToFind: [{ name: "foo" }] },
],
})
searchTests("numbers", { it("fails to find nonexistent row", () =>
table: async ds => { expectQuery({ notEqual: { dob: JAN_10TH } }).toFind([{ dob: JAN_1ST }]))
return await config.api.table.save( })
tableForDatasource(ds, {
schema: {
age: {
name: "age",
type: FieldType.NUMBER,
},
},
})
)
},
rows: [{ age: 1 }, { age: 10 }],
tests: [
{ query: { equal: { age: 1 } }, expectToFind: [{ age: 1 }] },
{ query: { equal: { age: 2 } }, expectToFind: [] },
{ query: { notEqual: { age: 1 } }, expectToFind: [{ age: 10 }] },
{ query: { oneOf: { age: [1] } }, expectToFind: [{ age: 1 }] },
{
query: { range: { age: { low: 1, high: 5 } } },
expectToFind: [{ age: 1 }],
},
{
query: { range: { age: { low: 0, high: 1 } } },
expectToFind: [{ age: 1 }],
},
{ query: { range: { age: { low: 3, high: 4 } } }, expectToFind: [] },
{
query: { range: { age: { low: 0, high: 11 } } },
expectToFind: [{ age: 1 }, { age: 10 }],
},
],
})
searchTests("dates", { describe("oneOf", () => {
table: async ds => { it("successfully finds a row", () =>
return await config.api.table.save( expectQuery({ oneOf: { dob: [JAN_1ST] } }).toFind([{ dob: JAN_1ST }]))
tableForDatasource(ds, {
schema: { it("fails to find nonexistent row", () =>
dob: { expectQuery({ oneOf: { dob: [JAN_2ND] } }).toFindNothing())
name: "dob", })
type: FieldType.DATETIME,
}, describe("range", () => {
}, it("successfully finds a row", () =>
}) expectQuery({
) range: { dob: { low: JAN_1ST, high: JAN_5TH } },
}, }).toFind([{ dob: JAN_1ST }]))
rows: [
{ dob: "2020-01-01T00:00:00.000Z" }, it("successfully finds multiple rows", () =>
{ dob: "2020-01-10T00:00:00.000Z" }, expectQuery({
], range: { dob: { low: JAN_1ST, high: JAN_10TH } },
tests: [ }).toFind([{ dob: JAN_1ST }, { dob: JAN_10TH }]))
{
query: { equal: { dob: "2020-01-01T00:00:00.000Z" } }, it("successfully finds a row with a high bound", () =>
expectToFind: [{ dob: "2020-01-01T00:00:00.000Z" }], expectQuery({
}, range: { dob: { low: JAN_5TH, high: JAN_10TH } },
{ }).toFind([{ dob: JAN_10TH }]))
query: { equal: { dob: "2020-01-02T00:00:00.000Z" } }, })
expectToFind: [],
},
{
query: { notEqual: { dob: "2020-01-01T00:00:00.000Z" } },
expectToFind: [{ dob: "2020-01-10T00:00:00.000Z" }],
},
{
query: { oneOf: { dob: ["2020-01-01T00:00:00.000Z"] } },
expectToFind: [{ dob: "2020-01-01T00:00:00.000Z" }],
},
{
query: {
range: {
dob: {
low: "2020-01-01T00:00:00.000Z",
high: "2020-01-05T00:00:00.000Z",
},
},
},
expectToFind: [{ dob: "2020-01-01T00:00:00.000Z" }],
},
{
query: {
range: {
dob: {
low: "2020-01-01T00:00:00.000Z",
high: "2020-01-10T00:00:00.000Z",
},
},
},
expectToFind: [
{ dob: "2020-01-01T00:00:00.000Z" },
{ dob: "2020-01-10T00:00:00.000Z" },
],
},
{
query: {
range: {
dob: {
low: "2020-01-05T00:00:00.000Z",
high: "2020-01-10T00:00:00.000Z",
},
},
},
expectToFind: [{ dob: "2020-01-10T00:00:00.000Z" }],
},
],
}) })
}) })