Fix generic-sql.spec.ts

This commit is contained in:
Sam Rose 2024-08-05 12:05:42 +01:00
parent 165e368a24
commit eecd521a8a
No known key found for this signature in database
1 changed files with 153 additions and 147 deletions

View File

@ -22,9 +22,13 @@ describe.each(
DatabaseName.MYSQL, DatabaseName.MYSQL,
DatabaseName.SQL_SERVER, DatabaseName.SQL_SERVER,
DatabaseName.MARIADB, DatabaseName.MARIADB,
DatabaseName.ORACLE,
].map(name => [name, getDatasource(name)]) ].map(name => [name, getDatasource(name)])
)("queries (%s)", (dbName, dsProvider) => { )("queries (%s)", (dbName, dsProvider) => {
const config = setup.getConfig() const config = setup.getConfig()
const isOracle = dbName === DatabaseName.ORACLE
const isMsSQL = dbName === DatabaseName.SQL_SERVER
let rawDatasource: Datasource let rawDatasource: Datasource
let datasource: Datasource let datasource: Datasource
let client: Knex let client: Knex
@ -97,7 +101,7 @@ describe.each(
const query = await createQuery({ const query = await createQuery({
name: "New Query", name: "New Query",
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
}) })
@ -106,7 +110,7 @@ describe.each(
name: "New Query", name: "New Query",
parameters: [], parameters: [],
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
schema: {}, schema: {},
queryVerb: "read", queryVerb: "read",
@ -125,7 +129,7 @@ describe.each(
it("should be able to update a query", async () => { it("should be able to update a query", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
}) })
@ -135,7 +139,7 @@ describe.each(
...query, ...query,
name: "Updated Query", name: "Updated Query",
fields: { fields: {
sql: "SELECT * FROM test_table WHERE id = 1", sql: client("test_table").where({ id: 1 }).toString(),
}, },
}) })
@ -144,7 +148,7 @@ describe.each(
name: "Updated Query", name: "Updated Query",
parameters: [], parameters: [],
fields: { fields: {
sql: "SELECT * FROM test_table WHERE id = 1", sql: client("test_table").where({ id: 1 }).toString(),
}, },
schema: {}, schema: {},
queryVerb: "read", queryVerb: "read",
@ -161,7 +165,7 @@ describe.each(
it("should be able to delete a query", async () => { it("should be able to delete a query", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
}) })
@ -180,7 +184,7 @@ describe.each(
it("should be able to list queries", async () => { it("should be able to list queries", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
}) })
@ -191,7 +195,7 @@ describe.each(
it("should strip sensitive fields for prod apps", async () => { it("should strip sensitive fields for prod apps", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table", sql: client("test_table").select("*").toString(),
}, },
}) })
@ -212,7 +216,7 @@ describe.each(
datasourceId: datasource._id!, datasourceId: datasource._id!,
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT * FROM test_table WHERE id = 1`, sql: client("test_table").where({ id: 1 }).toString(),
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -270,7 +274,7 @@ describe.each(
name: "Test Query", name: "Test Query",
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT * FROM ${tableName}`, sql: client(tableName).select("*").toString(),
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -284,11 +288,13 @@ describe.each(
}) })
) )
await client(tableName).delete()
await client.schema.alterTable(tableName, table => { await client.schema.alterTable(tableName, table => {
table.string("data").alter() table.string("data").alter()
}) })
await client(tableName).update({ await client(tableName).insert({
name: "test",
data: "string value", data: "string value",
}) })
@ -297,7 +303,7 @@ describe.each(
name: "Test Query", name: "Test Query",
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT * FROM ${tableName}`, sql: client(tableName).select("*").toString(),
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -311,6 +317,7 @@ describe.each(
}) })
) )
}) })
it("should work with static variables", async () => { it("should work with static variables", async () => {
await config.api.datasource.update({ await config.api.datasource.update({
...datasource, ...datasource,
@ -326,7 +333,7 @@ describe.each(
datasourceId: datasource._id!, datasourceId: datasource._id!,
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT '{{ foo }}' as foo`, sql: `SELECT '{{ foo }}' AS foo ${isOracle ? "FROM dual" : ""}`,
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -337,16 +344,17 @@ describe.each(
const response = await config.api.query.preview(request) const response = await config.api.query.preview(request)
let key = isOracle ? "FOO" : "foo"
expect(response.schema).toEqual({ expect(response.schema).toEqual({
foo: { [key]: {
name: "foo", name: key,
type: "string", type: "string",
}, },
}) })
expect(response.rows).toEqual([ expect(response.rows).toEqual([
{ {
foo: "bar", [key]: "bar",
}, },
]) ])
}) })
@ -354,7 +362,7 @@ describe.each(
it("should work with dynamic variables", async () => { it("should work with dynamic variables", async () => {
const basedOnQuery = await createQuery({ const basedOnQuery = await createQuery({
fields: { fields: {
sql: "SELECT name FROM test_table WHERE id = 1", sql: client("test_table").select("name").where({ id: 1 }).toString(),
}, },
}) })
@ -376,7 +384,7 @@ describe.each(
datasourceId: datasource._id!, datasourceId: datasource._id!,
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT '{{ foo }}' as foo`, sql: `SELECT '{{ foo }}' AS foo ${isOracle ? "FROM dual" : ""}`,
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -385,16 +393,17 @@ describe.each(
readable: true, readable: true,
}) })
let key = isOracle ? "FOO" : "foo"
expect(preview.schema).toEqual({ expect(preview.schema).toEqual({
foo: { [key]: {
name: "foo", name: key,
type: "string", type: "string",
}, },
}) })
expect(preview.rows).toEqual([ expect(preview.rows).toEqual([
{ {
foo: "one", [key]: "one",
}, },
]) ])
}) })
@ -402,7 +411,7 @@ describe.each(
it("should handle the dynamic base query being deleted", async () => { it("should handle the dynamic base query being deleted", async () => {
const basedOnQuery = await createQuery({ const basedOnQuery = await createQuery({
fields: { fields: {
sql: "SELECT name FROM test_table WHERE id = 1", sql: client("test_table").select("name").where({ id: 1 }).toString(),
}, },
}) })
@ -426,7 +435,7 @@ describe.each(
datasourceId: datasource._id!, datasourceId: datasource._id!,
queryVerb: "read", queryVerb: "read",
fields: { fields: {
sql: `SELECT '{{ foo }}' as foo`, sql: `SELECT '{{ foo }}' AS foo ${isOracle ? "FROM dual" : ""}`,
}, },
parameters: [], parameters: [],
transformer: "return data", transformer: "return data",
@ -435,16 +444,17 @@ describe.each(
readable: true, readable: true,
}) })
let key = isOracle ? "FOO" : "foo"
expect(preview.schema).toEqual({ expect(preview.schema).toEqual({
foo: { [key]: {
name: "foo", name: key,
type: "string", type: "string",
}, },
}) })
expect(preview.rows).toEqual([ expect(preview.rows).toEqual([
{ {
foo: datasource.source === SourceName.SQL_SERVER ? "" : null, [key]: datasource.source === SourceName.SQL_SERVER ? "" : null,
}, },
]) ])
}) })
@ -455,7 +465,7 @@ describe.each(
it("should be able to insert with bindings", async () => { it("should be able to insert with bindings", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "INSERT INTO test_table (name) VALUES ({{ foo }})", sql: client("test_table").insert({ name: "{{ foo }}" }).toString(),
}, },
parameters: [ parameters: [
{ {
@ -488,7 +498,7 @@ describe.each(
it("should not allow handlebars as parameters", async () => { it("should not allow handlebars as parameters", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "INSERT INTO test_table (name) VALUES ({{ foo }})", sql: client("test_table").insert({ name: "{{ foo }}" }).toString(),
}, },
parameters: [ parameters: [
{ {
@ -516,46 +526,55 @@ describe.each(
) )
}) })
it.each(["2021-02-05T12:01:00.000Z", "2021-02-05"])( // Oracle doesn't automatically coerce strings into dates.
"should coerce %s into a date", !isOracle &&
async datetimeStr => { it.each(["2021-02-05T12:01:00.000Z", "2021-02-05"])(
const date = new Date(datetimeStr) "should coerce %s into a date",
const query = await createQuery({ async datetimeStr => {
fields: { const date = new Date(datetimeStr)
sql: `INSERT INTO test_table (name, birthday) VALUES ('foo', {{ birthday }})`, const query = await createQuery({
}, fields: {
parameters: [ sql: client("test_table")
{ .insert({
name: "birthday", name: "foo",
default: "", birthday: client.raw("{{ birthday }}"),
})
.toString(),
}, },
], parameters: [
queryVerb: "create", {
}) name: "birthday",
default: "",
},
],
queryVerb: "create",
})
const result = await config.api.query.execute(query._id!, { const result = await config.api.query.execute(query._id!, {
parameters: { birthday: datetimeStr }, parameters: { birthday: datetimeStr },
}) })
expect(result.data).toEqual([{ created: true }]) expect(result.data).toEqual([{ created: true }])
const rows = await client("test_table") const rows = await client("test_table")
.where({ birthday: datetimeStr }) .where({ birthday: datetimeStr })
.select() .select()
expect(rows).toHaveLength(1) expect(rows).toHaveLength(1)
for (const row of rows) { for (const row of rows) {
expect(new Date(row.birthday)).toEqual(date) expect(new Date(row.birthday)).toEqual(date)
}
} }
} )
)
it.each(["2021,02,05", "202205-1500"])( it.each(["2021,02,05", "202205-1500"])(
"should not coerce %s as a date", "should not coerce %s as a date",
async notDateStr => { async notDateStr => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "INSERT INTO test_table (name) VALUES ({{ name }})", sql: client("test_table")
.insert({ name: client.raw("{{ name }}") })
.toString(),
}, },
parameters: [ parameters: [
{ {
@ -586,7 +605,7 @@ describe.each(
it("should execute a query", async () => { it("should execute a query", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table ORDER BY id", sql: client("test_table").select("*").orderBy("id").toString(),
}, },
}) })
@ -629,7 +648,7 @@ describe.each(
it("should be able to transform a query", async () => { it("should be able to transform a query", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table WHERE id = 1", sql: client("test_table").where({ id: 1 }).select("*").toString(),
}, },
transformer: ` transformer: `
data[0].id = data[0].id + 1; data[0].id = data[0].id + 1;
@ -652,7 +671,10 @@ describe.each(
it("should coerce numeric bindings", async () => { it("should coerce numeric bindings", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "SELECT * FROM test_table WHERE id = {{ id }}", sql: client("test_table")
.where({ id: client.raw("{{ id }}") })
.select("*")
.toString(),
}, },
parameters: [ parameters: [
{ {
@ -683,7 +705,10 @@ describe.each(
it("should be able to update rows", async () => { it("should be able to update rows", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "UPDATE test_table SET name = {{ name }} WHERE id = {{ id }}", sql: client("test_table")
.update({ name: client.raw("{{ name }}") })
.where({ id: client.raw("{{ id }}") })
.toString(),
}, },
parameters: [ parameters: [
{ {
@ -698,19 +723,13 @@ describe.each(
queryVerb: "update", queryVerb: "update",
}) })
const result = await config.api.query.execute(query._id!, { await config.api.query.execute(query._id!, {
parameters: { parameters: {
id: "1", id: "1",
name: "foo", name: "foo",
}, },
}) })
expect(result.data).toEqual([
{
updated: true,
},
])
const rows = await client("test_table").where({ id: 1 }).select() const rows = await client("test_table").where({ id: 1 }).select()
expect(rows).toEqual([ expect(rows).toEqual([
{ id: 1, name: "foo", birthday: null, number: null }, { id: 1, name: "foo", birthday: null, number: null },
@ -720,35 +739,34 @@ describe.each(
it("should be able to execute an update that updates no rows", async () => { it("should be able to execute an update that updates no rows", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "UPDATE test_table SET name = 'updated' WHERE id = 100", sql: client("test_table")
.update({ name: "updated" })
.where({ id: 100 })
.toString(),
}, },
queryVerb: "update", queryVerb: "update",
}) })
const result = await config.api.query.execute(query._id!) await config.api.query.execute(query._id!)
expect(result.data).toEqual([ const rows = await client("test_table").select()
{ for (const row of rows) {
updated: true, expect(row.name).not.toEqual("updated")
}, }
])
}) })
it("should be able to execute a delete that deletes no rows", async () => { it("should be able to execute a delete that deletes no rows", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "DELETE FROM test_table WHERE id = 100", sql: client("test_table").where({ id: 100 }).delete().toString(),
}, },
queryVerb: "delete", queryVerb: "delete",
}) })
const result = await config.api.query.execute(query._id!) await config.api.query.execute(query._id!)
expect(result.data).toEqual([ const rows = await client("test_table").select()
{ expect(rows).toHaveLength(5)
deleted: true,
},
])
}) })
}) })
@ -756,7 +774,10 @@ describe.each(
it("should be able to delete rows", async () => { it("should be able to delete rows", async () => {
const query = await createQuery({ const query = await createQuery({
fields: { fields: {
sql: "DELETE FROM test_table WHERE id = {{ id }}", sql: client("test_table")
.where({ id: client.raw("{{ id }}") })
.delete()
.toString(),
}, },
parameters: [ parameters: [
{ {
@ -767,18 +788,12 @@ describe.each(
queryVerb: "delete", queryVerb: "delete",
}) })
const result = await config.api.query.execute(query._id!, { await config.api.query.execute(query._id!, {
parameters: { parameters: {
id: "1", id: "1",
}, },
}) })
expect(result.data).toEqual([
{
deleted: true,
},
])
const rows = await client("test_table").where({ id: 1 }).select() const rows = await client("test_table").where({ id: 1 }).select()
expect(rows).toHaveLength(0) expect(rows).toHaveLength(0)
}) })
@ -823,72 +838,63 @@ describe.each(
}) })
}) })
it("should be able to execute an update that updates no rows", async () => { // this parameter really only impacts SQL queries
const query = await createQuery({ describe("confirm nullDefaultSupport", () => {
fields: { let queryParams: Partial<Query>
sql: "UPDATE test_table SET name = 'updated' WHERE id = 100", beforeAll(async () => {
}, queryParams = {
queryVerb: "update", fields: {
sql: client("test_table")
.insert({
name: client.raw("{{ bindingName }}"),
number: client.raw("{{ bindingNumber }}"),
})
.toString(),
},
parameters: [
{
name: "bindingName",
default: "",
},
{
name: "bindingNumber",
default: "",
},
],
queryVerb: "create",
}
}) })
const result = await config.api.query.execute(query._id!, {}) it("should error for old queries", async () => {
const query = await createQuery(queryParams)
await config.api.query.save({ ...query, nullDefaultSupport: false })
let error: string | undefined
try {
await config.api.query.execute(query._id!, {
parameters: {
bindingName: "testing",
},
})
} catch (err: any) {
error = err.message
}
if (isMsSQL || isOracle) {
expect(error).toBeUndefined()
} else {
expect(error).toBeDefined()
expect(error).toContain("integer")
}
})
expect(result.data).toEqual([ it("should not error for new queries", async () => {
{ const query = await createQuery(queryParams)
updated: true, const results = await config.api.query.execute(query._id!, {
},
])
})
})
// this parameter really only impacts SQL queries
describe("confirm nullDefaultSupport", () => {
const queryParams = {
fields: {
sql: "INSERT INTO test_table (name, number) VALUES ({{ bindingName }}, {{ bindingNumber }})",
},
parameters: [
{
name: "bindingName",
default: "",
},
{
name: "bindingNumber",
default: "",
},
],
queryVerb: "create",
}
it("should error for old queries", async () => {
const query = await createQuery(queryParams)
await config.api.query.save({ ...query, nullDefaultSupport: false })
let error: string | undefined
try {
await config.api.query.execute(query._id!, {
parameters: { parameters: {
bindingName: "testing", bindingName: "testing",
}, },
}) })
} catch (err: any) { expect(results).toEqual({ data: [{ created: true }] })
error = err.message
}
if (dbName === "mssql") {
expect(error).toBeUndefined()
} else {
expect(error).toBeDefined()
expect(error).toContain("integer")
}
})
it("should not error for new queries", async () => {
const query = await createQuery(queryParams)
const results = await config.api.query.execute(query._id!, {
parameters: {
bindingName: "testing",
},
}) })
expect(results).toEqual({ data: [{ created: true }] })
}) })
}) })
}) })