From 544302a12992b22eed209626d01fe12567f006e3 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 2 Dec 2024 12:42:48 +0100 Subject: [PATCH 01/60] Request only needed fields --- packages/backend-core/src/sql/sql.ts | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 5f462ee144..a4828a9f4a 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1537,11 +1537,16 @@ class InternalBuilder { limits?: { base: number; query: number } } = {} ): Knex.QueryBuilder { - let { operation, filters, paginate, relationships, table } = this.query + const { operation, filters, paginate, relationships, table, resource } = + this.query const { limits } = opts // start building the query let query = this.qualifiedKnex() + if (resource?.fields) { + query = query.columns(resource?.fields) + } + // handle pagination let foundOffset: number | null = null let foundLimit = limits?.query || limits?.base From 0e0b6471d64e93f2559ab51374c734f5d103eed9 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 11 Dec 2024 10:22:16 +0100 Subject: [PATCH 02/60] Generate proper select --- packages/backend-core/src/sql/sql.ts | 5 +---- packages/server/src/api/controllers/row/utils/sqlUtils.ts | 1 + 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index a4828a9f4a..76d554d211 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -292,7 +292,7 @@ class InternalBuilder { const alias = this.getTableName(table) const schema = this.table.schema - if (!this.isFullSelectStatementRequired()) { + if (this.isFullSelectStatementRequired()) { return [this.knex.raw("??", [`${alias}.*`])] } // get just the fields for this table @@ -1543,9 +1543,6 @@ class InternalBuilder { // start building the query let query = this.qualifiedKnex() - if (resource?.fields) { - query = query.columns(resource?.fields) - } // handle pagination let foundOffset: number | null = null diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index af696c0758..9729018bf7 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -142,6 +142,7 @@ export async function buildSqlFieldList( let table: Table if (sdk.views.isView(source)) { table = await sdk.views.getTable(source.id) + fields = fields.filter(f => table.schema[f].type !== FieldType.LINK) } else { table = source } From 191b63270cabd2a76eb85c401f6308821d02de47 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 11 Dec 2024 13:30:08 +0100 Subject: [PATCH 03/60] Dry --- packages/server/src/api/controllers/row/utils/sqlUtils.ts | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 9729018bf7..19f251bef3 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -133,14 +133,17 @@ export async function buildSqlFieldList( } let fields: string[] = [] - if (sdk.views.isView(source)) { + + const isView = sdk.views.isView(source) + + if (isView) { fields = Object.keys(helpers.views.basicFields(source)) } else { fields = extractRealFields(source) } let table: Table - if (sdk.views.isView(source)) { + if (isView) { table = await sdk.views.getTable(source.id) fields = fields.filter(f => table.schema[f].type !== FieldType.LINK) } else { From a415095c470f2d8a59575c4b6a38c0e0ecc67a5b Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 11 Dec 2024 14:09:11 +0100 Subject: [PATCH 04/60] Lint --- packages/backend-core/src/sql/sql.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 76d554d211..758dfa7fec 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1537,8 +1537,7 @@ class InternalBuilder { limits?: { base: number; query: number } } = {} ): Knex.QueryBuilder { - const { operation, filters, paginate, relationships, table, resource } = - this.query + const { operation, filters, paginate, relationships, table } = this.query const { limits } = opts // start building the query From 74e1bbc7c66faa507edac9e56eb1214aa0b93384 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 12 Dec 2024 13:57:44 +0100 Subject: [PATCH 05/60] Require only visible fields on views --- .../server/src/api/controllers/row/utils/sqlUtils.ts | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 19f251bef3..b67ef6be92 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -150,10 +150,19 @@ export async function buildSqlFieldList( table = source } - for (let field of Object.values(table.schema)) { + for (const field of Object.values(table.schema)) { if (field.type !== FieldType.LINK || !relationships || !field.tableId) { continue } + + if ( + isView && + source.schema?.[field.name] && + !helpers.views.isVisible(source.schema[field.name]) + ) { + continue + } + const { tableName } = breakExternalTableId(field.tableId) if (tables[tableName]) { fields = fields.concat(extractRealFields(tables[tableName], fields)) From e3c9156aef80c593b35936f853fba064936e8b04 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 12 Dec 2024 13:58:37 +0100 Subject: [PATCH 06/60] Trim selected fields --- packages/backend-core/src/sql/sql.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 758dfa7fec..4691cd71ba 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1591,7 +1591,7 @@ class InternalBuilder { const mainTable = this.query.tableAliases?.[table.name] || table.name const cte = this.addSorting( this.knex - .with("paginated", query) + .with("paginated", query.clone().clearSelect().select("*")) .select(this.generateSelectStatement()) .from({ [mainTable]: "paginated", From 00557e5f7ea87f44dfe0cc6ca7202a3af28653e9 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 12 Dec 2024 14:01:56 +0100 Subject: [PATCH 07/60] Don't include unnecessary joins --- packages/backend-core/src/sql/sql.ts | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 4691cd71ba..0fe7e0ca57 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1239,6 +1239,12 @@ class InternalBuilder { if (!toTable || !fromTable) { continue } + + // Don't include if not required + if (relationship.from && !fields.find(f => f === relationship.from)) { + continue + } + const relatedTable = tables[toTable] if (!relatedTable) { throw new Error(`related table "${toTable}" not found in datasource`) From 910faa6f33720cc3b396449cf8e036a6f77534ca Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 12 Dec 2024 17:10:01 +0100 Subject: [PATCH 08/60] Fix getting relations --- packages/backend-core/src/sql/sql.ts | 16 +--------------- 1 file changed, 1 insertion(+), 15 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 0fe7e0ca57..24f30d57fa 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1240,11 +1240,6 @@ class InternalBuilder { continue } - // Don't include if not required - if (relationship.from && !fields.find(f => f === relationship.from)) { - continue - } - const relatedTable = tables[toTable] if (!relatedTable) { throw new Error(`related table "${toTable}" not found in datasource`) @@ -1594,17 +1589,8 @@ class InternalBuilder { // handle relationships with a CTE for all others if (relationships?.length && aggregations.length === 0) { - const mainTable = this.query.tableAliases?.[table.name] || table.name - const cte = this.addSorting( - this.knex - .with("paginated", query.clone().clearSelect().select("*")) - .select(this.generateSelectStatement()) - .from({ - [mainTable]: "paginated", - }) - ) // add JSON aggregations attached to the CTE - return this.addJsonRelationships(cte, table.name, relationships) + return this.addJsonRelationships(query, table.name, relationships) } return query From be0074105cc8126c2029acc76b03bad8846b998e Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 12 Dec 2024 17:25:07 +0100 Subject: [PATCH 09/60] Request relation only when required --- packages/backend-core/src/sql/sql.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 24f30d57fa..c38c0bd2c8 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1268,6 +1268,10 @@ class InternalBuilder { const fieldList = relationshipFields.map(field => this.buildJsonField(relatedTable, field) ) + if (!fieldList.length) { + continue + } + const fieldListFormatted = fieldList .map(f => { const separator = this.client === SqlClient.ORACLE ? " VALUE " : "," From 112b7c70e23e6d2c2b3f0d1f655ff1d622391ddf Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Fri, 13 Dec 2024 11:21:24 +0100 Subject: [PATCH 10/60] Fix sqs views --- .../server/src/sdk/app/rows/search/internal/sqs.ts | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index aa799390b8..6eb9bd913b 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -234,6 +234,17 @@ async function runSqlQuery( json.operation = Operation.COUNT } const processSQLQuery = async (json: EnrichedQueryJson) => { + const fields = json.resource?.fields + if (fields) { + const tableId = json.tableAliases?.[json.table._id!] ?? json.table._id! + for (const key of ["_id", "_rev", "tableId"]) { + const field = `${tableId}.${key}` + if (fields.includes(field)) { + continue + } + fields.push(field) + } + } const query = builder._query(json, { disableReturning: true, }) From bbb60afdc2d01c5c0f9e6e1bc4196139a351645b Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Fri, 13 Dec 2024 13:30:39 +0100 Subject: [PATCH 11/60] Relations, select only required fields --- .../src/api/controllers/row/utils/sqlUtils.ts | 20 +++++++++++++++++-- 1 file changed, 18 insertions(+), 2 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index b67ef6be92..51fba4bb79 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -164,8 +164,24 @@ export async function buildSqlFieldList( } const { tableName } = breakExternalTableId(field.tableId) - if (tables[tableName]) { - fields = fields.concat(extractRealFields(tables[tableName], fields)) + const relatedTable = tables[tableName] + if (relatedTable) { + const viewFields = new Set() + relatedTable.primary?.forEach(f => viewFields.add(f)) + if (relatedTable.primaryDisplay) { + viewFields.add(relatedTable.primaryDisplay) + } + + if (isView) { + Object.entries(source.schema?.[field.name].columns || {}) + .filter(([_, column]) => helpers.views.isVisible(column)) + .forEach(([field]) => viewFields.add(field)) + } + + const fieldsToAdd = Array.from(viewFields) + .map(f => `${relatedTable.name}.${f}`) + .filter(f => !fields.includes(f)) + fields.push(...fieldsToAdd) } } From 80dcc519235d419ed3224ac729c747beebd86517 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 09:55:18 +0100 Subject: [PATCH 12/60] Fix sql alias tests --- .../src/integrations/tests/sqlAlias.spec.ts | 35 ++++++++----------- 1 file changed, 15 insertions(+), 20 deletions(-) diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index 7c6f583762..f7dab0ff46 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -79,7 +79,7 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [primaryLimit, relationshipLimit, relationshipLimit], + bindings: [relationshipLimit, relationshipLimit, primaryLimit], sql: expect.stringContaining( multiline( `select json_agg(json_build_object('executorid',"b"."executorid",'executorid',"b"."executorid",'qaid',"b"."qaid",'qaid',"b"."qaid",'taskid',"b"."taskid",'taskid',"b"."taskid",'completed',"b"."completed",'completed',"b"."completed",'taskname',"b"."taskname",'taskname',"b"."taskname"` @@ -94,10 +94,10 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: ["assembling", primaryLimit, relationshipLimit], + bindings: [relationshipLimit, "assembling", primaryLimit], sql: expect.stringContaining( multiline( - `where (exists (select 1 from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" and (COALESCE("b"."taskname" = $1, FALSE)))` + `where (exists (select 1 from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" and (COALESCE("b"."taskname" = $2, FALSE)))` ) ), }) @@ -109,13 +109,13 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [primaryLimit, relationshipLimit], + bindings: [relationshipLimit, primaryLimit], sql: expect.stringContaining( multiline( - `with "paginated" as (select "a".* from "products" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc limit $1) - select "a".*, (select json_agg(json_build_object('executorid',"b"."executorid",'qaid',"b"."qaid",'taskid',"b"."taskid",'completed',"b"."completed",'taskname',"b"."taskname")) - from (select "b".* from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $2) as "b") as "tasks" - from "paginated" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc` + `select "a"."productname", + "a"."productid", + (select json_agg(json_build_object('executorid',"b"."executorid",'qaid',"b"."qaid",'taskid',"b"."taskid",'completed',"b"."completed",'taskname',"b"."taskname")) from (select "b".* from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $1) as "b") as "tasks" + from "products" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc limit $2` ) ), }) @@ -128,13 +128,8 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [...filters, relationshipLimit, relationshipLimit], - sql: multiline( - `with "paginated" as (select "a".* from "tasks" as "a" where "a"."taskid" in ($1, $2) order by "a"."taskid" asc limit $3) - select "a".*, (select json_agg(json_build_object('productid',"b"."productid",'productname',"b"."productname")) - from (select "b".* from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" - where "c"."taskid" = "a"."taskid" order by "b"."productid" asc limit $4) as "b") as "products" from "paginated" as "a" order by "a"."taskid" asc` - ), + bindings: [relationshipLimit, ...filters, relationshipLimit], + sql: `select "a"."executorid", "a"."taskname", "a"."taskid", "a"."completed", "a"."qaid", (select json_agg(json_build_object('productid',"b"."productid",'productname',"b"."productname")) from (select "b".* from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" where "c"."taskid" = "a"."taskid" order by "b"."productid" asc limit $1) as "b") as "products" from "tasks" as "a" where "a"."taskid" in ($2, $3) order by "a"."taskid" asc limit $4`, }) }) @@ -151,6 +146,9 @@ describe("Captures of real examples", () => { expect(query).toEqual({ bindings: [ + relationshipLimit, + relationshipLimit, + relationshipLimit, rangeValue.low, rangeValue.high, rangeValue.low, @@ -158,13 +156,10 @@ describe("Captures of real examples", () => { equalValue, notEqualsValue, primaryLimit, - relationshipLimit, - relationshipLimit, - relationshipLimit, ], sql: expect.stringContaining( multiline( - `where (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."executorid" and ("c"."year" between $1 and $2))) and (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."qaid" and ("c"."year" between $3 and $4))) and (exists (select 1 from "products" as "b" inner join "products_tasks" as "d" on "b"."productid" = "d"."productid" where "d"."taskid" = "a"."taskid" and (COALESCE("b"."productname" = $5, FALSE))))` + `where (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."executorid" and ("c"."year" between $4 and $5))) and (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."qaid" and ("c"."year" between $6 and $7))) and (exists (select 1 from "products" as "b" inner join "products_tasks" as "d" on "b"."productid" = "d"."productid" where "d"."taskid" = "a"."taskid" and (COALESCE("b"."productname" = $8, FALSE))))` ) ), }) @@ -209,7 +204,7 @@ describe("Captures of real examples", () => { bindings: ["ddd", ""], sql: multiline(`delete from "compositetable" as "a" where COALESCE("a"."keypartone" = $1, FALSE) and COALESCE("a"."keyparttwo" = $2, FALSE) - returning "a".*`), + returning "a"."keyparttwo", "a"."keypartone", "a"."name"`), }) }) }) From f92fcea42abab49afc413aba4f51cb3b4c650b14 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 10:41:37 +0100 Subject: [PATCH 13/60] Include * when having formulas --- packages/backend-core/src/sql/sql.ts | 72 +++++++++++-------- .../src/api/controllers/row/utils/sqlUtils.ts | 2 +- 2 files changed, 43 insertions(+), 31 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index c38c0bd2c8..337c65ff0f 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -272,12 +272,17 @@ class InternalBuilder { return parts.join(".") } - private isFullSelectStatementRequired(): boolean { - for (let column of Object.values(this.table.schema)) { + private isFullSelectStatementRequired(includedFields: string[]): boolean { + for (const column of Object.values(this.table.schema)) { if (this.SPECIAL_SELECT_CASES.POSTGRES_MONEY(column)) { return true } else if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(column)) { return true + } else if ( + column.type === FieldType.FORMULA && + includedFields.includes(column.name) + ) { + return true } } return false @@ -292,11 +297,9 @@ class InternalBuilder { const alias = this.getTableName(table) const schema = this.table.schema - if (this.isFullSelectStatementRequired()) { - return [this.knex.raw("??", [`${alias}.*`])] - } + // get just the fields for this table - return resource.fields + const tableFields = resource.fields .map(field => { const parts = field.split(/\./g) let table: string | undefined = undefined @@ -311,34 +314,43 @@ class InternalBuilder { return { table, column, field } }) .filter(({ table }) => !table || table === alias) - .map(({ table, column, field }) => { - const columnSchema = schema[column] - if (this.SPECIAL_SELECT_CASES.POSTGRES_MONEY(columnSchema)) { - return this.knex.raw(`??::money::numeric as ??`, [ - this.rawQuotedIdentifier([table, column].join(".")), - this.knex.raw(this.quote(field)), - ]) - } + if ( + this.isFullSelectStatementRequired( + tableFields.map(({ column }) => column) + ) + ) { + return [this.knex.raw("??", [`${alias}.*`])] + } - if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(columnSchema)) { - // Time gets returned as timestamp from mssql, not matching the expected - // HH:mm format + return tableFields.map(({ table, column, field }) => { + const columnSchema = schema[column] - // TODO: figure out how to express this safely without string - // interpolation. - return this.knex.raw(`CONVERT(varchar, ??, 108) as ??`, [ - this.rawQuotedIdentifier(field), - this.knex.raw(this.quote(field)), - ]) - } + if (this.SPECIAL_SELECT_CASES.POSTGRES_MONEY(columnSchema)) { + return this.knex.raw(`??::money::numeric as ??`, [ + this.rawQuotedIdentifier([table, column].join(".")), + this.knex.raw(this.quote(field)), + ]) + } - if (table) { - return this.rawQuotedIdentifier(`${table}.${column}`) - } else { - return this.rawQuotedIdentifier(field) - } - }) + if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(columnSchema)) { + // Time gets returned as timestamp from mssql, not matching the expected + // HH:mm format + + // TODO: figure out how to express this safely without string + // interpolation. + return this.knex.raw(`CONVERT(varchar, ??, 108) as ??`, [ + this.rawQuotedIdentifier(field), + this.knex.raw(this.quote(field)), + ]) + } + + if (table) { + return this.rawQuotedIdentifier(`${table}.${column}`) + } else { + return this.rawQuotedIdentifier(field) + } + }) } // OracleDB can't use character-large-objects (CLOBs) in WHERE clauses, diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 51fba4bb79..2d5fc35761 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -173,7 +173,7 @@ export async function buildSqlFieldList( } if (isView) { - Object.entries(source.schema?.[field.name].columns || {}) + Object.entries(source.schema?.[field.name]?.columns || {}) .filter(([_, column]) => helpers.views.isVisible(column)) .forEach(([field]) => viewFields.add(field)) } From f8ece826488927c314fec3245a4d4224cd66c7c3 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 10:53:45 +0100 Subject: [PATCH 14/60] Fix formulas on internal --- packages/backend-core/src/sql/sql.ts | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 337c65ff0f..ba3076e553 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -315,11 +315,10 @@ class InternalBuilder { }) .filter(({ table }) => !table || table === alias) - if ( - this.isFullSelectStatementRequired( - tableFields.map(({ column }) => column) - ) - ) { + const requestedTableColumns = tableFields.map(({ column }) => + column.replace(new RegExp(`^${this.query.meta?.columnPrefix}`), "") + ) + if (this.isFullSelectStatementRequired(requestedTableColumns)) { return [this.knex.raw("??", [`${alias}.*`])] } From e6a27ad4d50ebc4776091a0020c4afbc3a44ff17 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 12:25:29 +0100 Subject: [PATCH 15/60] Ensure required fields are included --- .../src/api/controllers/row/utils/sqlUtils.ts | 32 +++++++++++++++++++ 1 file changed, 32 insertions(+) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 2d5fc35761..0c3d2b8607 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -15,6 +15,7 @@ import { breakExternalTableId } from "../../../../integrations/utils" import { generateJunctionTableID } from "../../../../db/utils" import sdk from "../../../../sdk" import { helpers } from "@budibase/shared-core" +import { sql } from "@budibase/backend-core" type TableMap = Record @@ -132,6 +133,27 @@ export async function buildSqlFieldList( .map(([columnName]) => `${table.name}.${columnName}`) } + function getRequiredFields(table: Table, existing: string[] = []) { + const requiredFields: string[] = [] + if (table.primary) { + requiredFields.push(...table.primary) + } + if (table.primaryDisplay) { + requiredFields.push(table.primaryDisplay) + } + + if (!sql.utils.isExternalTable(table)) { + requiredFields.push(...["_id", "_rev", "_tableId"]) + } + + return requiredFields + .filter( + column => + !existing.find((field: string) => field === `${table.name}.${column}`) + ) + .map(column => `${table.name}.${column}`) + } + let fields: string[] = [] const isView = sdk.views.isView(source) @@ -150,6 +172,16 @@ export async function buildSqlFieldList( table = source } + fields.push( + ...getRequiredFields( + { + ...table, + primaryDisplay: source.primaryDisplay || table.primaryDisplay, + }, + fields + ) + ) + for (const field of Object.values(table.schema)) { if (field.type !== FieldType.LINK || !relationships || !field.tableId) { continue From 2d771c96dd81f663ea6360cb6e3997138f03e045 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 12:53:40 +0100 Subject: [PATCH 16/60] Fix wrong relationship select --- .../server/src/api/controllers/row/utils/sqlUtils.ts | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 0c3d2b8607..80c5f95b05 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -206,7 +206,14 @@ export async function buildSqlFieldList( if (isView) { Object.entries(source.schema?.[field.name]?.columns || {}) - .filter(([_, column]) => helpers.views.isVisible(column)) + .filter( + ([columnName, columnConfig]) => + relatedTable.schema[columnName] && + helpers.views.isVisible(columnConfig) && + ![FieldType.LINK, FieldType.FORMULA].includes( + relatedTable.schema[columnName].type + ) + ) .forEach(([field]) => viewFields.add(field)) } From 14b5a4264700e66bf2a7aef0f130bc41d034e885 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 13:57:38 +0100 Subject: [PATCH 17/60] Fix patch issues --- packages/server/src/api/controllers/row/external.ts | 7 +++++-- packages/server/src/api/controllers/row/utils/utils.ts | 2 +- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/packages/server/src/api/controllers/row/external.ts b/packages/server/src/api/controllers/row/external.ts index 082d07283b..6dbd4dbb81 100644 --- a/packages/server/src/api/controllers/row/external.ts +++ b/packages/server/src/api/controllers/row/external.ts @@ -45,6 +45,9 @@ export async function handleRequest( export async function patch(ctx: UserCtx) { const source = await utils.getSource(ctx) + const { viewId, tableId } = utils.getSourceId(ctx) + const sourceId = viewId || tableId + if (sdk.views.isView(source) && helpers.views.isCalculationView(source)) { ctx.throw(400, "Cannot update rows through a calculation view") } @@ -66,7 +69,7 @@ export async function patch(ctx: UserCtx) { throw { validation: validateResult.errors } } - const beforeRow = await sdk.rows.external.getRow(table._id!, _id, { + const beforeRow = await sdk.rows.external.getRow(sourceId, _id, { relationships: true, }) @@ -78,7 +81,7 @@ export async function patch(ctx: UserCtx) { // The id might have been changed, so the refetching would fail. Recalculating the id just in case const updatedId = generateIdForRow({ ...beforeRow, ...dataToUpdate }, table) || _id - const row = await sdk.rows.external.getRow(table._id!, updatedId, { + const row = await sdk.rows.external.getRow(sourceId, updatedId, { relationships: true, }) diff --git a/packages/server/src/api/controllers/row/utils/utils.ts b/packages/server/src/api/controllers/row/utils/utils.ts index baa811fe90..8701ba9197 100644 --- a/packages/server/src/api/controllers/row/utils/utils.ts +++ b/packages/server/src/api/controllers/row/utils/utils.ts @@ -66,7 +66,7 @@ export function getSourceId(ctx: Ctx): { tableId: string; viewId?: string } { if (docIds.isViewId(sourceId)) { return { tableId: utils.extractViewInfoFromID(sourceId).tableId, - viewId: sourceId, + viewId: encodeURI(sourceId), } } return { tableId: sql.utils.encodeTableId(ctx.params.sourceId) } From d60cc7aaf4c4c47ff80d2caecac0837ae1c9c8fd Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 15:51:17 +0100 Subject: [PATCH 18/60] Fix encoding --- packages/server/src/api/controllers/row/utils/utils.ts | 2 +- packages/server/src/db/utils.ts | 10 ++-------- 2 files changed, 3 insertions(+), 9 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/utils.ts b/packages/server/src/api/controllers/row/utils/utils.ts index 8701ba9197..021a6317ea 100644 --- a/packages/server/src/api/controllers/row/utils/utils.ts +++ b/packages/server/src/api/controllers/row/utils/utils.ts @@ -66,7 +66,7 @@ export function getSourceId(ctx: Ctx): { tableId: string; viewId?: string } { if (docIds.isViewId(sourceId)) { return { tableId: utils.extractViewInfoFromID(sourceId).tableId, - viewId: encodeURI(sourceId), + viewId: sql.utils.encodeTableId(sourceId), } } return { tableId: sql.utils.encodeTableId(ctx.params.sourceId) } diff --git a/packages/server/src/db/utils.ts b/packages/server/src/db/utils.ts index 70c69b3c60..6c1065e847 100644 --- a/packages/server/src/db/utils.ts +++ b/packages/server/src/db/utils.ts @@ -1,10 +1,4 @@ -import { - context, - db as dbCore, - docIds, - utils, - sql, -} from "@budibase/backend-core" +import { context, db as dbCore, docIds, utils } from "@budibase/backend-core" import { DatabaseQueryOpts, Datasource, @@ -334,7 +328,7 @@ export function extractViewInfoFromID(viewId: string) { const regex = new RegExp(`^(?.+)${SEPARATOR}([^${SEPARATOR}]+)$`) const res = regex.exec(viewId) return { - tableId: sql.utils.encodeTableId(res!.groups!["tableId"]), + tableId: res!.groups!["tableId"], } } From 0ef4a154ef9b98b41eed2fcfdf424ff6ba3a4724 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 16:02:10 +0100 Subject: [PATCH 19/60] Prevent repeated fields on select --- .../src/api/controllers/row/utils/sqlUtils.ts | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 80c5f95b05..b6215cc5b6 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -158,12 +158,6 @@ export async function buildSqlFieldList( const isView = sdk.views.isView(source) - if (isView) { - fields = Object.keys(helpers.views.basicFields(source)) - } else { - fields = extractRealFields(source) - } - let table: Table if (isView) { table = await sdk.views.getTable(source.id) @@ -172,6 +166,14 @@ export async function buildSqlFieldList( table = source } + if (isView) { + fields = Object.keys(helpers.views.basicFields(source)).map( + c => `${table.name}.${c}` + ) + } else { + fields = extractRealFields(source) + } + fields.push( ...getRequiredFields( { From fc22db35ac5a5d341f617446789fdc6d06cb6655 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 16:16:22 +0100 Subject: [PATCH 20/60] Add back cte --- packages/backend-core/src/sql/sql.ts | 11 ++++++++++- .../server/src/api/controllers/row/utils/sqlUtils.ts | 7 +++---- 2 files changed, 13 insertions(+), 5 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index ba3076e553..1a7e3beba4 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1604,8 +1604,17 @@ class InternalBuilder { // handle relationships with a CTE for all others if (relationships?.length && aggregations.length === 0) { + const mainTable = this.query.tableAliases?.[table.name] || table.name + const cte = this.addSorting( + this.knex + .with("paginated", query.clone().clearSelect().select("*")) + .select(this.generateSelectStatement()) + .from({ + [mainTable]: "paginated", + }) + ) // add JSON aggregations attached to the CTE - return this.addJsonRelationships(query, table.name, relationships) + return this.addJsonRelationships(cte, table.name, relationships) } return query diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index b6215cc5b6..9f5e14d732 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -161,15 +161,14 @@ export async function buildSqlFieldList( let table: Table if (isView) { table = await sdk.views.getTable(source.id) - fields = fields.filter(f => table.schema[f].type !== FieldType.LINK) } else { table = source } if (isView) { - fields = Object.keys(helpers.views.basicFields(source)).map( - c => `${table.name}.${c}` - ) + fields = Object.keys(helpers.views.basicFields(source)) + .filter(f => table.schema[f].type !== FieldType.LINK) + .map(c => `${table.name}.${c}`) } else { fields = extractRealFields(source) } From aa288966d89e40ca85a8286f39a8f8eb13cec607 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 16:23:05 +0100 Subject: [PATCH 21/60] Fix tests back --- .../src/integrations/tests/sqlAlias.spec.ts | 33 +++++++++++-------- 1 file changed, 19 insertions(+), 14 deletions(-) diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index f7dab0ff46..1630e9b204 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -79,7 +79,7 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [relationshipLimit, relationshipLimit, primaryLimit], + bindings: [primaryLimit, relationshipLimit, relationshipLimit], sql: expect.stringContaining( multiline( `select json_agg(json_build_object('executorid',"b"."executorid",'executorid',"b"."executorid",'qaid',"b"."qaid",'qaid',"b"."qaid",'taskid',"b"."taskid",'taskid',"b"."taskid",'completed',"b"."completed",'completed',"b"."completed",'taskname',"b"."taskname",'taskname',"b"."taskname"` @@ -94,10 +94,10 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [relationshipLimit, "assembling", primaryLimit], + bindings: ["assembling", primaryLimit, relationshipLimit], sql: expect.stringContaining( multiline( - `where (exists (select 1 from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" and (COALESCE("b"."taskname" = $2, FALSE)))` + `where (exists (select 1 from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" and (COALESCE("b"."taskname" = $1, FALSE)))` ) ), }) @@ -109,13 +109,13 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [relationshipLimit, primaryLimit], + bindings: [primaryLimit, relationshipLimit], sql: expect.stringContaining( multiline( - `select "a"."productname", - "a"."productid", - (select json_agg(json_build_object('executorid',"b"."executorid",'qaid',"b"."qaid",'taskid',"b"."taskid",'completed',"b"."completed",'taskname',"b"."taskname")) from (select "b".* from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $1) as "b") as "tasks" - from "products" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc limit $2` + `with "paginated" as (select * from "products" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc limit $1) + select "a"."productname", "a"."productid", (select json_agg(json_build_object('executorid',"b"."executorid",'qaid',"b"."qaid",'taskid',"b"."taskid",'completed',"b"."completed",'taskname',"b"."taskname")) + from (select "b".* from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $2) as "b") as "tasks" + from "paginated" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc` ) ), }) @@ -128,8 +128,13 @@ describe("Captures of real examples", () => { queryJson ) expect(query).toEqual({ - bindings: [relationshipLimit, ...filters, relationshipLimit], - sql: `select "a"."executorid", "a"."taskname", "a"."taskid", "a"."completed", "a"."qaid", (select json_agg(json_build_object('productid',"b"."productid",'productname',"b"."productname")) from (select "b".* from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" where "c"."taskid" = "a"."taskid" order by "b"."productid" asc limit $1) as "b") as "products" from "tasks" as "a" where "a"."taskid" in ($2, $3) order by "a"."taskid" asc limit $4`, + bindings: [...filters, relationshipLimit, relationshipLimit], + sql: multiline( + `with "paginated" as (select * from "tasks" as "a" where "a"."taskid" in ($1, $2) order by "a"."taskid" asc limit $3) + select "a"."executorid", "a"."taskname", "a"."taskid", "a"."completed", "a"."qaid", (select json_agg(json_build_object('productid',"b"."productid",'productname',"b"."productname")) + from (select "b".* from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" + where "c"."taskid" = "a"."taskid" order by "b"."productid" asc limit $4) as "b") as "products" from "paginated" as "a" order by "a"."taskid" asc` + ), }) }) @@ -146,9 +151,6 @@ describe("Captures of real examples", () => { expect(query).toEqual({ bindings: [ - relationshipLimit, - relationshipLimit, - relationshipLimit, rangeValue.low, rangeValue.high, rangeValue.low, @@ -156,10 +158,13 @@ describe("Captures of real examples", () => { equalValue, notEqualsValue, primaryLimit, + relationshipLimit, + relationshipLimit, + relationshipLimit, ], sql: expect.stringContaining( multiline( - `where (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."executorid" and ("c"."year" between $4 and $5))) and (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."qaid" and ("c"."year" between $6 and $7))) and (exists (select 1 from "products" as "b" inner join "products_tasks" as "d" on "b"."productid" = "d"."productid" where "d"."taskid" = "a"."taskid" and (COALESCE("b"."productname" = $8, FALSE))))` + `where (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."executorid" and ("c"."year" between $1 and $2))) and (exists (select 1 from "persons" as "c" where "c"."personid" = "a"."qaid" and ("c"."year" between $3 and $4))) and (exists (select 1 from "products" as "b" inner join "products_tasks" as "d" on "b"."productid" = "d"."productid" where "d"."taskid" = "a"."taskid" and (COALESCE("b"."productname" = $5, FALSE))))` ) ), }) From eb7fcd0219d33eeb9fc07696e5ece0b519e189b2 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 16:33:43 +0100 Subject: [PATCH 22/60] Don't select * on relationships --- packages/backend-core/src/sql/sql.ts | 4 +++- packages/server/src/integrations/tests/sqlAlias.spec.ts | 4 ++-- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 1a7e3beba4..394ab92fd3 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -1323,7 +1323,9 @@ class InternalBuilder { ) const standardWrap = (select: Knex.Raw): Knex.QueryBuilder => { - subQuery = subQuery.select(`${toAlias}.*`).limit(getRelationshipLimit()) + subQuery = subQuery + .select(relationshipFields) + .limit(getRelationshipLimit()) // @ts-ignore - the from alias syntax isn't in Knex typing return knex.select(select).from({ [toAlias]: subQuery, diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index 1630e9b204..704c003994 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -114,7 +114,7 @@ describe("Captures of real examples", () => { multiline( `with "paginated" as (select * from "products" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc limit $1) select "a"."productname", "a"."productid", (select json_agg(json_build_object('executorid',"b"."executorid",'qaid',"b"."qaid",'taskid',"b"."taskid",'completed',"b"."completed",'taskname',"b"."taskname")) - from (select "b".* from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $2) as "b") as "tasks" + from (select "b"."executorid", "b"."qaid", "b"."taskid", "b"."completed", "b"."taskname" from "tasks" as "b" inner join "products_tasks" as "c" on "b"."taskid" = "c"."taskid" where "c"."productid" = "a"."productid" order by "b"."taskid" asc limit $2) as "b") as "tasks" from "paginated" as "a" order by "a"."productname" asc nulls first, "a"."productid" asc` ) ), @@ -132,7 +132,7 @@ describe("Captures of real examples", () => { sql: multiline( `with "paginated" as (select * from "tasks" as "a" where "a"."taskid" in ($1, $2) order by "a"."taskid" asc limit $3) select "a"."executorid", "a"."taskname", "a"."taskid", "a"."completed", "a"."qaid", (select json_agg(json_build_object('productid',"b"."productid",'productname',"b"."productname")) - from (select "b".* from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" + from (select "b"."productid", "b"."productname" from "products" as "b" inner join "products_tasks" as "c" on "b"."productid" = "c"."productid" where "c"."taskid" = "a"."taskid" order by "b"."productid" asc limit $4) as "b") as "products" from "paginated" as "a" order by "a"."taskid" asc` ), }) From df62845cf9c492aec568e6903ceb0882aec83517 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 16:55:38 +0100 Subject: [PATCH 23/60] Fix encodings --- packages/backend-core/src/sql/utils.ts | 4 ++++ packages/server/src/api/controllers/row/utils/utils.ts | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/packages/backend-core/src/sql/utils.ts b/packages/backend-core/src/sql/utils.ts index 14127a189f..16b352995b 100644 --- a/packages/backend-core/src/sql/utils.ts +++ b/packages/backend-core/src/sql/utils.ts @@ -70,6 +70,10 @@ export function encodeTableId(tableId: string) { } } +export function encodeViewId(viewId: string) { + return encodeURIComponent(viewId) +} + export function breakExternalTableId(tableId: string) { const parts = tableId.split(DOUBLE_SEPARATOR) let datasourceId = parts.shift() diff --git a/packages/server/src/api/controllers/row/utils/utils.ts b/packages/server/src/api/controllers/row/utils/utils.ts index 021a6317ea..86986b64e8 100644 --- a/packages/server/src/api/controllers/row/utils/utils.ts +++ b/packages/server/src/api/controllers/row/utils/utils.ts @@ -66,7 +66,7 @@ export function getSourceId(ctx: Ctx): { tableId: string; viewId?: string } { if (docIds.isViewId(sourceId)) { return { tableId: utils.extractViewInfoFromID(sourceId).tableId, - viewId: sql.utils.encodeTableId(sourceId), + viewId: sql.utils.encodeViewId(sourceId), } } return { tableId: sql.utils.encodeTableId(ctx.params.sourceId) } From 740069ea78a7866ca1373f0b423526d24792d232 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 17:01:02 +0100 Subject: [PATCH 24/60] Use table for get before row --- packages/server/src/api/controllers/row/external.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/src/api/controllers/row/external.ts b/packages/server/src/api/controllers/row/external.ts index 6dbd4dbb81..0405203f2f 100644 --- a/packages/server/src/api/controllers/row/external.ts +++ b/packages/server/src/api/controllers/row/external.ts @@ -69,7 +69,7 @@ export async function patch(ctx: UserCtx) { throw { validation: validateResult.errors } } - const beforeRow = await sdk.rows.external.getRow(sourceId, _id, { + const beforeRow = await sdk.rows.external.getRow(table._id!, _id, { relationships: true, }) From 7cd412bfdcd15e6815c99f2a5b5e6a93ee1caf1c Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 17:46:37 +0100 Subject: [PATCH 25/60] Add basic sql alias test --- .../src/integrations/tests/sqlAlias.spec.ts | 9 ++ .../tests/sqlQueryJson/basicFetch.json | 137 ++++++++++++++++++ 2 files changed, 146 insertions(+) create mode 100644 packages/server/src/integrations/tests/sqlQueryJson/basicFetch.json diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index 704c003994..4d39fa7338 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -73,6 +73,15 @@ describe("Captures of real examples", () => { }) describe("read", () => { + it("should retrieve only requested fields", () => { + const queryJson = getJson("basicFetch.json") + let query = new Sql(SqlClient.POSTGRES)._query(queryJson) + expect(query).toEqual({ + bindings: [primaryLimit], + sql: `select "a"."year", "a"."firstname", "a"."personid", "a"."age", "a"."type", "a"."lastname" from "persons" as "a" order by "a"."firstname" asc nulls first, "a"."personid" asc limit $1`, + }) + }) + it("should handle basic retrieval with relationships", () => { const queryJson = getJson("basicFetchWithRelationships.json") let query = new Sql(SqlClient.POSTGRES, relationshipLimit)._query( diff --git a/packages/server/src/integrations/tests/sqlQueryJson/basicFetch.json b/packages/server/src/integrations/tests/sqlQueryJson/basicFetch.json new file mode 100644 index 0000000000..9d9026c922 --- /dev/null +++ b/packages/server/src/integrations/tests/sqlQueryJson/basicFetch.json @@ -0,0 +1,137 @@ +{ + "operation": "READ", + "resource": { + "fields": [ + "a.year", + "a.firstname", + "a.personid", + "a.age", + "a.type", + "a.lastname" + ] + }, + "filters": {}, + "sort": { + "firstname": { + "direction": "ascending" + } + }, + "paginate": { + "limit": 100, + "page": 1 + }, + "relationships": [], + "extra": { + "idFilter": {} + }, + "table": { + "type": "table", + "_id": "datasource_plus_8066e56456784eb2a00129d31be5c3e7__persons", + "primary": ["personid"], + "name": "persons", + "schema": { + "year": { + "type": "number", + "externalType": "integer", + "autocolumn": false, + "name": "year", + "constraints": { + "presence": false + } + }, + "firstname": { + "type": "string", + "externalType": "character varying", + "autocolumn": false, + "name": "firstname", + "constraints": { + "presence": false + } + }, + "personid": { + "type": "number", + "externalType": "integer", + "autocolumn": true, + "name": "personid", + "constraints": { + "presence": false + } + }, + "address": { + "type": "string", + "externalType": "character varying", + "autocolumn": false, + "name": "address", + "constraints": { + "presence": false + } + }, + "age": { + "type": "number", + "externalType": "integer", + "autocolumn": false, + "name": "age", + "constraints": { + "presence": false + } + }, + "type": { + "type": "options", + "externalType": "USER-DEFINED", + "autocolumn": false, + "name": "type", + "constraints": { + "presence": false, + "inclusion": ["support", "designer", "programmer", "qa"] + } + }, + "city": { + "type": "string", + "externalType": "character varying", + "autocolumn": false, + "name": "city", + "constraints": { + "presence": false + } + }, + "lastname": { + "type": "string", + "externalType": "character varying", + "autocolumn": false, + "name": "lastname", + "constraints": { + "presence": false + } + }, + "QA": { + "tableId": "datasource_plus_8066e56456784eb2a00129d31be5c3e7__tasks", + "name": "QA", + "relationshipType": "many-to-one", + "fieldName": "qaid", + "type": "link", + "main": true, + "_id": "ccb68481c80c34217a4540a2c6c27fe46", + "foreignKey": "personid" + }, + "executor": { + "tableId": "datasource_plus_8066e56456784eb2a00129d31be5c3e7__tasks", + "name": "executor", + "relationshipType": "many-to-one", + "fieldName": "executorid", + "type": "link", + "main": true, + "_id": "c89530b9770d94bec851e062b5cff3001", + "foreignKey": "personid", + "tableName": "persons" + } + }, + "sourceId": "datasource_plus_8066e56456784eb2a00129d31be5c3e7", + "sourceType": "external", + "primaryDisplay": "firstname", + "views": {} + }, + "tableAliases": { + "persons": "a", + "tasks": "b" + } +} From fc75728a1e70f86f214ba2071fc0eaaf6a32a008 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 16 Dec 2024 17:52:30 +0100 Subject: [PATCH 26/60] Add more tests --- .../src/integrations/tests/sqlAlias.spec.ts | 29 +++++++++++++++++++ 1 file changed, 29 insertions(+) diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index 4d39fa7338..647c18d4ae 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -8,6 +8,7 @@ import { TableSchema, Table, TableSourceType, + FieldType, } from "@budibase/types" import { sql } from "@budibase/backend-core" import { join } from "path" @@ -73,8 +74,20 @@ describe("Captures of real examples", () => { }) describe("read", () => { + it("should retrieve all fields if non are specified", () => { + const queryJson = getJson("basicFetch.json") + delete queryJson.resource + + let query = new Sql(SqlClient.POSTGRES)._query(queryJson) + expect(query).toEqual({ + bindings: [primaryLimit], + sql: `select * from "persons" as "a" order by "a"."firstname" asc nulls first, "a"."personid" asc limit $1`, + }) + }) + it("should retrieve only requested fields", () => { const queryJson = getJson("basicFetch.json") + let query = new Sql(SqlClient.POSTGRES)._query(queryJson) expect(query).toEqual({ bindings: [primaryLimit], @@ -82,6 +95,22 @@ describe("Captures of real examples", () => { }) }) + it("should retrieve all fields if a formula column is requested", () => { + const queryJson = getJson("basicFetch.json") + queryJson.table.schema["formula"] = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + queryJson.resource!.fields.push("formula") + + let query = new Sql(SqlClient.POSTGRES)._query(queryJson) + expect(query).toEqual({ + bindings: [primaryLimit], + sql: `select "a".* from "persons" as "a" order by "a"."firstname" asc nulls first, "a"."personid" asc limit $1`, + }) + }) + it("should handle basic retrieval with relationships", () => { const queryJson = getJson("basicFetchWithRelationships.json") let query = new Sql(SqlClient.POSTGRES, relationshipLimit)._query( From 7932ee76204b96bc19b1295403449ed87762dfc1 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 10:42:11 +0100 Subject: [PATCH 27/60] Fix sqs calculations --- .../src/sdk/app/rows/search/internal/sqs.ts | 26 +++++++++---------- 1 file changed, 12 insertions(+), 14 deletions(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index 6eb9bd913b..e1ccb4fb85 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -62,7 +62,8 @@ async function buildInternalFieldList( ) { const { relationships, allowedFields } = opts || {} let schemaFields: string[] = [] - if (sdk.views.isView(source)) { + const isView = sdk.views.isView(source) + if (isView) { schemaFields = Object.keys(helpers.views.basicFields(source)) } else { schemaFields = Object.keys(source.schema).filter( @@ -75,7 +76,7 @@ async function buildInternalFieldList( } let table: Table - if (sdk.views.isView(source)) { + if (isView) { table = await sdk.views.getTable(source.id) } else { table = source @@ -125,6 +126,13 @@ async function buildInternalFieldList( fieldList = fieldList.concat(relatedFields) } } + + if (isView && !helpers.views.isCalculationView(source)) { + for (const field of ["_id", "_rev", "tableId"]) { + fieldList.push(field) + } + } + return [...new Set(fieldList)] } @@ -234,17 +242,6 @@ async function runSqlQuery( json.operation = Operation.COUNT } const processSQLQuery = async (json: EnrichedQueryJson) => { - const fields = json.resource?.fields - if (fields) { - const tableId = json.tableAliases?.[json.table._id!] ?? json.table._id! - for (const key of ["_id", "_rev", "tableId"]) { - const field = `${tableId}.${key}` - if (fields.includes(field)) { - continue - } - fields.push(field) - } - } const query = builder._query(json, { disableReturning: true, }) @@ -334,8 +331,9 @@ export async function search( } let aggregations: Aggregation[] = [] - if (sdk.views.isView(source)) { + if (sdk.views.isView(source) && helpers.views.isCalculationView(source)) { const calculationFields = helpers.views.calculationFields(source) + for (const [key, field] of Object.entries(calculationFields)) { if (options.fields && !options.fields.includes(key)) { continue From 499d3f204128393719fcc166b7c984e2fe17b4a2 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 10:47:43 +0100 Subject: [PATCH 28/60] Fix sql calculations --- .../src/api/controllers/row/utils/sqlUtils.ts | 22 ++++++++++--------- 1 file changed, 12 insertions(+), 10 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 9f5e14d732..ce611b502b 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -169,20 +169,22 @@ export async function buildSqlFieldList( fields = Object.keys(helpers.views.basicFields(source)) .filter(f => table.schema[f].type !== FieldType.LINK) .map(c => `${table.name}.${c}`) + + if (!helpers.views.isCalculationView(source)) { + fields.push( + ...getRequiredFields( + { + ...table, + primaryDisplay: source.primaryDisplay || table.primaryDisplay, + }, + fields + ) + ) + } } else { fields = extractRealFields(source) } - fields.push( - ...getRequiredFields( - { - ...table, - primaryDisplay: source.primaryDisplay || table.primaryDisplay, - }, - fields - ) - ) - for (const field of Object.values(table.schema)) { if (field.type !== FieldType.LINK || !relationships || !field.tableId) { continue From 51ba1f072bd940b5c70f2edad483762dd7c7e12c Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 11:02:54 +0100 Subject: [PATCH 29/60] Add required fields on table search as well --- packages/server/src/sdk/app/rows/search/internal/sqs.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index e1ccb4fb85..a5967acfb0 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -127,7 +127,7 @@ async function buildInternalFieldList( } } - if (isView && !helpers.views.isCalculationView(source)) { + if (!isView || !helpers.views.isCalculationView(source)) { for (const field of ["_id", "_rev", "tableId"]) { fieldList.push(field) } From c398412e00b52eb1aa8bd9298010dc072ea8f0b7 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 11:16:27 +0100 Subject: [PATCH 30/60] Fix pg money --- packages/backend-core/src/sql/sql.ts | 4 +--- packages/server/src/integration-test/postgres.spec.ts | 2 +- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 394ab92fd3..015b7c4751 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -274,9 +274,7 @@ class InternalBuilder { private isFullSelectStatementRequired(includedFields: string[]): boolean { for (const column of Object.values(this.table.schema)) { - if (this.SPECIAL_SELECT_CASES.POSTGRES_MONEY(column)) { - return true - } else if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(column)) { + if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(column)) { return true } else if ( column.type === FieldType.FORMULA && diff --git a/packages/server/src/integration-test/postgres.spec.ts b/packages/server/src/integration-test/postgres.spec.ts index 6e674aa58e..713605eadb 100644 --- a/packages/server/src/integration-test/postgres.spec.ts +++ b/packages/server/src/integration-test/postgres.spec.ts @@ -13,7 +13,7 @@ const mainDescriptions = datasourceDescribe({ if (mainDescriptions.length) { describe.each(mainDescriptions)( - "/postgres integrations", + "/postgres integrations ($dbName)", ({ config, dsProvider }) => { let datasource: Datasource let client: Knex From 95f7eeacce750c1b050f59e94bbdf140f58b3ae0 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 11:30:26 +0100 Subject: [PATCH 31/60] Don't add breaking changes --- packages/server/src/sdk/app/rows/search/internal/sqs.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index a5967acfb0..4eca0e4c09 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -128,7 +128,7 @@ async function buildInternalFieldList( } if (!isView || !helpers.views.isCalculationView(source)) { - for (const field of ["_id", "_rev", "tableId"]) { + for (const field of PROTECTED_INTERNAL_COLUMNS) { fieldList.push(field) } } From 8765a28f0412aa3cb5353c23d655c34ab0116864 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 13:02:34 +0100 Subject: [PATCH 32/60] Move responsability to sql utils --- packages/backend-core/src/sql/sql.ts | 21 ------- .../src/api/controllers/row/utils/sqlUtils.ts | 63 +++++++++---------- .../src/integrations/tests/sqlAlias.spec.ts | 17 ----- 3 files changed, 31 insertions(+), 70 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index 015b7c4751..cc5f226d56 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -272,20 +272,6 @@ class InternalBuilder { return parts.join(".") } - private isFullSelectStatementRequired(includedFields: string[]): boolean { - for (const column of Object.values(this.table.schema)) { - if (this.SPECIAL_SELECT_CASES.MSSQL_DATES(column)) { - return true - } else if ( - column.type === FieldType.FORMULA && - includedFields.includes(column.name) - ) { - return true - } - } - return false - } - private generateSelectStatement(): (string | Knex.Raw)[] | "*" { const { table, resource } = this.query @@ -313,13 +299,6 @@ class InternalBuilder { }) .filter(({ table }) => !table || table === alias) - const requestedTableColumns = tableFields.map(({ column }) => - column.replace(new RegExp(`^${this.query.meta?.columnPrefix}`), "") - ) - if (this.isFullSelectStatementRequired(requestedTableColumns)) { - return [this.knex.raw("??", [`${alias}.*`])] - } - return tableFields.map(({ table, column, field }) => { const columnSchema = schema[column] diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index ce611b502b..750b53861f 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -14,7 +14,7 @@ import { import { breakExternalTableId } from "../../../../integrations/utils" import { generateJunctionTableID } from "../../../../db/utils" import sdk from "../../../../sdk" -import { helpers } from "@budibase/shared-core" +import { helpers, PROTECTED_INTERNAL_COLUMNS } from "@budibase/shared-core" import { sql } from "@budibase/backend-core" type TableMap = Record @@ -126,11 +126,9 @@ export async function buildSqlFieldList( column.type !== FieldType.LINK && column.type !== FieldType.FORMULA && column.type !== FieldType.AI && - !existing.find( - (field: string) => field === `${table.name}.${columnName}` - ) + !existing.find((field: string) => field === columnName) ) - .map(([columnName]) => `${table.name}.${columnName}`) + .map(([columnName]) => columnName) } function getRequiredFields(table: Table, existing: string[] = []) { @@ -143,15 +141,12 @@ export async function buildSqlFieldList( } if (!sql.utils.isExternalTable(table)) { - requiredFields.push(...["_id", "_rev", "_tableId"]) + requiredFields.push(...PROTECTED_INTERNAL_COLUMNS) } - return requiredFields - .filter( - column => - !existing.find((field: string) => field === `${table.name}.${column}`) - ) - .map(column => `${table.name}.${column}`) + return requiredFields.filter( + column => !existing.find((field: string) => field === column) + ) } let fields: string[] = [] @@ -161,30 +156,34 @@ export async function buildSqlFieldList( let table: Table if (isView) { table = await sdk.views.getTable(source.id) + + fields = Object.keys(helpers.views.basicFields(source)).filter( + f => table.schema[f].type !== FieldType.LINK + ) } else { table = source - } - - if (isView) { - fields = Object.keys(helpers.views.basicFields(source)) - .filter(f => table.schema[f].type !== FieldType.LINK) - .map(c => `${table.name}.${c}`) - - if (!helpers.views.isCalculationView(source)) { - fields.push( - ...getRequiredFields( - { - ...table, - primaryDisplay: source.primaryDisplay || table.primaryDisplay, - }, - fields - ) - ) - } - } else { fields = extractRealFields(source) } + // If are requesting for a formula field, we need to retrieve all fields + if (fields.find(f => table.schema[f]?.type === FieldType.FORMULA)) { + fields = extractRealFields(table) + } + + if (!isView || !helpers.views.isCalculationView(source)) { + fields.push( + ...getRequiredFields( + { + ...table, + primaryDisplay: source.primaryDisplay || table.primaryDisplay, + }, + fields + ) + ) + } + + fields = fields.map(c => `${table.name}.${c}`) + for (const field of Object.values(table.schema)) { if (field.type !== FieldType.LINK || !relationships || !field.tableId) { continue @@ -227,7 +226,7 @@ export async function buildSqlFieldList( } } - return fields + return [...new Set(fields)] } export function isKnexEmptyReadResponse(resp: DatasourcePlusQueryResponse) { diff --git a/packages/server/src/integrations/tests/sqlAlias.spec.ts b/packages/server/src/integrations/tests/sqlAlias.spec.ts index 647c18d4ae..898ab9314a 100644 --- a/packages/server/src/integrations/tests/sqlAlias.spec.ts +++ b/packages/server/src/integrations/tests/sqlAlias.spec.ts @@ -8,7 +8,6 @@ import { TableSchema, Table, TableSourceType, - FieldType, } from "@budibase/types" import { sql } from "@budibase/backend-core" import { join } from "path" @@ -95,22 +94,6 @@ describe("Captures of real examples", () => { }) }) - it("should retrieve all fields if a formula column is requested", () => { - const queryJson = getJson("basicFetch.json") - queryJson.table.schema["formula"] = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - queryJson.resource!.fields.push("formula") - - let query = new Sql(SqlClient.POSTGRES)._query(queryJson) - expect(query).toEqual({ - bindings: [primaryLimit], - sql: `select "a".* from "persons" as "a" order by "a"."firstname" asc nulls first, "a"."personid" asc limit $1`, - }) - }) - it("should handle basic retrieval with relationships", () => { const queryJson = getJson("basicFetchWithRelationships.json") let query = new Sql(SqlClient.POSTGRES, relationshipLimit)._query( From 8eb82d3b0597cf6f8151f8e5ef871e8e4df2d3d7 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 16:10:15 +0100 Subject: [PATCH 33/60] Fix mysql formula test --- .../src/api/controllers/row/utils/sqlUtils.ts | 28 +++++++++++++------ 1 file changed, 19 insertions(+), 9 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 750b53861f..0cfc429afe 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -165,8 +165,11 @@ export async function buildSqlFieldList( fields = extractRealFields(source) } + const containsFormula = (isView ? fields : Object.keys(table.schema)).some( + f => table.schema[f]?.type === FieldType.FORMULA + ) // If are requesting for a formula field, we need to retrieve all fields - if (fields.find(f => table.schema[f]?.type === FieldType.FORMULA)) { + if (containsFormula) { fields = extractRealFields(table) } @@ -192,15 +195,22 @@ export async function buildSqlFieldList( if ( isView && source.schema?.[field.name] && - !helpers.views.isVisible(source.schema[field.name]) + !helpers.views.isVisible(source.schema[field.name]) && + !containsFormula ) { continue } const { tableName } = breakExternalTableId(field.tableId) const relatedTable = tables[tableName] - if (relatedTable) { - const viewFields = new Set() + if (!relatedTable) { + continue + } + + const viewFields = new Set() + if (containsFormula) { + extractRealFields(relatedTable).forEach(f => viewFields.add(f)) + } else { relatedTable.primary?.forEach(f => viewFields.add(f)) if (relatedTable.primaryDisplay) { viewFields.add(relatedTable.primaryDisplay) @@ -218,12 +228,12 @@ export async function buildSqlFieldList( ) .forEach(([field]) => viewFields.add(field)) } - - const fieldsToAdd = Array.from(viewFields) - .map(f => `${relatedTable.name}.${f}`) - .filter(f => !fields.includes(f)) - fields.push(...fieldsToAdd) } + + const fieldsToAdd = Array.from(viewFields) + .map(f => `${relatedTable.name}.${f}`) + .filter(f => !fields.includes(f)) + fields.push(...fieldsToAdd) } return [...new Set(fields)] From 23531e15096fec99f100a71c406658ee47230978 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 16:10:21 +0100 Subject: [PATCH 34/60] Fix sqs formula --- .../src/sdk/app/rows/search/internal/sqs.ts | 23 ++++++++++++------- 1 file changed, 15 insertions(+), 8 deletions(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index 4eca0e4c09..c92d186a37 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -62,7 +62,15 @@ async function buildInternalFieldList( ) { const { relationships, allowedFields } = opts || {} let schemaFields: string[] = [] + const isView = sdk.views.isView(source) + let table: Table + if (isView) { + table = await sdk.views.getTable(source.id) + } else { + table = source + } + if (isView) { schemaFields = Object.keys(helpers.views.basicFields(source)) } else { @@ -71,17 +79,16 @@ async function buildInternalFieldList( ) } - if (allowedFields) { + const containsFormula = schemaFields.some( + f => table.schema[f]?.type === FieldType.FORMULA + ) + // If are requesting for a formula field, we need to retrieve all fields + if (containsFormula) { + schemaFields = Object.keys(table.schema) + } else if (allowedFields) { schemaFields = schemaFields.filter(field => allowedFields.includes(field)) } - let table: Table - if (isView) { - table = await sdk.views.getTable(source.id) - } else { - table = source - } - let fieldList: string[] = [] const getJunctionFields = (relatedTable: Table, fields: string[]) => { const junctionFields: string[] = [] From bcc9bba254ff4e75ee275041f5488875de2575dd Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 16:58:15 +0100 Subject: [PATCH 35/60] Fix test --- packages/server/src/api/routes/tests/search.spec.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/server/src/api/routes/tests/search.spec.ts b/packages/server/src/api/routes/tests/search.spec.ts index e97f48afbe..e94e567b43 100644 --- a/packages/server/src/api/routes/tests/search.spec.ts +++ b/packages/server/src/api/routes/tests/search.spec.ts @@ -3399,7 +3399,7 @@ if (descriptions.length) { type: FieldType.LINK, relationshipType: RelationshipType.MANY_TO_ONE, tableId: toRelateTableId, - fieldName: "link", + fieldName: "main", }, }) @@ -3408,7 +3408,7 @@ if (descriptions.length) { ) await config.api.table.save({ ...toRelateTable, - primaryDisplay: "link", + primaryDisplay: "name", }) const relatedRows = await Promise.all([ config.api.row.save(toRelateTable._id!, { From b05b523a6e96169e23dc1cfc7a78b038493c2fb7 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Tue, 17 Dec 2024 17:01:06 +0100 Subject: [PATCH 36/60] Fix issues with display names not being sql tables --- .../src/api/controllers/row/utils/sqlUtils.ts | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index 0cfc429afe..f28075bf84 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -119,13 +119,14 @@ export async function buildSqlFieldList( opts?: { relationships: boolean } ) { const { relationships } = opts || {} + + const nonMappedColumns = [FieldType.LINK, FieldType.FORMULA, FieldType.AI] + function extractRealFields(table: Table, existing: string[] = []) { return Object.entries(table.schema) .filter( ([columnName, column]) => - column.type !== FieldType.LINK && - column.type !== FieldType.FORMULA && - column.type !== FieldType.AI && + !nonMappedColumns.includes(column.type) && !existing.find((field: string) => field === columnName) ) .map(([columnName]) => columnName) @@ -145,7 +146,10 @@ export async function buildSqlFieldList( } return requiredFields.filter( - column => !existing.find((field: string) => field === column) + column => + !existing.find((field: string) => field === column) && + table.schema[column] && + !nonMappedColumns.includes(table.schema[column].type) ) } @@ -231,6 +235,7 @@ export async function buildSqlFieldList( } const fieldsToAdd = Array.from(viewFields) + .filter(f => !nonMappedColumns.includes(relatedTable.schema[f].type)) .map(f => `${relatedTable.name}.${f}`) .filter(f => !fields.includes(f)) fields.push(...fieldsToAdd) From 27d1929388de83c56798dd2433685ec3376e67e2 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 14:09:26 +0100 Subject: [PATCH 37/60] Add basic sqlUtils test --- .../src/api/controllers/row/utils/sqlUtils.ts | 4 +- .../row/utils/tests/sqlUtils.spec.ts | 79 +++++++++++++++++++ 2 files changed, 82 insertions(+), 1 deletion(-) create mode 100644 packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index f28075bf84..e66a0b5bf6 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -166,7 +166,9 @@ export async function buildSqlFieldList( ) } else { table = source - fields = extractRealFields(source) + fields = extractRealFields(source).filter( + f => table.schema[f].visible !== false + ) } const containsFormula = (isView ? fields : Object.keys(table.schema)).some( diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts new file mode 100644 index 0000000000..2ab8e251ea --- /dev/null +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -0,0 +1,79 @@ +import { + AIOperationEnum, + FieldType, + RelationshipType, + Table, +} from "@budibase/types" +import { buildSqlFieldList } from "../sqlUtils" +import { structures } from "../../../../routes/tests/utilities" +import { cloneDeep } from "lodash" + +describe("buildSqlFieldList", () => { + const basicTable: Table = { + ...structures.basicTable(), + name: "table", + schema: { + name: { + type: FieldType.STRING, + name: "name", + }, + description: { + type: FieldType.STRING, + name: "description", + }, + amount: { + type: FieldType.NUMBER, + name: "amount", + }, + }, + } + + it("extracts fields from table schema", async () => { + const result = await buildSqlFieldList(basicTable, {}) + expect(result).toEqual(["table.name", "table.description", "table.amount"]) + }) + + it("excludes hidden fields", async () => { + const table = cloneDeep(basicTable) + table.schema.description.visible = false + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual(["table.name", "table.amount"]) + }) + + it("excludes non-sql fields fields", async () => { + const table = cloneDeep(basicTable) + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + table.schema.ai = { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + } + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + } + + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual(["table.name", "table.description", "table.amount"]) + }) + + it("includes hidden fields if there is a formula column", async () => { + const table = cloneDeep(basicTable) + table.schema.description.visible = false + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual(["table.name", "table.description", "table.amount"]) + }) +}) From 875319e85c1fa97abc4e052845ea61ef1dfde449 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 15:50:51 +0100 Subject: [PATCH 38/60] Relationship tests --- .../row/utils/tests/sqlUtils.spec.ts | 105 +++++++++++++++++- 1 file changed, 101 insertions(+), 4 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 2ab8e251ea..5a1b8ae49e 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -2,28 +2,34 @@ import { AIOperationEnum, FieldType, RelationshipType, + SourceName, Table, } from "@budibase/types" import { buildSqlFieldList } from "../sqlUtils" import { structures } from "../../../../routes/tests/utilities" import { cloneDeep } from "lodash" +import { sql } from "@budibase/backend-core" describe("buildSqlFieldList", () => { const basicTable: Table = { - ...structures.basicTable(), + ...structures.tableForDatasource({ + type: "datasource", + source: SourceName.POSTGRES, + }), name: "table", + _id: sql.utils.buildExternalTableId("ds_id", "table"), schema: { name: { - type: FieldType.STRING, name: "name", + type: FieldType.STRING, }, description: { - type: FieldType.STRING, name: "description", + type: FieldType.STRING, }, amount: { - type: FieldType.NUMBER, name: "amount", + type: FieldType.NUMBER, }, }, } @@ -76,4 +82,95 @@ describe("buildSqlFieldList", () => { const result = await buildSqlFieldList(table, {}) expect(result).toEqual(["table.name", "table.description", "table.amount"]) }) + + it("includes relationships fields when flag", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + primary: ["id"], + primaryDisplay: "name", + schema: { + ...cloneDeep(basicTable).schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.id", + "linkedTable.name", + ]) + }) + + it("includes all relationship fields if there is a formula column", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + schema: { + ...cloneDeep(basicTable).schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + hidden: { + name: "other", + type: FieldType.STRING, + visible: false, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.name", + "linkedTable.description", + "linkedTable.amount", + "linkedTable.id", + "linkedTable.hidden", + ]) + }) }) From 96bddbb545b8ddc1602e20e981d2999f9b5d46a0 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 15:53:09 +0100 Subject: [PATCH 39/60] More relationship tests --- .../row/utils/tests/sqlUtils.spec.ts | 81 +++++++++++++++++++ 1 file changed, 81 insertions(+) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 5a1b8ae49e..767a325c7a 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -152,6 +152,23 @@ describe("buildSqlFieldList", () => { type: FieldType.STRING, visible: false, }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + ai: { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + }, + link: { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + }, }, } @@ -173,4 +190,68 @@ describe("buildSqlFieldList", () => { "linkedTable.hidden", ]) }) + + it("never includes non-sql columns from relationships", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + schema: { + id: { + name: "id", + type: FieldType.NUMBER, + }, + hidden: { + name: "other", + type: FieldType.STRING, + visible: false, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + ai: { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + }, + link: { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.id", + "linkedTable.hidden", + ]) + }) }) From e23753ac6134c33482ddcd1802907ec3d9ab9f11 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 15:54:07 +0100 Subject: [PATCH 40/60] Add describe --- .../row/utils/tests/sqlUtils.spec.ts | 440 +++++++++--------- 1 file changed, 227 insertions(+), 213 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 767a325c7a..41cc9f6a03 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -34,224 +34,238 @@ describe("buildSqlFieldList", () => { }, } - it("extracts fields from table schema", async () => { - const result = await buildSqlFieldList(basicTable, {}) - expect(result).toEqual(["table.name", "table.description", "table.amount"]) - }) - - it("excludes hidden fields", async () => { - const table = cloneDeep(basicTable) - table.schema.description.visible = false - const result = await buildSqlFieldList(table, {}) - expect(result).toEqual(["table.name", "table.amount"]) - }) - - it("excludes non-sql fields fields", async () => { - const table = cloneDeep(basicTable) - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - table.schema.ai = { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - } - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - } - - const result = await buildSqlFieldList(table, {}) - expect(result).toEqual(["table.name", "table.description", "table.amount"]) - }) - - it("includes hidden fields if there is a formula column", async () => { - const table = cloneDeep(basicTable) - table.schema.description.visible = false - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - - const result = await buildSqlFieldList(table, {}) - expect(result).toEqual(["table.name", "table.description", "table.amount"]) - }) - - it("includes relationships fields when flag", async () => { - const table = cloneDeep(basicTable) - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - - const otherTable: Table = { - ...cloneDeep(basicTable), - name: "linkedTable", - primary: ["id"], - primaryDisplay: "name", - schema: { - ...cloneDeep(basicTable).schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } - - const result = await buildSqlFieldList(table, allTables, { - relationships: true, + describe("table", () => { + it("extracts fields from table schema", async () => { + const result = await buildSqlFieldList(basicTable, {}) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + ]) }) - expect(result).toEqual([ - "table.name", - "table.description", - "table.amount", - "linkedTable.id", - "linkedTable.name", - ]) - }) - it("includes all relationship fields if there is a formula column", async () => { - const table = cloneDeep(basicTable) - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - - const otherTable: Table = { - ...cloneDeep(basicTable), - name: "linkedTable", - schema: { - ...cloneDeep(basicTable).schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - hidden: { - name: "other", - type: FieldType.STRING, - visible: false, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - ai: { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - }, - link: { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } - - const result = await buildSqlFieldList(table, allTables, { - relationships: true, + it("excludes hidden fields", async () => { + const table = cloneDeep(basicTable) + table.schema.description.visible = false + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual(["table.name", "table.amount"]) }) - expect(result).toEqual([ - "table.name", - "table.description", - "table.amount", - "linkedTable.name", - "linkedTable.description", - "linkedTable.amount", - "linkedTable.id", - "linkedTable.hidden", - ]) - }) - it("never includes non-sql columns from relationships", async () => { - const table = cloneDeep(basicTable) - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } + it("excludes non-sql fields fields", async () => { + const table = cloneDeep(basicTable) + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + table.schema.ai = { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + } + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + } - const otherTable: Table = { - ...cloneDeep(basicTable), - name: "linkedTable", - schema: { - id: { - name: "id", - type: FieldType.NUMBER, - }, - hidden: { - name: "other", - type: FieldType.STRING, - visible: false, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - ai: { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - }, - link: { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } - - const result = await buildSqlFieldList(table, allTables, { - relationships: true, + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + ]) + }) + + it("includes hidden fields if there is a formula column", async () => { + const table = cloneDeep(basicTable) + table.schema.description.visible = false + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const result = await buildSqlFieldList(table, {}) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + ]) + }) + + it("includes relationships fields when flag", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + primary: ["id"], + primaryDisplay: "name", + schema: { + ...cloneDeep(basicTable).schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.id", + "linkedTable.name", + ]) + }) + + it("includes all relationship fields if there is a formula column", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + schema: { + ...cloneDeep(basicTable).schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + hidden: { + name: "other", + type: FieldType.STRING, + visible: false, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + ai: { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + }, + link: { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.name", + "linkedTable.description", + "linkedTable.amount", + "linkedTable.id", + "linkedTable.hidden", + ]) + }) + + it("never includes non-sql columns from relationships", async () => { + const table = cloneDeep(basicTable) + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + const otherTable: Table = { + ...cloneDeep(basicTable), + name: "linkedTable", + schema: { + id: { + name: "id", + type: FieldType.NUMBER, + }, + hidden: { + name: "other", + type: FieldType.STRING, + visible: false, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + ai: { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + }, + link: { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.id", + "linkedTable.hidden", + ]) }) - expect(result).toEqual([ - "table.name", - "table.description", - "table.amount", - "linkedTable.id", - "linkedTable.hidden", - ]) }) }) From 14da90296b4d44fa56146d0fef304a1bdee79eba Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 16:23:53 +0100 Subject: [PATCH 41/60] Regenerate table before each test --- .../row/utils/tests/sqlUtils.spec.ts | 101 +++++++++--------- 1 file changed, 49 insertions(+), 52 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 41cc9f6a03..b1be459f75 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -7,11 +7,13 @@ import { } from "@budibase/types" import { buildSqlFieldList } from "../sqlUtils" import { structures } from "../../../../routes/tests/utilities" -import { cloneDeep } from "lodash" import { sql } from "@budibase/backend-core" describe("buildSqlFieldList", () => { - const basicTable: Table = { + let table: Table & { _id: string } + + beforeEach(() => { + table = { ...structures.tableForDatasource({ type: "datasource", source: SourceName.POSTGRES, @@ -33,10 +35,11 @@ describe("buildSqlFieldList", () => { }, }, } + }) describe("table", () => { it("extracts fields from table schema", async () => { - const result = await buildSqlFieldList(basicTable, {}) + const result = await buildSqlFieldList(table, {}) expect(result).toEqual([ "table.name", "table.description", @@ -45,14 +48,12 @@ describe("buildSqlFieldList", () => { }) it("excludes hidden fields", async () => { - const table = cloneDeep(basicTable) table.schema.description.visible = false const result = await buildSqlFieldList(table, {}) expect(result).toEqual(["table.name", "table.amount"]) }) it("excludes non-sql fields fields", async () => { - const table = cloneDeep(basicTable) table.schema.formula = { name: "formula", type: FieldType.FORMULA, @@ -80,7 +81,6 @@ describe("buildSqlFieldList", () => { }) it("includes hidden fields if there is a formula column", async () => { - const table = cloneDeep(basicTable) table.schema.description.visible = false table.schema.formula = { name: "formula", @@ -97,7 +97,20 @@ describe("buildSqlFieldList", () => { }) it("includes relationships fields when flag", async () => { - const table = cloneDeep(basicTable) + const otherTable: Table = { + ...table, + name: "linkedTable", + primary: ["id"], + primaryDisplay: "name", + schema: { + ...table.schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + }, + } + table.schema.link = { name: "link", type: FieldType.LINK, @@ -106,20 +119,6 @@ describe("buildSqlFieldList", () => { tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), } - const otherTable: Table = { - ...cloneDeep(basicTable), - name: "linkedTable", - primary: ["id"], - primaryDisplay: "name", - schema: { - ...cloneDeep(basicTable).schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - }, - } - const allTables: Record = { otherTableId: otherTable, } @@ -137,25 +136,11 @@ describe("buildSqlFieldList", () => { }) it("includes all relationship fields if there is a formula column", async () => { - const table = cloneDeep(basicTable) - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - const otherTable: Table = { - ...cloneDeep(basicTable), + ...table, name: "linkedTable", schema: { - ...cloneDeep(basicTable).schema, + ...table.schema, id: { name: "id", type: FieldType.NUMBER, @@ -185,6 +170,19 @@ describe("buildSqlFieldList", () => { }, } + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + const allTables: Record = { otherTableId: otherTable, } @@ -205,22 +203,8 @@ describe("buildSqlFieldList", () => { }) it("never includes non-sql columns from relationships", async () => { - const table = cloneDeep(basicTable) - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - const otherTable: Table = { - ...cloneDeep(basicTable), + ...table, name: "linkedTable", schema: { id: { @@ -252,6 +236,19 @@ describe("buildSqlFieldList", () => { }, } + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + const allTables: Record = { otherTableId: otherTable, } From 8da96ab271c09e8c805608d095116235b4a9f840 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 16:36:20 +0100 Subject: [PATCH 42/60] Add initial view tests --- .../row/utils/tests/sqlUtils.spec.ts | 119 ++++++++++++++---- 1 file changed, 98 insertions(+), 21 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index b1be459f75..6b7f0d9ce0 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -4,37 +4,51 @@ import { RelationshipType, SourceName, Table, + ViewV2, } from "@budibase/types" import { buildSqlFieldList } from "../sqlUtils" import { structures } from "../../../../routes/tests/utilities" import { sql } from "@budibase/backend-core" +import { generator } from "@budibase/backend-core/tests" +import { generateViewID } from "../../../../../db/utils" + +import sdk from "../../../../../sdk" + +jest.mock("../../../../../sdk/app/views", () => ({ + ...jest.requireActual("../../../../../sdk/app/views"), + getTable: jest.fn(), +})) +const getTableMock = sdk.views.getTable as jest.MockedFunction< + typeof sdk.views.getTable +> describe("buildSqlFieldList", () => { let table: Table & { _id: string } beforeEach(() => { + jest.clearAllMocks() table = { - ...structures.tableForDatasource({ - type: "datasource", - source: SourceName.POSTGRES, - }), - name: "table", - _id: sql.utils.buildExternalTableId("ds_id", "table"), - schema: { - name: { - name: "name", - type: FieldType.STRING, + ...structures.tableForDatasource({ + type: "datasource", + source: SourceName.POSTGRES, + }), + name: "table", + _id: sql.utils.buildExternalTableId("ds_id", "table"), + schema: { + name: { + name: "name", + type: FieldType.STRING, + }, + description: { + name: "description", + type: FieldType.STRING, + }, + amount: { + name: "amount", + type: FieldType.NUMBER, + }, }, - description: { - name: "description", - type: FieldType.STRING, - }, - amount: { - name: "amount", - type: FieldType.NUMBER, - }, - }, - } + } }) describe("table", () => { @@ -96,7 +110,7 @@ describe("buildSqlFieldList", () => { ]) }) - it("includes relationships fields when flag", async () => { + it("includes relationships fields when flagged", async () => { const otherTable: Table = { ...table, name: "linkedTable", @@ -265,4 +279,67 @@ describe("buildSqlFieldList", () => { ]) }) }) + + describe("view", () => { + let view: ViewV2 + + beforeEach(() => { + getTableMock.mockResolvedValueOnce(table) + + view = { + version: 2, + id: generateViewID(table._id), + name: generator.word(), + tableId: table._id, + } + }) + + it("extracts fields from table schema", async () => { + view.schema = { + name: { visible: false }, + amount: { visible: true }, + } + + const result = await buildSqlFieldList(view, {}) + expect(result).toEqual(["table.amount"]) + }) + + it("includes all fields if there is a formula column", async () => { + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + view.schema = { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: true }, + } + + const result = await buildSqlFieldList(view, {}) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + ]) + }) + + it("does not includes all fields if the formula column is not included", async () => { + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + view.schema = { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + } + + const result = await buildSqlFieldList(view, {}) + expect(result).toEqual(["table.amount"]) + }) + }) }) From 460b5fd7dd5c8dc839c361d857a774009feda72a Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 16:45:08 +0100 Subject: [PATCH 43/60] More tests --- .../row/utils/tests/sqlUtils.spec.ts | 224 ++++++++++++++++++ 1 file changed, 224 insertions(+) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 6b7f0d9ce0..e1719d980b 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -341,5 +341,229 @@ describe("buildSqlFieldList", () => { const result = await buildSqlFieldList(view, {}) expect(result).toEqual(["table.amount"]) }) + + it("includes relationships fields when flagged", async () => { + const otherTable: Table = { + ...table, + name: "linkedTable", + primary: ["id"], + primaryDisplay: "name", + schema: { + ...table.schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + }, + } + + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + + view.schema = { + name: { visible: true }, + link: { visible: true }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(view, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "linkedTable.id", + "linkedTable.name", + ]) + }) + + it("includes relationships columns", async () => { + const otherTable: Table = { + ...table, + name: "linkedTable", + primary: ["id"], + schema: { + ...table.schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + }, + } + + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + + view.schema = { + name: { visible: true }, + link: { + visible: true, + columns: { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(view, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "linkedTable.id", + "linkedTable.amount", + ]) + }) + + it("does not include relationships columns for hidden links", async () => { + const otherTable: Table = { + ...table, + name: "linkedTable", + primary: ["id"], + schema: { + ...table.schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + }, + } + + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + + view.schema = { + name: { visible: true }, + link: { + visible: false, + columns: { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(view, allTables, { + relationships: true, + }) + expect(result).toEqual(["table.name"]) + }) + + it("includes all relationship fields if there is a formula column", async () => { + const otherTable: Table = { + ...table, + name: "linkedTable", + schema: { + ...table.schema, + id: { + name: "id", + type: FieldType.NUMBER, + }, + hidden: { + name: "other", + type: FieldType.STRING, + visible: false, + }, + formula: { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + }, + ai: { + name: "ai", + type: FieldType.AI, + operation: AIOperationEnum.PROMPT, + }, + link: { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: "otherTableId", + }, + }, + } + + table.schema.link = { + name: "link", + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), + } + table.schema.formula = { + name: "formula", + type: FieldType.FORMULA, + formula: "any", + } + + view.schema = { + name: { visible: true }, + formula: { visible: true }, + link: { + visible: false, + columns: { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }, + }, + } + + const allTables: Record = { + otherTableId: otherTable, + } + + const result = await buildSqlFieldList(view, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.name", + "linkedTable.description", + "linkedTable.amount", + "linkedTable.id", + "linkedTable.hidden", + ]) + }) }) }) From 9396292c4a451d83bb76002cb816f56011dfbbe0 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Wed, 18 Dec 2024 18:34:40 +0100 Subject: [PATCH 44/60] Create table test builder --- .../row/utils/tests/sqlUtils.spec.ts | 681 ++++++++---------- 1 file changed, 293 insertions(+), 388 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index e1719d980b..eafdd170f8 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -13,6 +13,8 @@ import { generator } from "@budibase/backend-core/tests" import { generateViewID } from "../../../../../db/utils" import sdk from "../../../../../sdk" +import { cloneDeep } from "lodash" +import { utils } from "@budibase/shared-core" jest.mock("../../../../../sdk/app/views", () => ({ ...jest.requireActual("../../../../../sdk/app/views"), @@ -23,36 +25,161 @@ const getTableMock = sdk.views.getTable as jest.MockedFunction< > describe("buildSqlFieldList", () => { - let table: Table & { _id: string } + let allTables: Record + + class TableConfig { + private _table: Table & { _id: string } + + constructor(name: string) { + this._table = { + ...structures.tableForDatasource({ + type: "datasource", + source: SourceName.POSTGRES, + }), + name, + _id: sql.utils.buildExternalTableId("ds_id", name), + schema: { + name: { + name: "name", + type: FieldType.STRING, + }, + description: { + name: "description", + type: FieldType.STRING, + }, + amount: { + name: "amount", + type: FieldType.NUMBER, + }, + }, + } + + allTables[name] = this._table + } + + withHiddenField(field: string) { + this._table.schema[field].visible = false + return this + } + + withField( + name: string, + type: + | FieldType.STRING + | FieldType.NUMBER + | FieldType.FORMULA + | FieldType.AI, + options?: { visible: boolean } + ) { + switch (type) { + case FieldType.NUMBER: + case FieldType.STRING: + this._table.schema[name] = { + name, + type, + ...options, + } + break + case FieldType.FORMULA: + this._table.schema[name] = { + name, + type, + formula: "any", + ...options, + } + break + case FieldType.AI: + this._table.schema[name] = { + name, + type, + operation: AIOperationEnum.PROMPT, + ...options, + } + break + default: + utils.unreachable(type) + } + return this + } + + withRelation(name: string, toTableId: string) { + this._table.schema[name] = { + name, + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: toTableId, + } + return this + } + + withPrimary(field: string) { + this._table.primary = [field] + return this + } + + withDisplay(field: string) { + this._table.primaryDisplay = field + return this + } + + create() { + return cloneDeep(this._table) + } + } + + class ViewConfig { + private _table: Table + private _view: ViewV2 + + constructor(table: Table) { + this._table = table + this._view = { + version: 2, + id: generateViewID(table._id!), + name: generator.word(), + tableId: table._id!, + } + } + + withVisible(field: string) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].visible = true + return this + } + + withHidden(field: string) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].visible = false + return this + } + + withRelationshipColumns( + field: string, + columns: Record + ) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].columns = columns + return this + } + + create() { + getTableMock.mockResolvedValueOnce(this._table) + return cloneDeep(this._view) + } + } beforeEach(() => { jest.clearAllMocks() - table = { - ...structures.tableForDatasource({ - type: "datasource", - source: SourceName.POSTGRES, - }), - name: "table", - _id: sql.utils.buildExternalTableId("ds_id", "table"), - schema: { - name: { - name: "name", - type: FieldType.STRING, - }, - description: { - name: "description", - type: FieldType.STRING, - }, - amount: { - name: "amount", - type: FieldType.NUMBER, - }, - }, - } + allTables = {} }) describe("table", () => { it("extracts fields from table schema", async () => { + const table = new TableConfig("table").create() const result = await buildSqlFieldList(table, {}) expect(result).toEqual([ "table.name", @@ -62,29 +189,19 @@ describe("buildSqlFieldList", () => { }) it("excludes hidden fields", async () => { - table.schema.description.visible = false + const table = new TableConfig("table") + .withHiddenField("description") + .create() const result = await buildSqlFieldList(table, {}) expect(result).toEqual(["table.name", "table.amount"]) }) it("excludes non-sql fields fields", async () => { - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - table.schema.ai = { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - } - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - } + const table = new TableConfig("table") + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .create() const result = await buildSqlFieldList(table, {}) expect(result).toEqual([ @@ -95,12 +212,10 @@ describe("buildSqlFieldList", () => { }) it("includes hidden fields if there is a formula column", async () => { - table.schema.description.visible = false - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } + const table = new TableConfig("table") + .withHiddenField("description") + .withField("formula", FieldType.FORMULA) + .create() const result = await buildSqlFieldList(table, {}) expect(result).toEqual([ @@ -111,31 +226,15 @@ describe("buildSqlFieldList", () => { }) it("includes relationships fields when flagged", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - primary: ["id"], - primaryDisplay: "name", - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withPrimary("id") + .withDisplay("name") + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - - const allTables: Record = { - otherTableId: otherTable, - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .create() const result = await buildSqlFieldList(table, allTables, { relationships: true, @@ -150,56 +249,42 @@ describe("buildSqlFieldList", () => { }) it("includes all relationship fields if there is a formula column", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - hidden: { - name: "other", - type: FieldType.STRING, - visible: false, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - ai: { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - }, - link: { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("hidden", FieldType.STRING, { visible: false }) + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() - const allTables: Record = { - otherTableId: otherTable, - } + const result = await buildSqlFieldList(table, allTables, { + relationships: true, + }) + expect(result).toEqual([ + "table.name", + "table.description", + "table.amount", + "linkedTable.name", + "linkedTable.description", + "linkedTable.amount", + "linkedTable.hidden", + ]) + }) + + it("never includes non-sql columns from relationships", async () => { + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withField("hidden", FieldType.STRING, { visible: false }) + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .create() + + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() const result = await buildSqlFieldList(table, allTables, { relationships: true, @@ -215,107 +300,28 @@ describe("buildSqlFieldList", () => { "linkedTable.hidden", ]) }) - - it("never includes non-sql columns from relationships", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - schema: { - id: { - name: "id", - type: FieldType.NUMBER, - }, - hidden: { - name: "other", - type: FieldType.STRING, - visible: false, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - ai: { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - }, - link: { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - }, - }, - } - - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - - const allTables: Record = { - otherTableId: otherTable, - } - - const result = await buildSqlFieldList(table, allTables, { - relationships: true, - }) - expect(result).toEqual([ - "table.name", - "table.description", - "table.amount", - "linkedTable.id", - "linkedTable.hidden", - ]) - }) }) describe("view", () => { - let view: ViewV2 - - beforeEach(() => { - getTableMock.mockResolvedValueOnce(table) - - view = { - version: 2, - id: generateViewID(table._id), - name: generator.word(), - tableId: table._id, - } - }) - it("extracts fields from table schema", async () => { - view.schema = { - name: { visible: false }, - amount: { visible: true }, - } + const view = new ViewConfig(new TableConfig("table").create()) + .withVisible("amount") + .withHidden("name") + .create() const result = await buildSqlFieldList(view, {}) expect(result).toEqual(["table.amount"]) }) it("includes all fields if there is a formula column", async () => { - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - - view.schema = { - name: { visible: false }, - amount: { visible: true }, - formula: { visible: true }, - } + const table = new TableConfig("table") + .withField("formula", FieldType.FORMULA) + .create() + const view = new ViewConfig(table) + .withHidden("name") + .withVisible("amount") + .withVisible("formula") + .create() const result = await buildSqlFieldList(view, {}) expect(result).toEqual([ @@ -326,53 +332,35 @@ describe("buildSqlFieldList", () => { }) it("does not includes all fields if the formula column is not included", async () => { - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } - - view.schema = { - name: { visible: false }, - amount: { visible: true }, - formula: { visible: false }, - } + const table = new TableConfig("table") + .withField("formula", FieldType.FORMULA) + .create() + const view = new ViewConfig(table) + .withHidden("name") + .withVisible("amount") + .withHidden("formula") + .create() const result = await buildSqlFieldList(view, {}) expect(result).toEqual(["table.amount"]) }) it("includes relationships fields when flagged", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - primary: ["id"], - primaryDisplay: "name", - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withPrimary("id") + .withDisplay("name") + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() - view.schema = { - name: { visible: true }, - link: { visible: true }, - } - - const allTables: Record = { - otherTableId: otherTable, - } + const view = new ViewConfig(table) + .withVisible("name") + .withHidden("amount") + .create() const result = await buildSqlFieldList(view, allTables, { relationships: true, @@ -385,47 +373,25 @@ describe("buildSqlFieldList", () => { }) it("includes relationships columns", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - primary: ["id"], - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withField("formula", FieldType.FORMULA) + .withPrimary("id") + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .create() - view.schema = { - name: { visible: true }, - link: { - visible: true, - columns: { - name: { visible: false }, - amount: { visible: true }, - formula: { visible: false }, - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } + const view = new ViewConfig(table) + .withVisible("name") + .withVisible("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() const result = await buildSqlFieldList(view, allTables, { relationships: true, @@ -438,47 +404,25 @@ describe("buildSqlFieldList", () => { }) it("does not include relationships columns for hidden links", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - primary: ["id"], - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withField("formula", FieldType.FORMULA) + .withPrimary("id") + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .create() - view.schema = { - name: { visible: true }, - link: { - visible: false, - columns: { - name: { visible: false }, - amount: { visible: true }, - formula: { visible: false }, - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } + const view = new ViewConfig(table) + .withVisible("name") + .withHidden("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() const result = await buildSqlFieldList(view, allTables, { relationships: true, @@ -487,69 +431,30 @@ describe("buildSqlFieldList", () => { }) it("includes all relationship fields if there is a formula column", async () => { - const otherTable: Table = { - ...table, - name: "linkedTable", - schema: { - ...table.schema, - id: { - name: "id", - type: FieldType.NUMBER, - }, - hidden: { - name: "other", - type: FieldType.STRING, - visible: false, - }, - formula: { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - }, - ai: { - name: "ai", - type: FieldType.AI, - operation: AIOperationEnum.PROMPT, - }, - link: { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: "otherTableId", - }, - }, - } + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withField("hidden", FieldType.STRING, { visible: false }) + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .withPrimary("id") + .create() - table.schema.link = { - name: "link", - type: FieldType.LINK, - relationshipType: RelationshipType.ONE_TO_MANY, - fieldName: "link", - tableId: sql.utils.buildExternalTableId("ds_id", "otherTableId"), - } - table.schema.formula = { - name: "formula", - type: FieldType.FORMULA, - formula: "any", - } + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() - view.schema = { - name: { visible: true }, - formula: { visible: true }, - link: { - visible: false, - columns: { - name: { visible: false }, - amount: { visible: true }, - formula: { visible: false }, - }, - }, - } - - const allTables: Record = { - otherTableId: otherTable, - } + const view = new ViewConfig(table) + .withVisible("name") + .withVisible("formula") + .withHidden("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() const result = await buildSqlFieldList(view, allTables, { relationships: true, From 92e791f9a715235856c2f318965329d713c7b893 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 12:48:33 +0100 Subject: [PATCH 45/60] Add calculation tests --- .../row/utils/tests/sqlUtils.spec.ts | 41 +++++++++++++++++++ 1 file changed, 41 insertions(+) diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index eafdd170f8..12a5edd30b 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -1,10 +1,12 @@ import { AIOperationEnum, + CalculationType, FieldType, RelationshipType, SourceName, Table, ViewV2, + ViewV2Type, } from "@budibase/types" import { buildSqlFieldList } from "../sqlUtils" import { structures } from "../../../../routes/tests/utilities" @@ -166,6 +168,21 @@ describe("buildSqlFieldList", () => { return this } + withCalculation( + name: string, + field: string, + calculationType: CalculationType + ) { + this._view.type = ViewV2Type.CALCULATION + this._view.schema ??= {} + this._view.schema[name] = { + field, + calculationType, + visible: true, + } + return this + } + create() { getTableMock.mockResolvedValueOnce(this._table) return cloneDeep(this._view) @@ -471,4 +488,28 @@ describe("buildSqlFieldList", () => { ]) }) }) + + describe("calculation view", () => { + it("does not include calculation fields", async () => { + const view = new ViewConfig(new TableConfig("table").create()) + .withCalculation("average", "amount", CalculationType.AVG) + + .create() + + const result = await buildSqlFieldList(view, {}) + expect(result).toEqual([]) + }) + + it("includes visible fields calculation fields", async () => { + const view = new ViewConfig(new TableConfig("table").create()) + .withCalculation("average", "amount", CalculationType.AVG) + .withHidden("name") + .withVisible("amount") + + .create() + + const result = await buildSqlFieldList(view, {}) + expect(result).toEqual(["table.amount"]) + }) + }) }) From 889197679e1595b78bff806292372997000cedc2 Mon Sep 17 00:00:00 2001 From: mike12345567 Date: Thu, 19 Dec 2024 12:39:00 +0000 Subject: [PATCH 46/60] Comment to explain the behaviour of the junction document select. --- packages/server/src/sdk/app/rows/search/internal/sqs.ts | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index c92d186a37..449ba477b7 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -121,6 +121,12 @@ async function buildInternalFieldList( if (!relatedTable) { continue } + // a quirk of how junction documents work in Budibase, refer to the "LinkDocument" type to see the full + // structure - essentially all relationships between two tables will be inserted into a single "table" + // we don't use an independent junction table ID for each separate relationship between two tables. For + // example if we have table A and B, with two relationships between them, all the junction documents will + // end up in the same junction table ID. We need to retrieve the field name property of the junction documents + // as part of the relationship to tell us which relationship column the junction is related to. const relatedFields = ( await buildInternalFieldList(relatedTable, tables) ).concat( From 0ee432dd9a34e478705a0c74a0535965fd33cd37 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 13:32:47 +0100 Subject: [PATCH 47/60] Always use tableid prefix for sqs --- packages/server/src/sdk/app/rows/search/internal/sqs.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index 449ba477b7..5b03887d36 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -142,7 +142,7 @@ async function buildInternalFieldList( if (!isView || !helpers.views.isCalculationView(source)) { for (const field of PROTECTED_INTERNAL_COLUMNS) { - fieldList.push(field) + fieldList.push(`${table._id}.${field}`) } } From 5a9ed4ff52a9c016b65df931f542f08569820948 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 13:48:10 +0100 Subject: [PATCH 48/60] Include hidden fields for formulas --- .../src/sdk/app/rows/search/internal/sqs.ts | 23 +++++++++++++++---- 1 file changed, 18 insertions(+), 5 deletions(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index 5b03887d36..68d9d3420f 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -55,12 +55,16 @@ const MISSING_COLUMN_REGEX = new RegExp(`no such column: .+`) const MISSING_TABLE_REGX = new RegExp(`no such table: .+`) const DUPLICATE_COLUMN_REGEX = new RegExp(`duplicate column name: .+`) -async function buildInternalFieldList( +export async function buildInternalFieldList( source: Table | ViewV2, tables: Table[], - opts?: { relationships?: RelationshipsJson[]; allowedFields?: string[] } + opts?: { + relationships?: RelationshipsJson[] + allowedFields?: string[] + includeHiddenFields?: boolean + } ) { - const { relationships, allowedFields } = opts || {} + const { relationships, allowedFields, includeHiddenFields } = opts || {} let schemaFields: string[] = [] const isView = sdk.views.isView(source) @@ -75,7 +79,7 @@ async function buildInternalFieldList( schemaFields = Object.keys(helpers.views.basicFields(source)) } else { schemaFields = Object.keys(source.schema).filter( - key => source.schema[key].visible !== false + key => includeHiddenFields || source.schema[key].visible !== false ) } @@ -109,10 +113,16 @@ async function buildInternalFieldList( } for (let key of schemaFields) { const col = table.schema[key] + + if ([FieldType.FORMULA, FieldType.AI].includes(col.type)) { + continue + } + const isRelationship = col.type === FieldType.LINK if (!relationships && isRelationship) { continue } + if (!isRelationship) { fieldList.push(`${table._id}.${mapToUserColumn(key)}`) } else { @@ -121,6 +131,7 @@ async function buildInternalFieldList( if (!relatedTable) { continue } + // a quirk of how junction documents work in Budibase, refer to the "LinkDocument" type to see the full // structure - essentially all relationships between two tables will be inserted into a single "table" // we don't use an independent junction table ID for each separate relationship between two tables. For @@ -128,7 +139,9 @@ async function buildInternalFieldList( // end up in the same junction table ID. We need to retrieve the field name property of the junction documents // as part of the relationship to tell us which relationship column the junction is related to. const relatedFields = ( - await buildInternalFieldList(relatedTable, tables) + await buildInternalFieldList(relatedTable, tables, { + includeHiddenFields: containsFormula, + }) ).concat( getJunctionFields(relatedTable, ["doc1.fieldName", "doc2.fieldName"]) ) From 7cc03b172486c32b98144398a1940e7559dba6bf Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 14:31:45 +0100 Subject: [PATCH 49/60] Create sql tests for sqs --- .../app/rows/search/internal/test/sqs.spec.ts | 674 ++++++++++++++++++ 1 file changed, 674 insertions(+) create mode 100644 packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts diff --git a/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts b/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts new file mode 100644 index 0000000000..31c25defad --- /dev/null +++ b/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts @@ -0,0 +1,674 @@ +import { + AIOperationEnum, + CalculationType, + FieldType, + RelationshipType, + SourceName, + Table, + ViewV2, + ViewV2Type, +} from "@budibase/types" +import { buildInternalFieldList } from "../sqs" +import { structures } from "../../../../../../api/routes/tests/utilities" +import { sql } from "@budibase/backend-core" +import { generator } from "@budibase/backend-core/tests" +import { + generateJunctionTableID, + generateViewID, +} from "../../../../../../db/utils" + +import sdk from "../../../../../../sdk" +import { cloneDeep } from "lodash" +import { utils } from "@budibase/shared-core" + +jest.mock("../../../../../../sdk/app/views", () => ({ + ...jest.requireActual("../../../../../../sdk/app/views"), + getTable: jest.fn(), +})) +const getTableMock = sdk.views.getTable as jest.MockedFunction< + typeof sdk.views.getTable +> + +describe("buildInternalFieldList", () => { + let allTables: Table[] + + class TableConfig { + private _table: Table & { _id: string } + + constructor() { + const name = generator.word() + this._table = { + ...structures.tableForDatasource({ + type: "datasource", + source: SourceName.POSTGRES, + }), + name, + _id: sql.utils.buildExternalTableId("ds_id", name), + schema: { + name: { + name: "name", + type: FieldType.STRING, + }, + description: { + name: "description", + type: FieldType.STRING, + }, + amount: { + name: "amount", + type: FieldType.NUMBER, + }, + }, + } + + allTables.push(this._table) + } + + withHiddenField(field: string) { + this._table.schema[field].visible = false + return this + } + + withField( + name: string, + type: + | FieldType.STRING + | FieldType.NUMBER + | FieldType.FORMULA + | FieldType.AI, + options?: { visible: boolean } + ) { + switch (type) { + case FieldType.NUMBER: + case FieldType.STRING: + this._table.schema[name] = { + name, + type, + ...options, + } + break + case FieldType.FORMULA: + this._table.schema[name] = { + name, + type, + formula: "any", + ...options, + } + break + case FieldType.AI: + this._table.schema[name] = { + name, + type, + operation: AIOperationEnum.PROMPT, + ...options, + } + break + default: + utils.unreachable(type) + } + return this + } + + withRelation(name: string, toTableId: string) { + this._table.schema[name] = { + name, + type: FieldType.LINK, + relationshipType: RelationshipType.ONE_TO_MANY, + fieldName: "link", + tableId: toTableId, + } + return this + } + + withEmptySchema() { + this._table.schema = {} + return this + } + + create() { + return cloneDeep(this._table) + } + } + + class ViewConfig { + private _table: Table + private _view: ViewV2 + + constructor(table: Table) { + this._table = table + this._view = { + version: 2, + id: generateViewID(table._id!), + name: generator.word(), + tableId: table._id!, + } + } + + withVisible(field: string) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].visible = true + return this + } + + withHidden(field: string) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].visible = false + return this + } + + withRelationshipColumns( + field: string, + columns: Record + ) { + this._view.schema ??= {} + this._view.schema[field] ??= {} + this._view.schema[field].columns = columns + return this + } + + withCalculation( + name: string, + field: string, + calculationType: CalculationType + ) { + this._view.type = ViewV2Type.CALCULATION + this._view.schema ??= {} + this._view.schema[name] = { + field, + calculationType, + visible: true, + } + return this + } + + create() { + getTableMock.mockResolvedValueOnce(this._table) + return cloneDeep(this._view) + } + } + + beforeEach(() => { + jest.clearAllMocks() + allTables = [] + }) + + describe("table", () => { + it("includes internal columns by default", async () => { + const table = new TableConfig().withEmptySchema().create() + const result = await buildInternalFieldList(table, []) + expect(result).toEqual([ + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("extracts fields from table schema", async () => { + const table = new TableConfig().create() + const result = await buildInternalFieldList(table, []) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("excludes hidden fields", async () => { + const table = new TableConfig().withHiddenField("description").create() + const result = await buildInternalFieldList(table, []) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_amount`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("excludes non-sql fields fields", async () => { + const table = new TableConfig() + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .create() + + const result = await buildInternalFieldList(table, []) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("includes hidden fields if there is a formula column", async () => { + const table = new TableConfig() + .withHiddenField("description") + .withField("formula", FieldType.FORMULA) + .create() + + const result = await buildInternalFieldList(table, []) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("includes relationships fields when flagged", async () => { + const otherTable = new TableConfig() + .withHiddenField("description") + .create() + + const table = new TableConfig() + .withHiddenField("amount") + .withRelation("link", otherTable._id) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + + const result = await buildInternalFieldList(table, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_amount`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("includes all relationship fields if there is a formula column", async () => { + const otherTable = new TableConfig() + .withField("hidden", FieldType.STRING, { visible: false }) + .create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .withHiddenField("description") + .withField("formula", FieldType.FORMULA) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(table, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_description`, + `${otherTable._id}.data_amount`, + `${otherTable._id}.data_hidden`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("never includes non-sql columns from relationships", async () => { + const otherTable = new TableConfig() + .withField("hidden", FieldType.STRING, { visible: false }) + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(table, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_description`, + `${otherTable._id}.data_amount`, + `${otherTable._id}.data_hidden`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + }) + + describe("view", () => { + it("includes internal columns by default", async () => { + const view = new ViewConfig(new TableConfig().create()).create() + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([ + `${view.tableId}._id`, + `${view.tableId}._rev`, + `${view.tableId}.type`, + `${view.tableId}.createdAt`, + `${view.tableId}.updatedAt`, + `${view.tableId}.tableId`, + ]) + }) + + it("extracts fields from table schema", async () => { + const view = new ViewConfig(new TableConfig().create()) + .withVisible("amount") + .withHidden("name") + .create() + + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([ + `${view.tableId}.data_amount`, + `${view.tableId}._id`, + `${view.tableId}._rev`, + `${view.tableId}.type`, + `${view.tableId}.createdAt`, + `${view.tableId}.updatedAt`, + `${view.tableId}.tableId`, + ]) + }) + + it("includes all fields if there is a formula column", async () => { + const table = new TableConfig() + .withField("formula", FieldType.FORMULA) + .create() + const view = new ViewConfig(table) + .withHidden("name") + .withVisible("amount") + .withVisible("formula") + .create() + + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([ + `${view.tableId}.data_name`, + `${view.tableId}.data_description`, + `${view.tableId}.data_amount`, + `${view.tableId}._id`, + `${view.tableId}._rev`, + `${view.tableId}.type`, + `${view.tableId}.createdAt`, + `${view.tableId}.updatedAt`, + `${view.tableId}.tableId`, + ]) + }) + + it("does not includes all fields if the formula column is not included", async () => { + const table = new TableConfig() + .withField("formula", FieldType.FORMULA) + .create() + const view = new ViewConfig(table) + .withHidden("name") + .withVisible("amount") + .withHidden("formula") + .create() + + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([ + `${view.tableId}.data_amount`, + `${view.tableId}._id`, + `${view.tableId}._rev`, + `${view.tableId}.type`, + `${view.tableId}.createdAt`, + `${view.tableId}.updatedAt`, + `${view.tableId}.tableId`, + ]) + }) + + it("includes relationships fields", async () => { + const otherTable = new TableConfig().create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() + + const view = new ViewConfig(table) + .withVisible("name") + .withVisible("link") + .withHidden("amount") + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(view, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_description`, + `${otherTable._id}.data_amount`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("includes relationships columns", async () => { + const otherTable = new TableConfig() + .withField("formula", FieldType.FORMULA) + .create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .create() + + const view = new ViewConfig(table) + .withVisible("name") + .withVisible("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(view, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_description`, + `${otherTable._id}.data_amount`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("does not include relationships columns for hidden links", async () => { + const otherTable = new TableConfig() + .withField("formula", FieldType.FORMULA) + .create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .create() + + const view = new ViewConfig(table) + .withVisible("name") + .withHidden("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(view, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + + it("includes all relationship fields if there is a formula column", async () => { + const otherTable = new TableConfig() + .withField("hidden", FieldType.STRING, { visible: false }) + .withField("formula", FieldType.FORMULA) + .withField("ai", FieldType.AI) + .withRelation("link", "otherTableId") + .create() + + const table = new TableConfig() + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() + + const view = new ViewConfig(table) + .withVisible("name") + .withVisible("formula") + .withHidden("link") + .withRelationshipColumns("link", { + name: { visible: false }, + amount: { visible: true }, + formula: { visible: false }, + }) + .create() + + const relationships = [{ tableName: otherTable.name, column: "link" }] + const result = await buildInternalFieldList(view, allTables, { + relationships, + }) + expect(result).toEqual([ + `${table._id}.data_name`, + `${table._id}.data_description`, + `${table._id}.data_amount`, + `${otherTable._id}.data_name`, + `${otherTable._id}.data_description`, + `${otherTable._id}.data_amount`, + `${otherTable._id}.data_hidden`, + `${otherTable._id}._id`, + `${otherTable._id}._rev`, + `${otherTable._id}.type`, + `${otherTable._id}.createdAt`, + `${otherTable._id}.updatedAt`, + `${otherTable._id}.tableId`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, + `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}._id`, + `${table._id}._rev`, + `${table._id}.type`, + `${table._id}.createdAt`, + `${table._id}.updatedAt`, + `${table._id}.tableId`, + ]) + }) + }) + + describe("calculation view", () => { + it("does not include calculation fields", async () => { + const view = new ViewConfig(new TableConfig().create()) + .withCalculation("average", "amount", CalculationType.AVG) + .create() + + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([]) + }) + + it("includes visible fields calculation fields", async () => { + const view = new ViewConfig(new TableConfig().create()) + .withCalculation("average", "amount", CalculationType.AVG) + .withHidden("name") + .withVisible("amount") + .create() + + const result = await buildInternalFieldList(view, []) + expect(result).toEqual([`${view.tableId}.data_amount`]) + }) + }) +}) From 16fb86549b7cb47148cb92f695d2e6be6449905c Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 14:42:49 +0100 Subject: [PATCH 50/60] Fix tests --- .../src/api/controllers/row/utils/sqlUtils.ts | 4 +- .../row/utils/tests/sqlUtils.spec.ts | 50 +++++++++---------- 2 files changed, 25 insertions(+), 29 deletions(-) diff --git a/packages/server/src/api/controllers/row/utils/sqlUtils.ts b/packages/server/src/api/controllers/row/utils/sqlUtils.ts index e66a0b5bf6..1793d64b89 100644 --- a/packages/server/src/api/controllers/row/utils/sqlUtils.ts +++ b/packages/server/src/api/controllers/row/utils/sqlUtils.ts @@ -200,8 +200,8 @@ export async function buildSqlFieldList( if ( isView && - source.schema?.[field.name] && - !helpers.views.isVisible(source.schema[field.name]) && + (!source.schema?.[field.name] || + !helpers.views.isVisible(source.schema[field.name])) && !containsFormula ) { continue diff --git a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts index 12a5edd30b..365f571fcf 100644 --- a/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts +++ b/packages/server/src/api/controllers/row/utils/tests/sqlUtils.spec.ts @@ -362,33 +362,6 @@ describe("buildSqlFieldList", () => { expect(result).toEqual(["table.amount"]) }) - it("includes relationships fields when flagged", async () => { - const otherTable = new TableConfig("linkedTable") - .withField("id", FieldType.NUMBER) - .withPrimary("id") - .withDisplay("name") - .create() - - const table = new TableConfig("table") - .withRelation("link", otherTable._id) - .withField("formula", FieldType.FORMULA) - .create() - - const view = new ViewConfig(table) - .withVisible("name") - .withHidden("amount") - .create() - - const result = await buildSqlFieldList(view, allTables, { - relationships: true, - }) - expect(result).toEqual([ - "table.name", - "linkedTable.id", - "linkedTable.name", - ]) - }) - it("includes relationships columns", async () => { const otherTable = new TableConfig("linkedTable") .withField("id", FieldType.NUMBER) @@ -420,6 +393,29 @@ describe("buildSqlFieldList", () => { ]) }) + it("excludes relationships fields when view is not included in the view", async () => { + const otherTable = new TableConfig("linkedTable") + .withField("id", FieldType.NUMBER) + .withPrimary("id") + .withDisplay("name") + .create() + + const table = new TableConfig("table") + .withRelation("link", otherTable._id) + .withField("formula", FieldType.FORMULA) + .create() + + const view = new ViewConfig(table) + .withVisible("name") + .withHidden("amount") + .create() + + const result = await buildSqlFieldList(view, allTables, { + relationships: true, + }) + expect(result).toEqual(["table.name"]) + }) + it("does not include relationships columns for hidden links", async () => { const otherTable = new TableConfig("linkedTable") .withField("id", FieldType.NUMBER) From 8905f9dd5d5cc6ad174eb6d3a4ba7420b19bad2c Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Thu, 19 Dec 2024 15:27:18 +0100 Subject: [PATCH 51/60] Fix tests --- .../src/sdk/app/rows/search/internal/sqs.ts | 4 -- .../app/rows/search/internal/test/sqs.spec.ts | 70 ++----------------- 2 files changed, 7 insertions(+), 67 deletions(-) diff --git a/packages/server/src/sdk/app/rows/search/internal/sqs.ts b/packages/server/src/sdk/app/rows/search/internal/sqs.ts index 68d9d3420f..84162a67af 100644 --- a/packages/server/src/sdk/app/rows/search/internal/sqs.ts +++ b/packages/server/src/sdk/app/rows/search/internal/sqs.ts @@ -114,10 +114,6 @@ export async function buildInternalFieldList( for (let key of schemaFields) { const col = table.schema[key] - if ([FieldType.FORMULA, FieldType.AI].includes(col.type)) { - continue - } - const isRelationship = col.type === FieldType.LINK if (!relationships && isRelationship) { continue diff --git a/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts b/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts index 31c25defad..91674089dc 100644 --- a/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts +++ b/packages/server/src/sdk/app/rows/search/internal/test/sqs.spec.ts @@ -238,27 +238,6 @@ describe("buildInternalFieldList", () => { ]) }) - it("excludes non-sql fields fields", async () => { - const table = new TableConfig() - .withField("formula", FieldType.FORMULA) - .withField("ai", FieldType.AI) - .withRelation("link", "otherTableId") - .create() - - const result = await buildInternalFieldList(table, []) - expect(result).toEqual([ - `${table._id}.data_name`, - `${table._id}.data_description`, - `${table._id}.data_amount`, - `${table._id}._id`, - `${table._id}._rev`, - `${table._id}.type`, - `${table._id}.createdAt`, - `${table._id}.updatedAt`, - `${table._id}.tableId`, - ]) - }) - it("includes hidden fields if there is a formula column", async () => { const table = new TableConfig() .withHiddenField("description") @@ -270,6 +249,7 @@ describe("buildInternalFieldList", () => { `${table._id}.data_name`, `${table._id}.data_description`, `${table._id}.data_amount`, + `${table._id}.data_formula`, `${table._id}._id`, `${table._id}._rev`, `${table._id}.type`, @@ -347,48 +327,7 @@ describe("buildInternalFieldList", () => { `${otherTable._id}.tableId`, `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, - `${table._id}._id`, - `${table._id}._rev`, - `${table._id}.type`, - `${table._id}.createdAt`, - `${table._id}.updatedAt`, - `${table._id}.tableId`, - ]) - }) - - it("never includes non-sql columns from relationships", async () => { - const otherTable = new TableConfig() - .withField("hidden", FieldType.STRING, { visible: false }) - .withField("formula", FieldType.FORMULA) - .withField("ai", FieldType.AI) - .withRelation("link", "otherTableId") - .create() - - const table = new TableConfig() - .withRelation("link", otherTable._id) - .withField("formula", FieldType.FORMULA) - .create() - - const relationships = [{ tableName: otherTable.name, column: "link" }] - const result = await buildInternalFieldList(table, allTables, { - relationships, - }) - expect(result).toEqual([ - `${table._id}.data_name`, - `${table._id}.data_description`, - `${table._id}.data_amount`, - `${otherTable._id}.data_name`, - `${otherTable._id}.data_description`, - `${otherTable._id}.data_amount`, - `${otherTable._id}.data_hidden`, - `${otherTable._id}._id`, - `${otherTable._id}._rev`, - `${otherTable._id}.type`, - `${otherTable._id}.createdAt`, - `${otherTable._id}.updatedAt`, - `${otherTable._id}.tableId`, - `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, - `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}.data_formula`, `${table._id}._id`, `${table._id}._rev`, `${table._id}.type`, @@ -446,6 +385,7 @@ describe("buildInternalFieldList", () => { `${view.tableId}.data_name`, `${view.tableId}.data_description`, `${view.tableId}.data_amount`, + `${view.tableId}.data_formula`, `${view.tableId}._id`, `${view.tableId}._rev`, `${view.tableId}.type`, @@ -545,6 +485,7 @@ describe("buildInternalFieldList", () => { `${otherTable._id}.data_name`, `${otherTable._id}.data_description`, `${otherTable._id}.data_amount`, + `${otherTable._id}.data_formula`, `${otherTable._id}._id`, `${otherTable._id}._rev`, `${otherTable._id}.type`, @@ -632,6 +573,8 @@ describe("buildInternalFieldList", () => { `${otherTable._id}.data_description`, `${otherTable._id}.data_amount`, `${otherTable._id}.data_hidden`, + `${otherTable._id}.data_formula`, + `${otherTable._id}.data_ai`, `${otherTable._id}._id`, `${otherTable._id}._rev`, `${otherTable._id}.type`, @@ -640,6 +583,7 @@ describe("buildInternalFieldList", () => { `${otherTable._id}.tableId`, `${generateJunctionTableID(table._id, otherTable._id)}.doc1.fieldName`, `${generateJunctionTableID(table._id, otherTable._id)}.doc2.fieldName`, + `${table._id}.data_formula`, `${table._id}._id`, `${table._id}._rev`, `${table._id}.type`, From 52916f11a8129cd8dec61dcd73fc30c2f894f741 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Tue, 7 Jan 2025 14:06:03 +0000 Subject: [PATCH 52/60] Convert portal user store to TS --- .../_components/BuilderSidePanel.svelte | 7 +- .../src/stores/portal/{users.js => users.ts} | 144 ++++++++++-------- packages/frontend-core/src/api/user.ts | 2 +- packages/types/src/api/web/user.ts | 2 +- 4 files changed, 85 insertions(+), 70 deletions(-) rename packages/builder/src/stores/portal/{users.js => users.ts} (55%) diff --git a/packages/builder/src/pages/builder/app/[application]/_components/BuilderSidePanel.svelte b/packages/builder/src/pages/builder/app/[application]/_components/BuilderSidePanel.svelte index 37abd7f1eb..2260892913 100644 --- a/packages/builder/src/pages/builder/app/[application]/_components/BuilderSidePanel.svelte +++ b/packages/builder/src/pages/builder/app/[application]/_components/BuilderSidePanel.svelte @@ -442,13 +442,11 @@ const onUpdateUserInvite = async (invite, role) => { let updateBody = { - code: invite.code, apps: { ...invite.apps, [prodAppId]: role, }, } - if (role === Constants.Roles.CREATOR) { updateBody.builder = updateBody.builder || {} updateBody.builder.apps = [...(updateBody.builder.apps ?? []), prodAppId] @@ -456,7 +454,7 @@ } else if (role !== Constants.Roles.CREATOR && invite?.builder?.apps) { invite.builder.apps = [] } - await users.updateInvite(updateBody) + await users.updateInvite(invite.code, updateBody) await filterInvites(query) } @@ -470,8 +468,7 @@ let updated = { ...invite } delete updated.info.apps[prodAppId] - return await users.updateInvite({ - code: updated.code, + return await users.updateInvite(updated.code, { apps: updated.apps, }) } diff --git a/packages/builder/src/stores/portal/users.js b/packages/builder/src/stores/portal/users.ts similarity index 55% rename from packages/builder/src/stores/portal/users.js rename to packages/builder/src/stores/portal/users.ts index 99ead22317..7c0bec296e 100644 --- a/packages/builder/src/stores/portal/users.js +++ b/packages/builder/src/stores/portal/users.ts @@ -1,41 +1,68 @@ -import { writable } from "svelte/store" import { API } from "@/api" -import { update } from "lodash" import { licensing } from "." import { sdk } from "@budibase/shared-core" import { Constants } from "@budibase/frontend-core" +import { + DeleteInviteUsersRequest, + InviteUsersRequest, + SearchUsersRequest, + SearchUsersResponse, + UpdateInviteRequest, + User, +} from "@budibase/types" +import { BudiStore } from "../BudiStore" -export function createUsersStore() { - const { subscribe, set } = writable({}) +type UserState = SearchUsersResponse & SearchUsersRequest - // opts can contain page and search params - async function search(opts = {}) { +class UserStore extends BudiStore { + constructor() { + super({ + data: [], + }) + + // Update quotas after any add or remove operation + this.create = this.refreshUsage(this.create) + this.save = this.refreshUsage(this.save) + this.delete = this.refreshUsage(this.delete) + this.bulkDelete = this.refreshUsage(this.bulkDelete) + } + + async search(opts: SearchUsersRequest = {}) { const paged = await API.searchUsers(opts) - set({ + this.set({ ...paged, ...opts, }) return paged } - async function get(userId) { + async get(userId: string) { try { return await API.getUser(userId) } catch (err) { return null } } - const fetch = async () => { + + async fetch() { return await API.getUsers() } - // One or more users. - async function onboard(payload) { + async onboard(payload: InviteUsersRequest) { return await API.onboardUsers(payload) } - async function invite(payload) { - const users = payload.map(user => { + async invite( + payload: { + admin?: boolean + builder?: boolean + creator?: boolean + email: string + apps?: any[] + groups?: any[] + }[] + ) { + const users: InviteUsersRequest = payload.map(user => { let builder = undefined if (user.admin || user.builder) { builder = { global: true } @@ -55,11 +82,16 @@ export function createUsersStore() { return API.inviteUsers(users) } - async function removeInvites(payload) { + async removeInvites(payload: DeleteInviteUsersRequest) { return API.removeUserInvites(payload) } - async function acceptInvite(inviteCode, password, firstName, lastName) { + async acceptInvite( + inviteCode: string, + password: string, + firstName: string, + lastName?: string + ) { return API.acceptInvite({ inviteCode, password, @@ -68,21 +100,25 @@ export function createUsersStore() { }) } - async function fetchInvite(inviteCode) { + async fetchInvite(inviteCode: string) { return API.getUserInvite(inviteCode) } - async function getInvites() { + async getInvites() { return API.getUserInvites() } - async function updateInvite(invite) { - return API.updateUserInvite(invite.code, invite) + async updateInvite(code: string, invite: UpdateInviteRequest) { + return API.updateUserInvite(code, invite) } - async function create(data) { - let mappedUsers = data.users.map(user => { - const body = { + async getUserCountByApp(appId: string) { + return await API.getUserCountByApp(appId) + } + + async create(data: any) { + let mappedUsers: Omit[] = data.users.map((user: any) => { + const body: Omit = { email: user.email, password: user.password, roles: {}, @@ -113,41 +149,44 @@ export function createUsersStore() { const response = await API.createUsers(mappedUsers, data.groups) // re-search from first page - await search() + await this.search() return response } - async function del(id) { + async delete(id: string) { await API.deleteUser(id) - update(users => users.filter(user => user._id !== id)) } - async function getUserCountByApp(appId) { - return await API.getUserCountByApp(appId) - } - - async function bulkDelete(users) { + async bulkDelete( + users: Array<{ + userId: string + email: string + }> + ) { return API.deleteUsers(users) } - async function save(user) { + async save(user: User) { return await API.saveUser(user) } - async function addAppBuilder(userId, appId) { + async addAppBuilder(userId: string, appId: string) { return await API.addAppBuilder(userId, appId) } - async function removeAppBuilder(userId, appId) { + async removeAppBuilder(userId: string, appId: string) { return await API.removeAppBuilder(userId, appId) } - async function getAccountHolder() { + async getAccountHolder() { return await API.getAccountHolder() } - const getUserRole = user => { - if (user && user.email === user.tenantOwnerEmail) { + getUserRole(user?: User & { tenantOwnerEmail?: string }) { + if (!user) { + return Constants.BudibaseRoles.AppUser + } + if (user.email === user.tenantOwnerEmail) { return Constants.BudibaseRoles.Owner } else if (sdk.users.isAdmin(user)) { return Constants.BudibaseRoles.Admin @@ -160,37 +199,16 @@ export function createUsersStore() { } } - const refreshUsage = - fn => - async (...args) => { + foo = this.refreshUsage(this.create) + bar = this.refreshUsage(this.save) + + refreshUsage(fn: (...args: T) => Promise) { + return async function (...args: T) { const response = await fn(...args) await licensing.setQuotaUsage() return response } - - return { - subscribe, - search, - get, - getUserRole, - fetch, - invite, - onboard, - fetchInvite, - getInvites, - removeInvites, - updateInvite, - getUserCountByApp, - addAppBuilder, - removeAppBuilder, - // any operation that adds or deletes users - acceptInvite, - create: refreshUsage(create), - save: refreshUsage(save), - bulkDelete: refreshUsage(bulkDelete), - delete: refreshUsage(del), - getAccountHolder, } } -export const users = createUsersStore() +export const users = new UserStore() diff --git a/packages/frontend-core/src/api/user.ts b/packages/frontend-core/src/api/user.ts index 84ec68644d..7464b1ec4a 100644 --- a/packages/frontend-core/src/api/user.ts +++ b/packages/frontend-core/src/api/user.ts @@ -60,7 +60,7 @@ export interface UserEndpoints { getAccountHolder: () => Promise searchUsers: (data: SearchUsersRequest) => Promise createUsers: ( - users: User[], + users: Omit[], groups: any[] ) => Promise updateUserInvite: ( diff --git a/packages/types/src/api/web/user.ts b/packages/types/src/api/web/user.ts index a42449d550..8b0dfef34b 100644 --- a/packages/types/src/api/web/user.ts +++ b/packages/types/src/api/web/user.ts @@ -124,7 +124,7 @@ export interface AcceptUserInviteRequest { inviteCode: string password: string firstName: string - lastName: string + lastName?: string } export interface AcceptUserInviteResponse { From 6bd4cb47c204d1c2626c0d8d93c71b2b695679a4 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Wed, 8 Jan 2025 15:54:09 +0000 Subject: [PATCH 53/60] Add new UnsavedUser type and update controllers --- packages/builder/src/stores/portal/users.ts | 35 ++++++++++--------- packages/frontend-core/src/api/user.ts | 14 +++----- packages/types/src/api/web/user.ts | 4 ++- .../src/api/controllers/global/users.ts | 16 ++++++--- 4 files changed, 38 insertions(+), 31 deletions(-) diff --git a/packages/builder/src/stores/portal/users.ts b/packages/builder/src/stores/portal/users.ts index 7c0bec296e..9284ad2992 100644 --- a/packages/builder/src/stores/portal/users.ts +++ b/packages/builder/src/stores/portal/users.ts @@ -9,9 +9,18 @@ import { SearchUsersResponse, UpdateInviteRequest, User, + UserIdentifier, + UnsavedUser, } from "@budibase/types" import { BudiStore } from "../BudiStore" +interface UserInfo { + email: string + password: string + forceResetPassword?: boolean + role: keyof typeof Constants.BudibaseRoles +} + type UserState = SearchUsersResponse & SearchUsersRequest class UserStore extends BudiStore { @@ -116,9 +125,9 @@ class UserStore extends BudiStore { return await API.getUserCountByApp(appId) } - async create(data: any) { - let mappedUsers: Omit[] = data.users.map((user: any) => { - const body: Omit = { + async create(data: { users: UserInfo[]; groups: any[] }) { + let mappedUsers: UnsavedUser[] = data.users.map((user: any) => { + const body: UnsavedUser = { email: user.email, password: user.password, roles: {}, @@ -128,17 +137,17 @@ class UserStore extends BudiStore { } switch (user.role) { - case "appUser": + case Constants.BudibaseRoles.AppUser: body.builder = { global: false } body.admin = { global: false } break - case "developer": + case Constants.BudibaseRoles.Developer: body.builder = { global: true } break - case "creator": + case Constants.BudibaseRoles.Creator: body.builder = { creator: true, global: false } break - case "admin": + case Constants.BudibaseRoles.Admin: body.admin = { global: true } body.builder = { global: true } break @@ -157,12 +166,7 @@ class UserStore extends BudiStore { await API.deleteUser(id) } - async bulkDelete( - users: Array<{ - userId: string - email: string - }> - ) { + async bulkDelete(users: UserIdentifier[]) { return API.deleteUsers(users) } @@ -199,9 +203,8 @@ class UserStore extends BudiStore { } } - foo = this.refreshUsage(this.create) - bar = this.refreshUsage(this.save) - + // Wrapper function to refresh quota usage after an operation, + // persisting argument and return types refreshUsage(fn: (...args: T) => Promise) { return async function (...args: T) { const response = await fn(...args) diff --git a/packages/frontend-core/src/api/user.ts b/packages/frontend-core/src/api/user.ts index 7464b1ec4a..cf66751078 100644 --- a/packages/frontend-core/src/api/user.ts +++ b/packages/frontend-core/src/api/user.ts @@ -21,11 +21,12 @@ import { SaveUserResponse, SearchUsersRequest, SearchUsersResponse, + UnsavedUser, UpdateInviteRequest, UpdateInviteResponse, UpdateSelfMetadataRequest, UpdateSelfMetadataResponse, - User, + UserIdentifier, } from "@budibase/types" import { BaseAPIClient } from "./types" @@ -38,14 +39,9 @@ export interface UserEndpoints { createAdminUser: ( user: CreateAdminUserRequest ) => Promise - saveUser: (user: User) => Promise + saveUser: (user: UnsavedUser) => Promise deleteUser: (userId: string) => Promise - deleteUsers: ( - users: Array<{ - userId: string - email: string - }> - ) => Promise + deleteUsers: (users: UserIdentifier[]) => Promise onboardUsers: (data: InviteUsersRequest) => Promise getUserInvite: (code: string) => Promise getUserInvites: () => Promise @@ -60,7 +56,7 @@ export interface UserEndpoints { getAccountHolder: () => Promise searchUsers: (data: SearchUsersRequest) => Promise createUsers: ( - users: Omit[], + users: UnsavedUser[], groups: any[] ) => Promise updateUserInvite: ( diff --git a/packages/types/src/api/web/user.ts b/packages/types/src/api/web/user.ts index 8b0dfef34b..c1f37fd3f0 100644 --- a/packages/types/src/api/web/user.ts +++ b/packages/types/src/api/web/user.ts @@ -22,6 +22,8 @@ export interface UserDetails { password?: string } +export type UnsavedUser = Omit + export interface BulkUserRequest { delete?: { users: Array<{ @@ -31,7 +33,7 @@ export interface BulkUserRequest { } create?: { roles?: any[] - users: User[] + users: UnsavedUser[] groups: any[] } } diff --git a/packages/worker/src/api/controllers/global/users.ts b/packages/worker/src/api/controllers/global/users.ts index a028f4fd33..4f9135e79e 100644 --- a/packages/worker/src/api/controllers/global/users.ts +++ b/packages/worker/src/api/controllers/global/users.ts @@ -33,6 +33,7 @@ import { SaveUserResponse, SearchUsersRequest, SearchUsersResponse, + UnsavedUser, UpdateInviteRequest, UpdateInviteResponse, User, @@ -49,6 +50,7 @@ import { tenancy, db, locks, + context, } from "@budibase/backend-core" import { checkAnyUserExists } from "../../../utilities/users" import { isEmailConfigured } from "../../../utilities/email" @@ -66,10 +68,11 @@ const generatePassword = (length: number) => { .slice(0, length) } -export const save = async (ctx: UserCtx) => { +export const save = async (ctx: UserCtx) => { try { const currentUserId = ctx.user?._id - const requestUser = ctx.request.body + const tenantId = context.getTenantId() + const requestUser: User = { ...ctx.request.body, tenantId } // Do not allow the account holder role to be changed const accountMetadata = await users.getExistingAccounts([requestUser.email]) @@ -149,7 +152,12 @@ export const bulkUpdate = async ( let created, deleted try { if (input.create) { - created = await bulkCreate(input.create.users, input.create.groups) + const tenantId = context.getTenantId() + const users: User[] = input.create.users.map(user => ({ + ...user, + tenantId, + })) + created = await bulkCreate(users, input.create.groups) } if (input.delete) { deleted = await bulkDelete(input.delete.users, currentUserId) @@ -441,7 +449,6 @@ export const checkInvite = async (ctx: UserCtx) => { } catch (e) { console.warn("Error getting invite from code", e) ctx.throw(400, "There was a problem with the invite") - return } ctx.body = { email: invite.email, @@ -472,7 +479,6 @@ export const updateInvite = async ( invite = await cache.invite.getCode(code) } catch (e) { ctx.throw(400, "There was a problem with the invite") - return } let updated = { From fcb3a3a1986eae17208ecf44cfa0d9b3e513d09c Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:24:23 +0000 Subject: [PATCH 54/60] Fix this reference --- packages/builder/src/stores/portal/users.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/builder/src/stores/portal/users.ts b/packages/builder/src/stores/portal/users.ts index 9284ad2992..1beb73a6c0 100644 --- a/packages/builder/src/stores/portal/users.ts +++ b/packages/builder/src/stores/portal/users.ts @@ -29,6 +29,8 @@ class UserStore extends BudiStore { data: [], }) + this.search = this.search.bind(this) + // Update quotas after any add or remove operation this.create = this.refreshUsage(this.create) this.save = this.refreshUsage(this.save) From a6ac76eb05f37c2402cd86c693334b36c6bee757 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:24:37 +0000 Subject: [PATCH 55/60] Fix group actions error --- .../builder/portal/users/groups/_components/GroupUsers.svelte | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/builder/src/pages/builder/portal/users/groups/_components/GroupUsers.svelte b/packages/builder/src/pages/builder/portal/users/groups/_components/GroupUsers.svelte index 8d99d406fd..71fd4c0be3 100644 --- a/packages/builder/src/pages/builder/portal/users/groups/_components/GroupUsers.svelte +++ b/packages/builder/src/pages/builder/portal/users/groups/_components/GroupUsers.svelte @@ -52,7 +52,7 @@ ] const removeUser = async id => { - await groups.actions.removeUser(groupId, id) + await groups.removeUser(groupId, id) fetchGroupUsers.refresh() } From 1ec4b4c6b7fc3f0a7f91b52797669612c061761e Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:38:00 +0000 Subject: [PATCH 56/60] Fix this reference --- .../src/pages/builder/portal/users/users/index.svelte | 1 + packages/builder/src/stores/portal/users.ts | 10 ++++------ 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/packages/builder/src/pages/builder/portal/users/users/index.svelte b/packages/builder/src/pages/builder/portal/users/users/index.svelte index 97120c55d4..c77e40c964 100644 --- a/packages/builder/src/pages/builder/portal/users/users/index.svelte +++ b/packages/builder/src/pages/builder/portal/users/users/index.svelte @@ -251,6 +251,7 @@ passwordModal.show() await fetch.refresh() } catch (error) { + console.error(error) notifications.error("Error creating user") } } diff --git a/packages/builder/src/stores/portal/users.ts b/packages/builder/src/stores/portal/users.ts index 1beb73a6c0..6503fc9280 100644 --- a/packages/builder/src/stores/portal/users.ts +++ b/packages/builder/src/stores/portal/users.ts @@ -29,13 +29,11 @@ class UserStore extends BudiStore { data: [], }) - this.search = this.search.bind(this) - // Update quotas after any add or remove operation - this.create = this.refreshUsage(this.create) - this.save = this.refreshUsage(this.save) - this.delete = this.refreshUsage(this.delete) - this.bulkDelete = this.refreshUsage(this.bulkDelete) + this.create = this.refreshUsage(this.create.bind(this)) + this.save = this.refreshUsage(this.save.bind(this)) + this.delete = this.refreshUsage(this.delete.bind(this)) + this.bulkDelete = this.refreshUsage(this.bulkDelete.bind(this)) } async search(opts: SearchUsersRequest = {}) { From bd378f0bd44f24bdd22c7fa55657ff96f4051d62 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:50:23 +0000 Subject: [PATCH 57/60] Simplify usage quota refreshing when doing user CRUD --- packages/builder/src/stores/portal/users.ts | 26 +++++++-------------- 1 file changed, 8 insertions(+), 18 deletions(-) diff --git a/packages/builder/src/stores/portal/users.ts b/packages/builder/src/stores/portal/users.ts index 6503fc9280..605f8612aa 100644 --- a/packages/builder/src/stores/portal/users.ts +++ b/packages/builder/src/stores/portal/users.ts @@ -28,12 +28,6 @@ class UserStore extends BudiStore { super({ data: [], }) - - // Update quotas after any add or remove operation - this.create = this.refreshUsage(this.create.bind(this)) - this.save = this.refreshUsage(this.save.bind(this)) - this.delete = this.refreshUsage(this.delete.bind(this)) - this.bulkDelete = this.refreshUsage(this.bulkDelete.bind(this)) } async search(opts: SearchUsersRequest = {}) { @@ -156,6 +150,7 @@ class UserStore extends BudiStore { return body }) const response = await API.createUsers(mappedUsers, data.groups) + licensing.setQuotaUsage() // re-search from first page await this.search() @@ -164,14 +159,19 @@ class UserStore extends BudiStore { async delete(id: string) { await API.deleteUser(id) + licensing.setQuotaUsage() } async bulkDelete(users: UserIdentifier[]) { - return API.deleteUsers(users) + const res = API.deleteUsers(users) + licensing.setQuotaUsage() + return res } async save(user: User) { - return await API.saveUser(user) + const res = await API.saveUser(user) + licensing.setQuotaUsage() + return res } async addAppBuilder(userId: string, appId: string) { @@ -202,16 +202,6 @@ class UserStore extends BudiStore { return Constants.BudibaseRoles.AppUser } } - - // Wrapper function to refresh quota usage after an operation, - // persisting argument and return types - refreshUsage(fn: (...args: T) => Promise) { - return async function (...args: T) { - const response = await fn(...args) - await licensing.setQuotaUsage() - return response - } - } } export const users = new UserStore() From 53b12075bb1b35bc7839f46384c6b7e907a92d15 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:57:10 +0000 Subject: [PATCH 58/60] Fix unrelated automation log error message --- .../src/pages/builder/portal/apps/index.svelte | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/packages/builder/src/pages/builder/portal/apps/index.svelte b/packages/builder/src/pages/builder/portal/apps/index.svelte index f94bad2147..5c3ee674e9 100644 --- a/packages/builder/src/pages/builder/portal/apps/index.svelte +++ b/packages/builder/src/pages/builder/portal/apps/index.svelte @@ -176,6 +176,8 @@ notifications.error("Error getting init info") } }) + + $: console.log(automationErrors) @@ -191,8 +193,14 @@ ? "View errors" : "View error"} on:dismiss={async () => { - await automationStore.actions.clearLogErrors({ appId }) - await appsStore.load() + const automationId = Object.keys(automationErrors[appId] || {})[0] + if (automationId) { + await automationStore.actions.clearLogErrors({ + appId, + automationId, + }) + await appsStore.load() + } }} message={automationErrorMessage(appId)} /> From 79a74ff709706ed37aa3a3a9084f0ba7f4fcb133 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 13 Jan 2025 10:58:31 +0000 Subject: [PATCH 59/60] Remove log --- packages/builder/src/pages/builder/portal/apps/index.svelte | 2 -- 1 file changed, 2 deletions(-) diff --git a/packages/builder/src/pages/builder/portal/apps/index.svelte b/packages/builder/src/pages/builder/portal/apps/index.svelte index 5c3ee674e9..bcd59cd948 100644 --- a/packages/builder/src/pages/builder/portal/apps/index.svelte +++ b/packages/builder/src/pages/builder/portal/apps/index.svelte @@ -176,8 +176,6 @@ notifications.error("Error getting init info") } }) - - $: console.log(automationErrors) From da4014e5aaa7b046638c781f8d12d90c7707de3e Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 13 Jan 2025 12:19:44 +0100 Subject: [PATCH 60/60] Remove outdated TODO --- packages/backend-core/src/sql/sql.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/packages/backend-core/src/sql/sql.ts b/packages/backend-core/src/sql/sql.ts index cc5f226d56..fa9ef9a4d6 100644 --- a/packages/backend-core/src/sql/sql.ts +++ b/packages/backend-core/src/sql/sql.ts @@ -313,8 +313,6 @@ class InternalBuilder { // Time gets returned as timestamp from mssql, not matching the expected // HH:mm format - // TODO: figure out how to express this safely without string - // interpolation. return this.knex.raw(`CONVERT(varchar, ??, 108) as ??`, [ this.rawQuotedIdentifier(field), this.knex.raw(this.quote(field)),