From d8d3d7152350675be31c5094413f47fc0703034d Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Thu, 1 Jun 2023 17:14:32 +0100 Subject: [PATCH 01/15] Update websocket logic to ignore events trigger by API requests originating from the same session --- packages/backend-core/src/constants/misc.ts | 1 + packages/client/src/websocket.js | 4 ++- packages/frontend-core/src/api/index.js | 10 +++++++ .../src/components/grid/lib/websocket.js | 8 +++--- packages/frontend-core/src/utils/websocket.js | 25 +++++++++++++++-- packages/server/src/websockets/builder.ts | 27 ++++++++++--------- packages/server/src/websockets/grid.ts | 20 ++++++++------ packages/server/src/websockets/websocket.ts | 11 +++++++- 8 files changed, 78 insertions(+), 28 deletions(-) diff --git a/packages/backend-core/src/constants/misc.ts b/packages/backend-core/src/constants/misc.ts index 2bb8f815cf..ba2533cf4a 100644 --- a/packages/backend-core/src/constants/misc.ts +++ b/packages/backend-core/src/constants/misc.ts @@ -16,6 +16,7 @@ export enum Header { LICENSE_KEY = "x-budibase-license-key", API_VER = "x-budibase-api-version", APP_ID = "x-budibase-app-id", + SESSION_ID = "x-budibase-session-id", TYPE = "x-budibase-type", PREVIEW_ROLE = "x-budibase-role", TENANT_ID = "x-budibase-tenant-id", diff --git a/packages/client/src/websocket.js b/packages/client/src/websocket.js index 1e8e908d2b..8ca95fdc84 100644 --- a/packages/client/src/websocket.js +++ b/packages/client/src/websocket.js @@ -18,7 +18,9 @@ export const initWebsocket = () => { } // Initialise connection - socket = createWebsocket("/socket/client", false) + socket = createWebsocket("/socket/client", { + heartbeat: false, + }) // Event handlers socket.on("plugin-update", data => { diff --git a/packages/frontend-core/src/api/index.js b/packages/frontend-core/src/api/index.js index f8eee45cb8..739e0fe711 100644 --- a/packages/frontend-core/src/api/index.js +++ b/packages/frontend-core/src/api/index.js @@ -1,3 +1,4 @@ +import { Helpers } from "@budibase/bbui" import { ApiVersion } from "../constants" import { buildAnalyticsEndpoints } from "./analytics" import { buildAppEndpoints } from "./app" @@ -30,6 +31,14 @@ import { buildEnvironmentVariableEndpoints } from "./environmentVariables" import { buildEventEndpoints } from "./events" import { buildAuditLogsEndpoints } from "./auditLogs" +/** + * Random identifier to uniquely identify a session in a tab. This is + * used to determine the originator of calls to the API, which is in + * turn used to determine who caused a websocket message to be sent, so + * that we can ignore events caused by ourselves. + */ +export const APISessionID = Helpers.uuid() + const defaultAPIClientConfig = { /** * Certain definitions can't change at runtime for client apps, such as the @@ -116,6 +125,7 @@ export const createAPIClient = config => { // Build headers let headers = { Accept: "application/json" } + headers["x-budibase-session-id"] = APISessionID if (!external) { headers["x-budibase-api-version"] = ApiVersion } diff --git a/packages/frontend-core/src/components/grid/lib/websocket.js b/packages/frontend-core/src/components/grid/lib/websocket.js index bb1d6991b0..11164b3148 100644 --- a/packages/frontend-core/src/components/grid/lib/websocket.js +++ b/packages/frontend-core/src/components/grid/lib/websocket.js @@ -26,15 +26,15 @@ export const createGridWebsocket = context => { }) // User events - socket.on(SocketEvent.UserUpdate, user => { + socket.onOther(SocketEvent.UserUpdate, user => { users.actions.updateUser(user) }) - socket.on(SocketEvent.UserDisconnect, user => { + socket.onOther(SocketEvent.UserDisconnect, user => { users.actions.removeUser(user) }) // Row events - socket.on(GridSocketEvent.RowChange, async data => { + socket.onOther(GridSocketEvent.RowChange, async data => { if (data.id) { rows.actions.replaceRow(data.id, data.row) } else if (data.row.id) { @@ -44,7 +44,7 @@ export const createGridWebsocket = context => { }) // Table events - socket.on(GridSocketEvent.TableChange, data => { + socket.onOther(GridSocketEvent.TableChange, data => { // Only update table if one exists. If the table was deleted then we don't // want to know - let the builder navigate away if (data.table) { diff --git a/packages/frontend-core/src/utils/websocket.js b/packages/frontend-core/src/utils/websocket.js index 561a020e13..3a0fa23179 100644 --- a/packages/frontend-core/src/utils/websocket.js +++ b/packages/frontend-core/src/utils/websocket.js @@ -1,17 +1,26 @@ import { io } from "socket.io-client" import { SocketEvent, SocketSessionTTL } from "@budibase/shared-core" +import { APISessionID } from "../api" -export const createWebsocket = (path, heartbeat = true) => { +const DefaultOptions = { + heartbeat: true, +} + +export const createWebsocket = (path, options = DefaultOptions) => { if (!path) { throw "A websocket path must be provided" } + const { heartbeat } = { + ...DefaultOptions, + ...options, + } // Determine connection info const tls = location.protocol === "https:" const proto = tls ? "wss:" : "ws:" const host = location.hostname const port = location.port || (tls ? 443 : 80) - const socket = io(`${proto}//${host}:${port}`, { + let socket = io(`${proto}//${host}:${port}`, { path, // Cap reconnection attempts to 3 (total of 15 seconds before giving up) reconnectionAttempts: 3, @@ -37,5 +46,17 @@ export const createWebsocket = (path, heartbeat = true) => { clearInterval(interval) }) + // Helper utility to ignore events that were triggered due to API + // changes made by us + socket.onOther = (event, callback) => { + socket.on(event, data => { + if (data?.apiSessionId !== APISessionID) { + callback(data) + } else { + console.log("ignore", event, data) + } + }) + } + return socket } diff --git a/packages/server/src/websockets/builder.ts b/packages/server/src/websockets/builder.ts index ff7c50ebf4..0015b7f601 100644 --- a/packages/server/src/websockets/builder.ts +++ b/packages/server/src/websockets/builder.ts @@ -46,29 +46,32 @@ export default class BuilderSocket extends BaseSocket { } emitTableUpdate(ctx: any, table: Table) { - this.io - .in(ctx.appId) - .emit(BuilderSocketEvent.TableChange, { id: table._id, table }) - gridSocket?.emitTableUpdate(table) + this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.TableChange, { + id: table._id, + table, + }) + gridSocket?.emitTableUpdate(ctx, table) } emitTableDeletion(ctx: any, id: string) { - this.io - .in(ctx.appId) - .emit(BuilderSocketEvent.TableChange, { id, table: null }) - gridSocket?.emitTableDeletion(id) + this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.TableChange, { + id, + table: null, + }) + gridSocket?.emitTableDeletion(ctx, id) } emitDatasourceUpdate(ctx: any, datasource: Datasource) { - this.io.in(ctx.appId).emit(BuilderSocketEvent.DatasourceChange, { + this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.DatasourceChange, { id: datasource._id, datasource, }) } emitDatasourceDeletion(ctx: any, id: string) { - this.io - .in(ctx.appId) - .emit(BuilderSocketEvent.DatasourceChange, { id, datasource: null }) + this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.DatasourceChange, { + id, + datasource: null, + }) } } diff --git a/packages/server/src/websockets/grid.ts b/packages/server/src/websockets/grid.ts index 440b4f78a5..c12715990e 100644 --- a/packages/server/src/websockets/grid.ts +++ b/packages/server/src/websockets/grid.ts @@ -31,21 +31,25 @@ export default class GridSocket extends BaseSocket { emitRowUpdate(ctx: any, row: Row) { const tableId = getTableId(ctx) - this.io.in(tableId).emit(GridSocketEvent.RowChange, { id: row._id, row }) + this.emitToRoom(ctx, tableId, GridSocketEvent.RowChange, { + id: row._id, + row, + }) } emitRowDeletion(ctx: any, id: string) { const tableId = getTableId(ctx) - this.io.in(tableId).emit(GridSocketEvent.RowChange, { id, row: null }) + this.emitToRoom(ctx, tableId, GridSocketEvent.RowChange, { id, row: null }) } - emitTableUpdate(table: Table) { - this.io - .in(table._id!) - .emit(GridSocketEvent.TableChange, { id: table._id, table }) + emitTableUpdate(ctx: any, table: Table) { + this.emitToRoom(ctx, table._id!, GridSocketEvent.TableChange, { + id: table._id, + table, + }) } - emitTableDeletion(id: string) { - this.io.in(id).emit(GridSocketEvent.TableChange, { id, table: null }) + emitTableDeletion(ctx: any, id: string) { + this.emitToRoom(ctx, id, GridSocketEvent.TableChange, { id, table: null }) } } diff --git a/packages/server/src/websockets/websocket.ts b/packages/server/src/websockets/websocket.ts index a2d2474c77..d9c105d77c 100644 --- a/packages/server/src/websockets/websocket.ts +++ b/packages/server/src/websockets/websocket.ts @@ -3,7 +3,7 @@ import http from "http" import Koa from "koa" import Cookies from "cookies" import { userAgent } from "koa-useragent" -import { auth, redis } from "@budibase/backend-core" +import { auth, Header, redis } from "@budibase/backend-core" import currentApp from "../middleware/currentapp" import { createAdapter } from "@socket.io/redis-adapter" import { Socket } from "socket.io" @@ -271,4 +271,13 @@ export class BaseSocket { emit(event: string, payload: any) { this.io.sockets.emit(event, payload) } + + // Emit an event to everyone in a room, including metadata of whom + // the originator of the request was + emitToRoom(ctx: any, room: string, event: string, payload: any) { + this.io.in(room).emit(event, { + ...payload, + apiSessionId: ctx.headers?.[Header.SESSION_ID], + }) + } } From adbdaf394a777e813afbd31d841d77d853d0428b Mon Sep 17 00:00:00 2001 From: Michael Drury Date: Thu, 1 Jun 2023 18:18:12 +0100 Subject: [PATCH 02/15] Small update to make sure that save/patch calls to the rows API (all DS+) return the squashed relationships, not the whole structure. --- .../src/api/controllers/row/external.ts | 54 +++++++++++++------ .../server/src/api/controllers/row/index.ts | 7 +-- .../src/api/controllers/row/staticFormula.ts | 7 ++- packages/server/src/db/linkedRows/index.ts | 8 +-- .../src/api/controllers/global/configs.ts | 1 - 5 files changed, 54 insertions(+), 23 deletions(-) diff --git a/packages/server/src/api/controllers/row/external.ts b/packages/server/src/api/controllers/row/external.ts index ee796e32d4..a68586b099 100644 --- a/packages/server/src/api/controllers/row/external.ts +++ b/packages/server/src/api/controllers/row/external.ts @@ -1,7 +1,7 @@ import { - SortDirection, FieldTypes, NoEmptyFilterStrings, + SortDirection, } from "../../../constants" import { breakExternalTableId, @@ -11,20 +11,34 @@ import { ExternalRequest, RunConfig } from "./ExternalRequest" import * as exporters from "../view/exporters" import { apiFileReturn } from "../../../utilities/fileSystem" import { - Operation, - UserCtx, - Row, - PaginationJson, - Table, Datasource, IncludeRelationship, + Operation, + PaginationJson, + Row, SortJson, + Table, + UserCtx, } from "@budibase/types" import sdk from "../../../sdk" import * as utils from "./utils" const { cleanExportRows } = require("./utils") +async function getRow( + tableId: string, + rowId: string, + opts?: { relationships?: boolean } +) { + const response = (await handleRequest(Operation.READ, tableId, { + id: breakRowIdField(rowId), + includeSqlRelationships: opts?.relationships + ? IncludeRelationship.INCLUDE + : IncludeRelationship.EXCLUDE, + })) as Row[] + return response ? response[0] : response +} + export async function handleRequest( operation: Operation, tableId: string, @@ -63,11 +77,15 @@ export async function patch(ctx: UserCtx) { if (!validateResult.valid) { throw { validation: validateResult.errors } } - return handleRequest(Operation.UPDATE, tableId, { + const response = await handleRequest(Operation.UPDATE, tableId, { id: breakRowIdField(id), row: inputs, - includeSqlRelationships: IncludeRelationship.INCLUDE, }) + const row = await getRow(tableId, id, { relationships: true }) + return { + ...response, + row, + } } export async function save(ctx: UserCtx) { @@ -80,10 +98,20 @@ export async function save(ctx: UserCtx) { if (!validateResult.valid) { throw { validation: validateResult.errors } } - return handleRequest(Operation.CREATE, tableId, { + const response = await handleRequest(Operation.CREATE, tableId, { row: inputs, - includeSqlRelationships: IncludeRelationship.EXCLUDE, }) + const responseRow = response as { row: Row } + const rowId = responseRow.row._id + if (rowId) { + const row = await getRow(tableId, rowId, { relationships: true }) + return { + ...response, + row, + } + } else { + return response + } } export async function fetchView(ctx: UserCtx) { @@ -104,11 +132,7 @@ export async function fetch(ctx: UserCtx) { export async function find(ctx: UserCtx) { const id = ctx.params.rowId const tableId = ctx.params.tableId - const response = (await handleRequest(Operation.READ, tableId, { - id: breakRowIdField(id), - includeSqlRelationships: IncludeRelationship.EXCLUDE, - })) as Row[] - return response ? response[0] : response + return getRow(tableId, id) } export async function destroy(ctx: UserCtx) { diff --git a/packages/server/src/api/controllers/row/index.ts b/packages/server/src/api/controllers/row/index.ts index 55d2d27cce..91270429a4 100644 --- a/packages/server/src/api/controllers/row/index.ts +++ b/packages/server/src/api/controllers/row/index.ts @@ -50,7 +50,7 @@ export const save = async (ctx: any) => { if (body && body._id) { return patch(ctx) } - const { row, table } = await quotas.addRow(() => + const { row, table, squashed } = await quotas.addRow(() => quotas.addQuery(() => pickApi(tableId).save(ctx), { datasourceId: tableId, }) @@ -58,8 +58,9 @@ export const save = async (ctx: any) => { ctx.status = 200 ctx.eventEmitter && ctx.eventEmitter.emitRow(`row:save`, appId, row, table) ctx.message = `${table.name} saved successfully` - ctx.body = row - gridSocket?.emitRowUpdate(ctx, row) + // prefer squashed for response + ctx.body = row || squashed + gridSocket?.emitRowUpdate(ctx, row || squashed) } export async function fetchView(ctx: any) { const tableId = utils.getTableId(ctx) diff --git a/packages/server/src/api/controllers/row/staticFormula.ts b/packages/server/src/api/controllers/row/staticFormula.ts index 6ba44dc23a..9f2c64c7b4 100644 --- a/packages/server/src/api/controllers/row/staticFormula.ts +++ b/packages/server/src/api/controllers/row/staticFormula.ts @@ -7,6 +7,7 @@ import { import { FieldTypes, FormulaTypes } from "../../../constants" import { context } from "@budibase/backend-core" import { Table, Row } from "@budibase/types" +import * as linkRows from "../../../db/linkedRows" const { isEqual } = require("lodash") const { cloneDeep } = require("lodash/fp") @@ -166,5 +167,9 @@ export async function finaliseRow( if (updateFormula) { await updateRelatedFormula(table, enrichedRow) } - return { row: enrichedRow, table } + const squashed = await linkRows.squashLinksToPrimaryDisplay( + table, + enrichedRow + ) + return { row: enrichedRow, squashed, table } } diff --git a/packages/server/src/db/linkedRows/index.ts b/packages/server/src/db/linkedRows/index.ts index a6ed7de161..4d58d32f9b 100644 --- a/packages/server/src/db/linkedRows/index.ts +++ b/packages/server/src/db/linkedRows/index.ts @@ -189,11 +189,13 @@ export async function attachFullLinkedDocs(table: Table, rows: Row[]) { */ export async function squashLinksToPrimaryDisplay( table: Table, - enriched: Row[] + enriched: Row[] | Row ) { // will populate this as we find them const linkedTables = [table] - for (let row of enriched) { + const isArray = Array.isArray(enriched) + let enrichedArray = !isArray ? [enriched] : enriched + for (let row of enrichedArray) { // this only fetches the table if its not already in array const rowTable = await getLinkedTable(row.tableId!, linkedTables) for (let [column, schema] of Object.entries(rowTable?.schema || {})) { @@ -213,5 +215,5 @@ export async function squashLinksToPrimaryDisplay( row[column] = newLinks } } - return enriched + return isArray ? enrichedArray : enrichedArray[0] } diff --git a/packages/worker/src/api/controllers/global/configs.ts b/packages/worker/src/api/controllers/global/configs.ts index 66804c3b9c..afbb7c931d 100644 --- a/packages/worker/src/api/controllers/global/configs.ts +++ b/packages/worker/src/api/controllers/global/configs.ts @@ -11,7 +11,6 @@ import { tenancy, } from "@budibase/backend-core" import { checkAnyUserExists } from "../../../utilities/users" -import { getLicensedConfig } from "../../../utilities/configs" import { Config, ConfigType, From b4def3675af2181064f2f0f6fc74ea6df62fe9c6 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 08:53:47 +0100 Subject: [PATCH 03/15] Remove manual refresh of row after creation --- packages/frontend-core/src/components/grid/stores/rows.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/frontend-core/src/components/grid/stores/rows.js b/packages/frontend-core/src/components/grid/stores/rows.js index 4b93e1d58f..c731d4ec44 100644 --- a/packages/frontend-core/src/components/grid/stores/rows.js +++ b/packages/frontend-core/src/components/grid/stores/rows.js @@ -214,8 +214,7 @@ export const deriveStores = context => { const addRow = async (row, idx, bubble = false) => { try { // Create row - let newRow = await API.saveRow({ ...row, tableId: get(tableId) }) - newRow = await fetchRow(newRow._id) + const newRow = await API.saveRow({ ...row, tableId: get(tableId) }) // Update state if (idx != null) { From dab32635691cb52ddb0a52a7a80c0aa7c7140c4a Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 08:58:32 +0100 Subject: [PATCH 04/15] Fix losing cell focus when another user makes changes while you're adding a row --- packages/frontend-core/src/components/grid/stores/rows.js | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packages/frontend-core/src/components/grid/stores/rows.js b/packages/frontend-core/src/components/grid/stores/rows.js index c731d4ec44..198c05025c 100644 --- a/packages/frontend-core/src/components/grid/stores/rows.js +++ b/packages/frontend-core/src/components/grid/stores/rows.js @@ -436,6 +436,9 @@ export const deriveStores = context => { // Checks if we have a row with a certain ID const hasRow = id => { + if (id === NewRowID) { + return true + } return get(rowLookupMap)[id] != null } From 53f1a4e67d67360ff034ea59072c71bc041f9aec Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 09:04:50 +0100 Subject: [PATCH 05/15] Update builder socket to ignore own events, and ensure table creation updates state without depending on the websocket --- .../builder/src/builderStore/websocket.js | 30 +++++++++---------- packages/builder/src/stores/backend/tables.js | 2 +- 2 files changed, 15 insertions(+), 17 deletions(-) diff --git a/packages/builder/src/builderStore/websocket.js b/packages/builder/src/builderStore/websocket.js index 6a3ce76ae6..e27e08e31d 100644 --- a/packages/builder/src/builderStore/websocket.js +++ b/packages/builder/src/builderStore/websocket.js @@ -15,25 +15,23 @@ export const createBuilderWebsocket = appId => { socket.on("connect_error", err => { console.log("Failed to connect to builder websocket:", err.message) }) - - // User events - socket.on(SocketEvent.UserUpdate, userStore.actions.updateUser) - socket.on(SocketEvent.UserDisconnect, userStore.actions.removeUser) - - // Table events - socket.on(BuilderSocketEvent.TableChange, ({ id, table }) => { - tables.replaceTable(id, table) - }) - - // Datasource events - socket.on(BuilderSocketEvent.DatasourceChange, ({ id, datasource }) => { - datasources.replaceDatasource(id, datasource) - }) - - // Clean up user store on disconnect socket.on("disconnect", () => { userStore.actions.reset() }) + // User events + socket.onOther(SocketEvent.UserUpdate, userStore.actions.updateUser) + socket.onOther(SocketEvent.UserDisconnect, userStore.actions.removeUser) + + // Table events + socket.onOther(BuilderSocketEvent.TableChange, ({ id, table }) => { + tables.replaceTable(id, table) + }) + + // Datasource events + socket.onOther(BuilderSocketEvent.DatasourceChange, ({ id, datasource }) => { + datasources.replaceDatasource(id, datasource) + }) + return socket } diff --git a/packages/builder/src/stores/backend/tables.js b/packages/builder/src/stores/backend/tables.js index a36c91d1b1..f8796712a8 100644 --- a/packages/builder/src/stores/backend/tables.js +++ b/packages/builder/src/stores/backend/tables.js @@ -62,7 +62,7 @@ export function createTablesStore() { } const savedTable = await API.saveTable(updatedTable) - replaceTable(table._id, savedTable) + replaceTable(savedTable._id, savedTable) await datasources.fetch() select(savedTable._id) return savedTable From 2db3012ae0c1f056e376275c3ec8fe39f7851a48 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 15:13:57 +0100 Subject: [PATCH 06/15] Fix JSON editor overflow row edit modal --- .../src/components/backend/DataTable/RowFieldControl.svelte | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte b/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte index 1a4ced9f3a..61b706e28e 100644 --- a/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte +++ b/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte @@ -81,6 +81,7 @@ (value = detail.value)} value={stringVal} From 0582dd40b229ad4e3ba148fb7cfb802eef48b5b8 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 15:28:48 +0100 Subject: [PATCH 07/15] Ensue table schema change made via grids are properly synced with backend stores, given absence of websocket events for own events --- .../src/components/backend/DataTable/DataTable.svelte | 1 + .../frontend-core/src/components/grid/stores/columns.js | 6 +++++- packages/frontend-core/src/utils/websocket.js | 2 -- 3 files changed, 6 insertions(+), 3 deletions(-) diff --git a/packages/builder/src/components/backend/DataTable/DataTable.svelte b/packages/builder/src/components/backend/DataTable/DataTable.svelte index 3e9b7d831e..1b0c92bde0 100644 --- a/packages/builder/src/components/backend/DataTable/DataTable.svelte +++ b/packages/builder/src/components/backend/DataTable/DataTable.svelte @@ -37,6 +37,7 @@ allowDeleteRows={!isUsersTable} schemaOverrides={isUsersTable ? userSchemaOverrides : null} showAvatars={false} + on:updatetable={e => tables.replaceTable(id, e.detail)} > {#if isInternal} diff --git a/packages/frontend-core/src/components/grid/stores/columns.js b/packages/frontend-core/src/components/grid/stores/columns.js index b9f0b340a9..024fdc29bc 100644 --- a/packages/frontend-core/src/components/grid/stores/columns.js +++ b/packages/frontend-core/src/components/grid/stores/columns.js @@ -46,7 +46,7 @@ export const createStores = () => { } export const deriveStores = context => { - const { table, columns, stickyColumn, API } = context + const { table, columns, stickyColumn, API, dispatch } = context // Updates the tables primary display column const changePrimaryDisplay = async column => { @@ -90,6 +90,10 @@ export const deriveStores = context => { // Update local state table.set(newTable) + // Broadcast change to external state can be updated, as this change + // will not be received by the builder websocket because we caused it ourselves + dispatch("updatetable", newTable) + // Update server await API.saveTable(newTable) } diff --git a/packages/frontend-core/src/utils/websocket.js b/packages/frontend-core/src/utils/websocket.js index 3a0fa23179..dee679eaef 100644 --- a/packages/frontend-core/src/utils/websocket.js +++ b/packages/frontend-core/src/utils/websocket.js @@ -52,8 +52,6 @@ export const createWebsocket = (path, options = DefaultOptions) => { socket.on(event, data => { if (data?.apiSessionId !== APISessionID) { callback(data) - } else { - console.log("ignore", event, data) } }) } From 6f51843875207e021b0c1a17e40bb71e2d370d3b Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Fri, 2 Jun 2023 16:22:07 +0100 Subject: [PATCH 08/15] Fix TS issue --- packages/server/src/utilities/rowProcessor/index.ts | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/packages/server/src/utilities/rowProcessor/index.ts b/packages/server/src/utilities/rowProcessor/index.ts index 44cab4d18b..4b6e0f6e87 100644 --- a/packages/server/src/utilities/rowProcessor/index.ts +++ b/packages/server/src/utilities/rowProcessor/index.ts @@ -216,7 +216,10 @@ export async function outputProcessing( } } if (opts.squash) { - enriched = await linkRows.squashLinksToPrimaryDisplay(table, enriched) + enriched = (await linkRows.squashLinksToPrimaryDisplay( + table, + enriched + )) as Row[] } return wasArray ? enriched : enriched[0] } From 0addd4eaf80af3454c051fd52d6aa61ed807e9a0 Mon Sep 17 00:00:00 2001 From: Budibase Staging Release Bot <> Date: Fri, 2 Jun 2023 17:07:44 +0000 Subject: [PATCH 09/15] Bump version to 2.6.19-alpha.49 --- lerna.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lerna.json b/lerna.json index 42d7ab617b..1778c2f37c 100644 --- a/lerna.json +++ b/lerna.json @@ -1,5 +1,5 @@ { - "version": "2.6.19-alpha.48", + "version": "2.6.19-alpha.49", "npmClient": "yarn", "packages": [ "packages/backend-core", From f7f9a359b24fe7d152a747afdaa49f08ade0abcc Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 5 Jun 2023 08:27:14 +0100 Subject: [PATCH 10/15] Ensure edit button in overview is never disabled as we don't use the lock any more --- .../portal/overview/[appId]/_layout.svelte | 21 +++---------------- 1 file changed, 3 insertions(+), 18 deletions(-) diff --git a/packages/builder/src/pages/builder/portal/overview/[appId]/_layout.svelte b/packages/builder/src/pages/builder/portal/overview/[appId]/_layout.svelte index 8ee469a914..333a10bef7 100644 --- a/packages/builder/src/pages/builder/portal/overview/[appId]/_layout.svelte +++ b/packages/builder/src/pages/builder/portal/overview/[appId]/_layout.svelte @@ -20,7 +20,7 @@ Breadcrumb, Header, } from "components/portal/page" - import { apps, auth, overview } from "stores/portal" + import { apps, overview } from "stores/portal" import { AppStatus } from "constants" import analytics, { Events, EventSource } from "analytics" import { store } from "builderStore" @@ -52,8 +52,6 @@ $: appId = $overview.selectedAppId $: initialiseApp(appId) $: isPublished = app?.status === AppStatus.DEPLOYED - $: appLocked = !!app?.lockedBy - $: lockedByYou = $auth.user.email === app?.lockedBy?.email const initialiseApp = async appId => { loaded = false @@ -139,14 +137,7 @@ - + @@ -158,13 +149,7 @@ - - Edit - + Edit exportApp({ published: false })} From 4f176b3edda8b6d0aba2633e31c3664a719d1f8b Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 5 Jun 2023 08:30:53 +0100 Subject: [PATCH 11/15] Increase padding around the grid --- packages/frontend-core/src/components/grid/lib/constants.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/frontend-core/src/components/grid/lib/constants.js b/packages/frontend-core/src/components/grid/lib/constants.js index a7209d6ea4..f078ef7274 100644 --- a/packages/frontend-core/src/components/grid/lib/constants.js +++ b/packages/frontend-core/src/components/grid/lib/constants.js @@ -1,4 +1,4 @@ -export const Padding = 128 +export const Padding = 256 export const MaxCellRenderHeight = 252 export const MaxCellRenderWidthOverflow = 200 export const ScrollBarSize = 8 From 920fea7b1f1d5233d9a154644cf07e0bf137e1a6 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 5 Jun 2023 08:48:02 +0100 Subject: [PATCH 12/15] Remove beta button from grid, add feedback button to data section linking to collaboration maze --- .../data/_components}/BetaButton.svelte | 24 ++++--------------- .../DatasourceOption.svelte} | 0 .../app/[application]/data/_layout.svelte | 3 +++ .../builder/app/[application]/data/new.svelte | 2 +- .../src/components/grid/layout/Grid.svelte | 2 -- 5 files changed, 8 insertions(+), 23 deletions(-) rename packages/{frontend-core/src/components/grid/controls => builder/src/pages/builder/app/[application]/data/_components}/BetaButton.svelte (50%) rename packages/builder/src/pages/builder/app/[application]/data/{_DatasourceOption.svelte => _components/DatasourceOption.svelte} (100%) diff --git a/packages/frontend-core/src/components/grid/controls/BetaButton.svelte b/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte similarity index 50% rename from packages/frontend-core/src/components/grid/controls/BetaButton.svelte rename to packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte index e5069388e7..1349fdb13c 100644 --- a/packages/frontend-core/src/components/grid/controls/BetaButton.svelte +++ b/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte @@ -2,13 +2,11 @@ import { Button } from "@budibase/bbui" -
- Enjoying the Grid? @@ -17,30 +15,16 @@ diff --git a/packages/builder/src/pages/builder/app/[application]/data/_DatasourceOption.svelte b/packages/builder/src/pages/builder/app/[application]/data/_components/DatasourceOption.svelte similarity index 100% rename from packages/builder/src/pages/builder/app/[application]/data/_DatasourceOption.svelte rename to packages/builder/src/pages/builder/app/[application]/data/_components/DatasourceOption.svelte diff --git a/packages/builder/src/pages/builder/app/[application]/data/_layout.svelte b/packages/builder/src/pages/builder/app/[application]/data/_layout.svelte index 87c4db81df..c0813fd2b8 100644 --- a/packages/builder/src/pages/builder/app/[application]/data/_layout.svelte +++ b/packages/builder/src/pages/builder/app/[application]/data/_layout.svelte @@ -3,6 +3,7 @@ import DatasourceNavigator from "components/backend/DatasourceNavigator/DatasourceNavigator.svelte" import Panel from "components/design/Panel.svelte" import { isActive, goto } from "@roxi/routify" + import BetaButton from "./_components/BetaButton.svelte" @@ -19,6 +20,7 @@
+
diff --git a/packages/builder/src/pages/builder/app/[application]/data/new.svelte b/packages/builder/src/pages/builder/app/[application]/data/new.svelte index f8e8fd85e7..fedaf013da 100644 --- a/packages/builder/src/pages/builder/app/[application]/data/new.svelte +++ b/packages/builder/src/pages/builder/app/[application]/data/new.svelte @@ -13,7 +13,7 @@ import DatasourceConfigModal from "components/backend/DatasourceNavigator/modals/DatasourceConfigModal.svelte" import GoogleDatasourceConfigModal from "components/backend/DatasourceNavigator/modals/GoogleDatasourceConfigModal.svelte" import { createRestDatasource } from "builderStore/datasource" - import DatasourceOption from "./_DatasourceOption.svelte" + import DatasourceOption from "./_components/DatasourceOption.svelte" import IntegrationIcon from "components/backend/DatasourceNavigator/IntegrationIcon.svelte" import ICONS from "components/backend/DatasourceNavigator/icons/index.js" import FontAwesomeIcon from "components/common/FontAwesomeIcon.svelte" diff --git a/packages/frontend-core/src/components/grid/layout/Grid.svelte b/packages/frontend-core/src/components/grid/layout/Grid.svelte index 08325857d6..81f42ca257 100644 --- a/packages/frontend-core/src/components/grid/layout/Grid.svelte +++ b/packages/frontend-core/src/components/grid/layout/Grid.svelte @@ -7,7 +7,6 @@ import { createAPIClient } from "../../../api" import { attachStores } from "../stores" import BulkDeleteHandler from "../controls/BulkDeleteHandler.svelte" - import BetaButton from "../controls/BetaButton.svelte" import GridBody from "./GridBody.svelte" import ResizeOverlay from "../overlays/ResizeOverlay.svelte" import ReorderOverlay from "../overlays/ReorderOverlay.svelte" @@ -144,7 +143,6 @@
- {#if allowAddRows} {/if} From ffde512307e9eeb2486e8bd6be5bdda15bbc63fc Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Mon, 5 Jun 2023 08:49:35 +0100 Subject: [PATCH 13/15] Update position of floating buttons to be consistent --- packages/builder/src/components/common/HelpMenu.svelte | 2 +- .../app/[application]/data/_components/BetaButton.svelte | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/builder/src/components/common/HelpMenu.svelte b/packages/builder/src/components/common/HelpMenu.svelte index 274565b255..06ff277d18 100644 --- a/packages/builder/src/components/common/HelpMenu.svelte +++ b/packages/builder/src/components/common/HelpMenu.svelte @@ -83,7 +83,7 @@ .help { z-index: 2; position: absolute; - bottom: var(--spacing-xl); + bottom: 24px; right: 24px; } diff --git a/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte b/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte index 1349fdb13c..3cfe4b4d8d 100644 --- a/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte +++ b/packages/builder/src/pages/builder/app/[application]/data/_components/BetaButton.svelte @@ -15,7 +15,7 @@