Update websocket logic to ignore events trigger by API requests originating from the same session

This commit is contained in:
Andrew Kingston 2023-06-01 17:14:32 +01:00
parent bdc03e12cc
commit d8d3d71523
8 changed files with 78 additions and 28 deletions

View File

@ -16,6 +16,7 @@ export enum Header {
LICENSE_KEY = "x-budibase-license-key", LICENSE_KEY = "x-budibase-license-key",
API_VER = "x-budibase-api-version", API_VER = "x-budibase-api-version",
APP_ID = "x-budibase-app-id", APP_ID = "x-budibase-app-id",
SESSION_ID = "x-budibase-session-id",
TYPE = "x-budibase-type", TYPE = "x-budibase-type",
PREVIEW_ROLE = "x-budibase-role", PREVIEW_ROLE = "x-budibase-role",
TENANT_ID = "x-budibase-tenant-id", TENANT_ID = "x-budibase-tenant-id",

View File

@ -18,7 +18,9 @@ export const initWebsocket = () => {
} }
// Initialise connection // Initialise connection
socket = createWebsocket("/socket/client", false) socket = createWebsocket("/socket/client", {
heartbeat: false,
})
// Event handlers // Event handlers
socket.on("plugin-update", data => { socket.on("plugin-update", data => {

View File

@ -1,3 +1,4 @@
import { Helpers } from "@budibase/bbui"
import { ApiVersion } from "../constants" import { ApiVersion } from "../constants"
import { buildAnalyticsEndpoints } from "./analytics" import { buildAnalyticsEndpoints } from "./analytics"
import { buildAppEndpoints } from "./app" import { buildAppEndpoints } from "./app"
@ -30,6 +31,14 @@ import { buildEnvironmentVariableEndpoints } from "./environmentVariables"
import { buildEventEndpoints } from "./events" import { buildEventEndpoints } from "./events"
import { buildAuditLogsEndpoints } from "./auditLogs" 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 = { const defaultAPIClientConfig = {
/** /**
* Certain definitions can't change at runtime for client apps, such as the * Certain definitions can't change at runtime for client apps, such as the
@ -116,6 +125,7 @@ export const createAPIClient = config => {
// Build headers // Build headers
let headers = { Accept: "application/json" } let headers = { Accept: "application/json" }
headers["x-budibase-session-id"] = APISessionID
if (!external) { if (!external) {
headers["x-budibase-api-version"] = ApiVersion headers["x-budibase-api-version"] = ApiVersion
} }

View File

@ -26,15 +26,15 @@ export const createGridWebsocket = context => {
}) })
// User events // User events
socket.on(SocketEvent.UserUpdate, user => { socket.onOther(SocketEvent.UserUpdate, user => {
users.actions.updateUser(user) users.actions.updateUser(user)
}) })
socket.on(SocketEvent.UserDisconnect, user => { socket.onOther(SocketEvent.UserDisconnect, user => {
users.actions.removeUser(user) users.actions.removeUser(user)
}) })
// Row events // Row events
socket.on(GridSocketEvent.RowChange, async data => { socket.onOther(GridSocketEvent.RowChange, async data => {
if (data.id) { if (data.id) {
rows.actions.replaceRow(data.id, data.row) rows.actions.replaceRow(data.id, data.row)
} else if (data.row.id) { } else if (data.row.id) {
@ -44,7 +44,7 @@ export const createGridWebsocket = context => {
}) })
// Table events // 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 // Only update table if one exists. If the table was deleted then we don't
// want to know - let the builder navigate away // want to know - let the builder navigate away
if (data.table) { if (data.table) {

View File

@ -1,17 +1,26 @@
import { io } from "socket.io-client" import { io } from "socket.io-client"
import { SocketEvent, SocketSessionTTL } from "@budibase/shared-core" 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) { if (!path) {
throw "A websocket path must be provided" throw "A websocket path must be provided"
} }
const { heartbeat } = {
...DefaultOptions,
...options,
}
// Determine connection info // Determine connection info
const tls = location.protocol === "https:" const tls = location.protocol === "https:"
const proto = tls ? "wss:" : "ws:" const proto = tls ? "wss:" : "ws:"
const host = location.hostname const host = location.hostname
const port = location.port || (tls ? 443 : 80) const port = location.port || (tls ? 443 : 80)
const socket = io(`${proto}//${host}:${port}`, { let socket = io(`${proto}//${host}:${port}`, {
path, path,
// Cap reconnection attempts to 3 (total of 15 seconds before giving up) // Cap reconnection attempts to 3 (total of 15 seconds before giving up)
reconnectionAttempts: 3, reconnectionAttempts: 3,
@ -37,5 +46,17 @@ export const createWebsocket = (path, heartbeat = true) => {
clearInterval(interval) 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 return socket
} }

View File

@ -46,29 +46,32 @@ export default class BuilderSocket extends BaseSocket {
} }
emitTableUpdate(ctx: any, table: Table) { emitTableUpdate(ctx: any, table: Table) {
this.io this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.TableChange, {
.in(ctx.appId) id: table._id,
.emit(BuilderSocketEvent.TableChange, { id: table._id, table }) table,
gridSocket?.emitTableUpdate(table) })
gridSocket?.emitTableUpdate(ctx, table)
} }
emitTableDeletion(ctx: any, id: string) { emitTableDeletion(ctx: any, id: string) {
this.io this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.TableChange, {
.in(ctx.appId) id,
.emit(BuilderSocketEvent.TableChange, { id, table: null }) table: null,
gridSocket?.emitTableDeletion(id) })
gridSocket?.emitTableDeletion(ctx, id)
} }
emitDatasourceUpdate(ctx: any, datasource: Datasource) { emitDatasourceUpdate(ctx: any, datasource: Datasource) {
this.io.in(ctx.appId).emit(BuilderSocketEvent.DatasourceChange, { this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.DatasourceChange, {
id: datasource._id, id: datasource._id,
datasource, datasource,
}) })
} }
emitDatasourceDeletion(ctx: any, id: string) { emitDatasourceDeletion(ctx: any, id: string) {
this.io this.emitToRoom(ctx, ctx.appId, BuilderSocketEvent.DatasourceChange, {
.in(ctx.appId) id,
.emit(BuilderSocketEvent.DatasourceChange, { id, datasource: null }) datasource: null,
})
} }
} }

View File

@ -31,21 +31,25 @@ export default class GridSocket extends BaseSocket {
emitRowUpdate(ctx: any, row: Row) { emitRowUpdate(ctx: any, row: Row) {
const tableId = getTableId(ctx) 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) { emitRowDeletion(ctx: any, id: string) {
const tableId = getTableId(ctx) 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) { emitTableUpdate(ctx: any, table: Table) {
this.io this.emitToRoom(ctx, table._id!, GridSocketEvent.TableChange, {
.in(table._id!) id: table._id,
.emit(GridSocketEvent.TableChange, { id: table._id, table }) table,
})
} }
emitTableDeletion(id: string) { emitTableDeletion(ctx: any, id: string) {
this.io.in(id).emit(GridSocketEvent.TableChange, { id, table: null }) this.emitToRoom(ctx, id, GridSocketEvent.TableChange, { id, table: null })
} }
} }

View File

@ -3,7 +3,7 @@ import http from "http"
import Koa from "koa" import Koa from "koa"
import Cookies from "cookies" import Cookies from "cookies"
import { userAgent } from "koa-useragent" 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 currentApp from "../middleware/currentapp"
import { createAdapter } from "@socket.io/redis-adapter" import { createAdapter } from "@socket.io/redis-adapter"
import { Socket } from "socket.io" import { Socket } from "socket.io"
@ -271,4 +271,13 @@ export class BaseSocket {
emit(event: string, payload: any) { emit(event: string, payload: any) {
this.io.sockets.emit(event, payload) 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],
})
}
} }