diff --git a/packages/backend-core/src/cloud/accounts.js b/packages/backend-core/src/cloud/accounts.js
index b2e8817ad6..5730bc67a5 100644
--- a/packages/backend-core/src/cloud/accounts.js
+++ b/packages/backend-core/src/cloud/accounts.js
@@ -22,3 +22,18 @@ exports.getAccount = async email => {
return json[0]
}
+
+exports.getStatus = async () => {
+ const response = await api.get(`/api/status`, {
+ headers: {
+ [Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
+ },
+ })
+ const json = await response.json()
+
+ if (response.status !== 200) {
+ throw new Error(`Error getting status`)
+ }
+
+ return json
+}
diff --git a/packages/bbui/src/Banner/Banner.svelte b/packages/bbui/src/Banner/Banner.svelte
index f28ee09d9c..f41fb5f803 100644
--- a/packages/bbui/src/Banner/Banner.svelte
+++ b/packages/bbui/src/Banner/Banner.svelte
@@ -57,3 +57,10 @@
{/if}
+
+
diff --git a/packages/bbui/src/Banner/BannerDisplay.svelte b/packages/bbui/src/Banner/BannerDisplay.svelte
new file mode 100644
index 0000000000..aad742b1bd
--- /dev/null
+++ b/packages/bbui/src/Banner/BannerDisplay.svelte
@@ -0,0 +1,31 @@
+
+
+
+
+ {#if $banner.message}
+
+
+ {$banner.message}
+
+
+ {/if}
+
+
+
+
diff --git a/packages/bbui/src/Stores/banner.js b/packages/bbui/src/Stores/banner.js
new file mode 100644
index 0000000000..81a9ee2204
--- /dev/null
+++ b/packages/bbui/src/Stores/banner.js
@@ -0,0 +1,37 @@
+import { writable } from "svelte/store"
+
+export function createBannerStore() {
+ const DEFAULT_CONFIG = {}
+
+ const banner = writable(DEFAULT_CONFIG)
+
+ const show = async (
+ // eslint-disable-next-line
+ config = { message, type, extraButtonText, extraButtonAction, onChange }
+ ) => {
+ banner.update(store => {
+ return {
+ ...store,
+ ...config,
+ }
+ })
+ }
+
+ const showStatus = async () => {
+ const config = {
+ message: "Some systems are experiencing issues",
+ type: "negative",
+ extraButtonText: "View Status",
+ extraButtonAction: () => window.open("https://status.budibase.com/"),
+ }
+
+ await show(config)
+ }
+
+ return {
+ subscribe: banner.subscribe,
+ showStatus,
+ }
+}
+
+export const banner = createBannerStore()
diff --git a/packages/bbui/src/index.js b/packages/bbui/src/index.js
index d3bc11cf9d..2b16f32b84 100644
--- a/packages/bbui/src/index.js
+++ b/packages/bbui/src/index.js
@@ -60,6 +60,7 @@ export { default as StatusLight } from "./StatusLight/StatusLight.svelte"
export { default as ColorPicker } from "./ColorPicker/ColorPicker.svelte"
export { default as InlineAlert } from "./InlineAlert/InlineAlert.svelte"
export { default as Banner } from "./Banner/Banner.svelte"
+export { default as BannerDisplay } from "./Banner/BannerDisplay.svelte"
export { default as MarkdownEditor } from "./Markdown/MarkdownEditor.svelte"
export { default as MarkdownViewer } from "./Markdown/MarkdownViewer.svelte"
export { default as RichTextField } from "./Form/RichTextField.svelte"
@@ -84,6 +85,7 @@ export { default as clickOutside } from "./Actions/click_outside"
// Stores
export { notifications, createNotificationStore } from "./Stores/notifications"
+export { banner } from "./Stores/banner"
// Helpers
export * as Helpers from "./helpers"
diff --git a/packages/builder/src/App.svelte b/packages/builder/src/App.svelte
index 60051ea043..0fb0fe59d5 100644
--- a/packages/builder/src/App.svelte
+++ b/packages/builder/src/App.svelte
@@ -1,13 +1,16 @@
+
+
+
diff --git a/packages/builder/src/stores/portal/admin.js b/packages/builder/src/stores/portal/admin.js
index dc68c43cc5..088d396291 100644
--- a/packages/builder/src/stores/portal/admin.js
+++ b/packages/builder/src/stores/portal/admin.js
@@ -1,6 +1,7 @@
import { writable, get } from "svelte/store"
import { API } from "api"
import { auth } from "stores/portal"
+import { banner } from "@budibase/bbui"
export function createAdminStore() {
const DEFAULT_CONFIG = {
@@ -30,6 +31,13 @@ export function createAdminStore() {
x => x?.checked
).length
await getEnvironment()
+
+ // enable system status checks in the cloud
+ if (get(admin).cloud) {
+ await getSystemStatus()
+ checkStatus()
+ }
+
admin.update(store => {
store.loaded = true
store.checklist = checklist
@@ -58,6 +66,21 @@ export function createAdminStore() {
})
}
+ const checkStatus = async () => {
+ const health = get(admin)?.status?.health
+ if (!health?.passing) {
+ await banner.showStatus()
+ }
+ }
+
+ async function getSystemStatus() {
+ const status = await API.getSystemStatus()
+ admin.update(store => {
+ store.status = status
+ return store
+ })
+ }
+
function unload() {
admin.update(store => {
store.loaded = false
diff --git a/packages/frontend-core/src/api/other.js b/packages/frontend-core/src/api/other.js
index 71ea19ccce..6903a120db 100644
--- a/packages/frontend-core/src/api/other.js
+++ b/packages/frontend-core/src/api/other.js
@@ -17,6 +17,15 @@ export const buildOtherEndpoints = API => ({
})
},
+ /**
+ * Gets the current system status.
+ */
+ getSystemStatus: async () => {
+ return await API.get({
+ url: "/api/system/status",
+ })
+ },
+
/**
* Gets the list of available integrations.
*/
diff --git a/packages/worker/src/api/controllers/global/self.js b/packages/worker/src/api/controllers/global/self.js
index ea29c496e8..da4d82c46b 100644
--- a/packages/worker/src/api/controllers/global/self.js
+++ b/packages/worker/src/api/controllers/global/self.js
@@ -1,10 +1,20 @@
-const { getGlobalDB, getTenantId } = require("@budibase/backend-core/tenancy")
+const {
+ getGlobalDB,
+ getTenantId,
+ isUserInAppTenant,
+} = require("@budibase/backend-core/tenancy")
const { generateDevInfoID, SEPARATOR } = require("@budibase/backend-core/db")
const { user: userCache } = require("@budibase/backend-core/cache")
-const { hash, platformLogout } = require("@budibase/backend-core/utils")
+const {
+ hash,
+ platformLogout,
+ getCookie,
+ clearCookie,
+} = require("@budibase/backend-core/utils")
const { encrypt } = require("@budibase/backend-core/encryption")
const { newid } = require("@budibase/backend-core/utils")
const { getUser } = require("../../utilities")
+const { Cookies } = require("@budibase/backend-core/constants")
function newApiKey() {
return encrypt(`${getTenantId()}${SEPARATOR}${newid()}`)
@@ -48,6 +58,16 @@ exports.fetchAPIKey = async ctx => {
ctx.body = cleanupDevInfo(devInfo)
}
+const checkCurrentApp = ctx => {
+ const appCookie = getCookie(ctx, Cookies.CurrentApp)
+ if (appCookie && !isUserInAppTenant(appCookie.appId)) {
+ // there is a currentapp cookie from another tenant
+ // remove the cookie as this is incompatible with the builder
+ // due to builder and admin permissions being removed
+ clearCookie(ctx, Cookies.CurrentApp)
+ }
+}
+
exports.getSelf = async ctx => {
if (!ctx.user) {
ctx.throw(403, "User not logged in")
@@ -56,6 +76,9 @@ exports.getSelf = async ctx => {
ctx.params = {
id: userId,
}
+
+ checkCurrentApp(ctx)
+
// get the main body of the user
ctx.body = await getUser(userId)
// forward session information not found in db
diff --git a/packages/worker/src/api/controllers/system/status.js b/packages/worker/src/api/controllers/system/status.js
new file mode 100644
index 0000000000..9d2bd6ecda
--- /dev/null
+++ b/packages/worker/src/api/controllers/system/status.js
@@ -0,0 +1,15 @@
+const accounts = require("@budibase/backend-core/accounts")
+const env = require("../../../environment")
+
+exports.fetch = async ctx => {
+ if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) {
+ const status = await accounts.getStatus()
+ ctx.body = status
+ } else {
+ ctx.body = {
+ health: {
+ passing: true,
+ },
+ }
+ }
+}
diff --git a/packages/worker/src/api/index.js b/packages/worker/src/api/index.js
index 76e45991ff..8be3c17866 100644
--- a/packages/worker/src/api/index.js
+++ b/packages/worker/src/api/index.js
@@ -39,7 +39,6 @@ const PUBLIC_ENDPOINTS = [
method: "GET",
},
{
- // TODO: Add an provisioning API key to this endpoint in the cloud
route: "/api/global/users/init",
method: "POST",
},
@@ -51,6 +50,10 @@ const PUBLIC_ENDPOINTS = [
route: "api/system/environment",
method: "GET",
},
+ {
+ route: "api/system/status",
+ method: "GET",
+ },
{
route: "/api/global/users/tenant/:id",
method: "GET",
diff --git a/packages/worker/src/api/routes/index.js b/packages/worker/src/api/routes/index.js
index b1c2435252..bc6e7842da 100644
--- a/packages/worker/src/api/routes/index.js
+++ b/packages/worker/src/api/routes/index.js
@@ -8,6 +8,7 @@ const roleRoutes = require("./global/roles")
const sessionRoutes = require("./global/sessions")
const environmentRoutes = require("./system/environment")
const tenantsRoutes = require("./system/tenants")
+const statusRoutes = require("./system/status")
const selfRoutes = require("./global/self")
exports.routes = [
@@ -21,5 +22,6 @@ exports.routes = [
sessionRoutes,
roleRoutes,
environmentRoutes,
+ statusRoutes,
selfRoutes,
]
diff --git a/packages/worker/src/api/routes/system/status.js b/packages/worker/src/api/routes/system/status.js
new file mode 100644
index 0000000000..a39801375b
--- /dev/null
+++ b/packages/worker/src/api/routes/system/status.js
@@ -0,0 +1,8 @@
+const Router = require("@koa/router")
+const controller = require("../../controllers/system/status")
+
+const router = Router()
+
+router.get("/api/system/status", controller.fetch)
+
+module.exports = router