diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml
index 588f0c54ae..aaee3923ef 100644
--- a/.github/workflows/release.yml
+++ b/.github/workflows/release.yml
@@ -1,6 +1,6 @@
name: Budibase Release
-on:
+on:
push:
branches:
- master
@@ -9,20 +9,20 @@ env:
POSTHOG_TOKEN: ${{ secrets.POSTHOG_TOKEN }}
POSTHOG_URL: ${{ secrets.POSTHOG_URL }}
SENTRY_DSN: ${{ secrets.SENTRY_DSN }}
-
+
jobs:
release:
- runs-on: ubuntu-latest
+ runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- uses: actions/setup-node@v1
with:
node-version: 12.x
- - run: yarn
- - run: yarn bootstrap
- - run: yarn lint
- - run: yarn build
+ - run: yarn
+ - run: yarn bootstrap
+ - run: yarn lint
+ - run: yarn build
- run: yarn test
- name: Configure AWS Credentials
@@ -35,19 +35,19 @@ jobs:
- name: Publish budibase packages to NPM
env:
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
- run: |
+ run: |
# setup the username and email. I tend to use 'GitHub Actions Bot' with no email by default
git config user.name "Budibase Release Bot"
git config user.email "<>"
- echo //registry.npmjs.org/:_authToken=${NPM_TOKEN} >> .npmrc
+ echo //registry.npmjs.org/:_authToken=${NPM_TOKEN} >> .npmrc
yarn release
- - name: Get Previous tag
- id: previoustag
- uses: "WyriHaximus/github-action-get-previous-tag@v1"
+ - name: 'Get Previous tag'
+ id: previoustag
+ uses: "WyriHaximus/github-action-get-previous-tag@v1"
- name: Build/release Docker images
- run: |
+ run: |
docker login -u $DOCKER_USER -p $DOCKER_PASSWORD
yarn build
yarn build:docker
@@ -68,4 +68,3 @@ jobs:
charts_dir: docs
env:
CR_TOKEN: "${{ secrets.GITHUB_TOKEN }}"
-
\ No newline at end of file
diff --git a/LICENSE b/LICENSE
index a6bd926020..9c0a8c22a0 100644
--- a/LICENSE
+++ b/LICENSE
@@ -5,8 +5,7 @@ Each Budibase package has its own license:
builder: GPLv3
server: GPLv3
client: MPLv2.0
-standard-components: MPLv2.0
-You can consider Budibase to be GPLv3 licensed.
+You can consider Budibase to be GPLv3 licensed.
The apps that you build with Budibase do not fall under GPLv3 - hence why our components and client library are licensed differently.
diff --git a/SECURITY.md b/SECURITY.md
new file mode 100644
index 0000000000..e414f48cb8
--- /dev/null
+++ b/SECURITY.md
@@ -0,0 +1,15 @@
+# Security Policy
+
+## Versions
+
+As an open source product, we will only patch the latest major version for security vulnerabilities. Previous versions of budibase will not be retroactively patched.
+
+## Disclosing
+
+You can get in touch with us regarding a vulnerability via email at community@budibase.com.
+
+You can also disclose via huntr.dev. If you believe you have found a vulnerability, please disclose it on huntr and let us know.
+
+https://huntr.dev/bounties/disclose
+
+This will enable us to review the vulnerability and potentially reward you for your work.
diff --git a/hosting/docker-compose.yaml b/hosting/docker-compose.yaml
index 9cdf2b2114..18b93fdf61 100644
--- a/hosting/docker-compose.yaml
+++ b/hosting/docker-compose.yaml
@@ -51,6 +51,7 @@ services:
INTERNAL_API_KEY: ${INTERNAL_API_KEY}
REDIS_URL: redis-service:6379
REDIS_PASSWORD: ${REDIS_PASSWORD}
+ ACCOUNT_PORTAL_URL: https://portal.budi.live
volumes:
- ./logs:/logs
depends_on:
@@ -107,7 +108,7 @@ services:
depends_on:
- couchdb-service
command: ["sh","-c","sleep 10 && $${PUT_CALL}/_users && $${PUT_CALL}/_replicator; fg;"]
-
+
redis-service:
restart: always
image: redis
@@ -116,9 +117,11 @@ services:
- "${REDIS_PORT}:6379"
volumes:
- redis_data:/data
-
+
watchtower-service:
image: containrrr/watchtower
+ ports:
+ - "${WATCHTOWER_PORT}:8080"
volumes:
- /var/run/docker.sock:/var/run/docker.sock
command: --debug --http-api-update bbapps bbworker
@@ -128,8 +131,6 @@ services:
- WATCHTOWER_CLEANUP=true
labels:
- "com.centurylinklabs.watchtower.enable=false"
- ports:
- - 6161:8080
volumes:
diff --git a/hosting/hosting.properties b/hosting/hosting.properties
index d11972bc4b..c8e2f5c606 100644
--- a/hosting/hosting.properties
+++ b/hosting/hosting.properties
@@ -17,4 +17,5 @@ WORKER_PORT=4003
MINIO_PORT=4004
COUCH_DB_PORT=4005
REDIS_PORT=6379
+WATCHTOWER_PORT=6161
BUDIBASE_ENVIRONMENT=PRODUCTION
diff --git a/hosting/kubernetes/budibase/templates/worker-service-deployment.yaml b/hosting/kubernetes/budibase/templates/worker-service-deployment.yaml
index 703d59c075..6c165872c8 100644
--- a/hosting/kubernetes/budibase/templates/worker-service-deployment.yaml
+++ b/hosting/kubernetes/budibase/templates/worker-service-deployment.yaml
@@ -87,6 +87,8 @@ spec:
{{ end }}
- name: SELF_HOSTED
value: {{ .Values.globals.selfHosted | quote }}
+ - name: ACCOUNT_PORTAL_URL
+ value: {{ .Values.globals.accountPortalUrl | quote }}
image: budibase/worker
imagePullPolicy: Always
name: bbworker
diff --git a/hosting/kubernetes/budibase/values.yaml b/hosting/kubernetes/budibase/values.yaml
index 30594f95e3..1113842c8b 100644
--- a/hosting/kubernetes/budibase/values.yaml
+++ b/hosting/kubernetes/budibase/values.yaml
@@ -44,7 +44,7 @@ ingress:
nginx: true
certificateArn: ""
className: ""
- annotations:
+ annotations:
kubernetes.io/ingress.class: nginx
hosts:
- host: # change if using custom domain
@@ -55,7 +55,7 @@ ingress:
service:
name: proxy-service
port:
- number: 10000
+ number: 10000
resources: {}
# We usually recommend not to specify default resources and to leave this as a conscious
@@ -86,9 +86,10 @@ globals:
budibaseEnv: PRODUCTION
enableAnalytics: false
posthogToken: ""
- sentryDSN: ""
+ sentryDSN: ""
logLevel: info
selfHosted: 1
+ accountPortalUrL: ""
createSecrets: true # creates an internal API key, JWT secrets and redis password for you
# if createSecrets is set to false, you can hard-code your secrets here
@@ -120,7 +121,7 @@ services:
password: "" # only change if pointing to existing couch server
port: 5984
storage: 100Mi
-
+
redis:
enabled: true # disable if using external redis
port: 6379
@@ -128,15 +129,15 @@ services:
url: "" # only change if pointing to existing redis cluster and enabled: false
password: "budibase" # recommended to override if using built-in redis
storage: 100Mi
-
+
objectStore:
minio: true
browser: true
port: 9000
replicaCount: 1
accessKey: "" # AWS_ACCESS_KEY if using S3 or existing minio access key
- secretKey: "" # AWS_SECRET_ACCESS_KEY if using S3 or existing minio secret
- region: "" # AWS_REGION if using S3 or existing minio secret
- url: "" # only change if pointing to existing minio cluster and minio: false
+ secretKey: "" # AWS_SECRET_ACCESS_KEY if using S3 or existing minio secret
+ region: "" # AWS_REGION if using S3 or existing minio secret
+ url: "" # only change if pointing to existing minio cluster and minio: false
storage: 100Mi
diff --git a/hosting/kubernetes/envoy/envoy.yaml b/hosting/kubernetes/envoy/envoy.yaml
index 0e7859d887..c36f04936f 100644
--- a/hosting/kubernetes/envoy/envoy.yaml
+++ b/hosting/kubernetes/envoy/envoy.yaml
@@ -28,27 +28,35 @@ static_resources:
- match: { prefix: "/builder" }
route:
cluster: app-service
-
+
- match: { prefix: "/app_" }
route:
cluster: app-service
- # special case for worker admin API
+ # special cases for worker admin (deprecated), global and system API
+ - match: { prefix: "/api/global/" }
+ route:
+ cluster: worker-service
+
- match: { prefix: "/api/admin/" }
route:
cluster: worker-service
+ - match: { prefix: "/api/system/" }
+ route:
+ cluster: worker-service
+
- match: { path: "/" }
route:
cluster: app-service
- # special case for when API requests are made, can just forward, not to minio
+ # special case for when API requests are made, can just forward, not to minio
- match: { prefix: "/api/" }
route:
cluster: app-service
- match: { prefix: "/worker/" }
- route:
+ route:
cluster: worker-service
prefix_rewrite: "/"
@@ -77,7 +85,7 @@ static_resources:
- lb_endpoints:
- endpoint:
address:
- socket_address:
+ socket_address:
address: app-service.budibase.svc.cluster.local
port_value: 4002
@@ -105,7 +113,7 @@ static_resources:
- lb_endpoints:
- endpoint:
address:
- socket_address:
+ socket_address:
address: worker-service.budibase.svc.cluster.local
port_value: 4001
diff --git a/lerna.json b/lerna.json
index 8188c1c229..6171a81f87 100644
--- a/lerna.json
+++ b/lerna.json
@@ -1,5 +1,5 @@
{
- "version": "0.9.105-alpha.31",
+ "version": "0.9.125-alpha.13",
"npmClient": "yarn",
"packages": [
"packages/*"
diff --git a/package.json b/package.json
index 05c69e54dc..f87c3715aa 100644
--- a/package.json
+++ b/package.json
@@ -20,7 +20,6 @@
"setup": "node ./hosting/scripts/setup.js && yarn && yarn bootstrap && yarn build && yarn dev",
"bootstrap": "lerna link && lerna bootstrap",
"build": "lerna run build",
- "initialise": "lerna run initialise",
"publishdev": "lerna run publishdev",
"publishnpm": "yarn build && lerna publish --force-publish",
"release": "yarn build && lerna publish patch --yes --force-publish",
@@ -48,6 +47,8 @@
"release:helm": "./scripts/release_helm_chart.sh",
"multi:enable": "lerna run multi:enable",
"multi:disable": "lerna run multi:disable",
+ "selfhost:enable": "lerna run selfhost:enable",
+ "selfhost:disable": "lerna run selfhost:disable",
"postinstall": "husky install"
}
}
diff --git a/packages/auth/package.json b/packages/auth/package.json
index ad5afd5691..4fc09157b0 100644
--- a/packages/auth/package.json
+++ b/packages/auth/package.json
@@ -1,6 +1,6 @@
{
"name": "@budibase/auth",
- "version": "0.9.105-alpha.31",
+ "version": "0.9.125-alpha.13",
"description": "Authentication middlewares for budibase builder and apps",
"main": "src/index.js",
"author": "Budibase",
diff --git a/packages/auth/src/cache/user.js b/packages/auth/src/cache/user.js
index 4a19da489f..2b2693ca01 100644
--- a/packages/auth/src/cache/user.js
+++ b/packages/auth/src/cache/user.js
@@ -3,7 +3,29 @@ const { getTenantId, lookupTenantId, getGlobalDB } = require("../tenancy")
const EXPIRY_SECONDS = 3600
-exports.getUser = async (userId, tenantId = null) => {
+/**
+ * The default populate user function
+ */
+const populateFromDB = async (userId, tenantId) => {
+ const user = await getGlobalDB(tenantId).get(userId)
+ user.budibaseAccess = true
+ return user
+}
+
+/**
+ * Get the requested user by id.
+ * Use redis cache to first read the user.
+ * If not present fallback to loading the user directly and re-caching.
+ * @param {*} userId the id of the user to get
+ * @param {*} tenantId the tenant of the user to get
+ * @param {*} populateUser function to provide the user for re-caching. default to couch db
+ * @returns
+ */
+exports.getUser = async (
+ userId,
+ tenantId = null,
+ populateUser = populateFromDB
+) => {
if (!tenantId) {
try {
tenantId = getTenantId()
@@ -15,9 +37,13 @@ exports.getUser = async (userId, tenantId = null) => {
// try cache
let user = await client.get(userId)
if (!user) {
- user = await getGlobalDB(tenantId).get(userId)
+ user = await populateUser(userId, tenantId)
client.store(userId, user, EXPIRY_SECONDS)
}
+ if (user && !user.tenantId && tenantId) {
+ // make sure the tenant ID is always correct/set
+ user.tenantId = tenantId
+ }
return user
}
diff --git a/packages/auth/src/db/utils.js b/packages/auth/src/db/utils.js
index 4cd29c9bc8..a5d7c1f100 100644
--- a/packages/auth/src/db/utils.js
+++ b/packages/auth/src/db/utils.js
@@ -35,10 +35,6 @@ exports.APP_PREFIX = DocumentTypes.APP + SEPARATOR
exports.APP_DEV = exports.APP_DEV_PREFIX = DocumentTypes.APP_DEV + SEPARATOR
exports.SEPARATOR = SEPARATOR
-function isDevApp(app) {
- return app.appId.startsWith(exports.APP_DEV_PREFIX)
-}
-
/**
* If creating DB allDocs/query params with only a single top level ID this can be used, this
* is usually the case as most of our docs are top level e.g. tables, automations, users and so on.
@@ -62,6 +58,35 @@ function getDocParams(docType, docId = null, otherProps = {}) {
}
}
+exports.isDevAppID = appId => {
+ return appId.startsWith(exports.APP_DEV_PREFIX)
+}
+
+exports.isProdAppID = appId => {
+ return appId.startsWith(exports.APP_PREFIX) && !exports.isDevAppID(appId)
+}
+
+function isDevApp(app) {
+ return exports.isDevAppID(app.appId)
+}
+
+/**
+ * Given an app ID this will attempt to retrieve the tenant ID from it.
+ * @return {null|string} The tenant ID found within the app ID.
+ */
+exports.getTenantIDFromAppID = appId => {
+ const split = appId.split(SEPARATOR)
+ const hasDev = split[1] === DocumentTypes.DEV
+ if ((hasDev && split.length === 3) || (!hasDev && split.length === 2)) {
+ return null
+ }
+ if (hasDev) {
+ return split[2]
+ } else {
+ return split[1]
+ }
+}
+
/**
* Generates a new workspace ID.
* @returns {string} The new workspace ID which the workspace doc can be stored under.
diff --git a/packages/auth/src/index.js b/packages/auth/src/index.js
index 5421dea214..569456ea10 100644
--- a/packages/auth/src/index.js
+++ b/packages/auth/src/index.js
@@ -11,6 +11,7 @@ const {
oidc,
auditLog,
tenancy,
+ appTenancy,
} = require("./middleware")
const { setDB } = require("./db")
const userCache = require("./cache/user")
@@ -57,6 +58,7 @@ module.exports = {
oidc,
jwt: require("jsonwebtoken"),
buildTenancyMiddleware: tenancy,
+ buildAppTenancyMiddleware: appTenancy,
auditLog,
},
cache: {
diff --git a/packages/auth/src/middleware/appTenancy.js b/packages/auth/src/middleware/appTenancy.js
new file mode 100644
index 0000000000..30fc4f7453
--- /dev/null
+++ b/packages/auth/src/middleware/appTenancy.js
@@ -0,0 +1,25 @@
+const {
+ isMultiTenant,
+ updateTenantId,
+ isTenantIdSet,
+ DEFAULT_TENANT_ID,
+} = require("../tenancy")
+const ContextFactory = require("../tenancy/FunctionContext")
+const { getTenantIDFromAppID } = require("../db/utils")
+
+module.exports = () => {
+ return ContextFactory.getMiddleware(ctx => {
+ // if not in multi-tenancy mode make sure its default and exit
+ if (!isMultiTenant()) {
+ updateTenantId(DEFAULT_TENANT_ID)
+ return
+ }
+ // if tenant ID already set no need to continue
+ if (isTenantIdSet()) {
+ return
+ }
+ const appId = ctx.appId ? ctx.appId : ctx.user ? ctx.user.appId : null
+ const tenantId = getTenantIDFromAppID(appId) || DEFAULT_TENANT_ID
+ updateTenantId(tenantId)
+ })
+}
diff --git a/packages/auth/src/middleware/authenticated.js b/packages/auth/src/middleware/authenticated.js
index e3705a9a24..944f3ee9d9 100644
--- a/packages/auth/src/middleware/authenticated.js
+++ b/packages/auth/src/middleware/authenticated.js
@@ -21,7 +21,10 @@ function finalise(
* The tenancy modules should not be used here and it should be assumed that the tenancy context
* has not yet been populated.
*/
-module.exports = (noAuthPatterns = [], opts = { publicAllowed: false }) => {
+module.exports = (
+ noAuthPatterns = [],
+ opts = { publicAllowed: false, populateUser: null }
+) => {
const noAuthOptions = noAuthPatterns ? buildMatcherRegex(noAuthPatterns) : []
return async (ctx, next) => {
let publicEndpoint = false
@@ -46,7 +49,15 @@ module.exports = (noAuthPatterns = [], opts = { publicAllowed: false }) => {
error = "No session found"
} else {
try {
- user = await getUser(userId, session.tenantId)
+ if (opts && opts.populateUser) {
+ user = await getUser(
+ userId,
+ session.tenantId,
+ opts.populateUser(ctx)
+ )
+ } else {
+ user = await getUser(userId, session.tenantId)
+ }
delete user.password
authenticated = true
} catch (err) {
diff --git a/packages/auth/src/middleware/index.js b/packages/auth/src/middleware/index.js
index 689859a139..059f20af8b 100644
--- a/packages/auth/src/middleware/index.js
+++ b/packages/auth/src/middleware/index.js
@@ -5,6 +5,7 @@ const oidc = require("./passport/oidc")
const authenticated = require("./authenticated")
const auditLog = require("./auditLog")
const tenancy = require("./tenancy")
+const appTenancy = require("./appTenancy")
module.exports = {
google,
@@ -14,4 +15,5 @@ module.exports = {
authenticated,
auditLog,
tenancy,
+ appTenancy,
}
diff --git a/packages/auth/src/middleware/tenancy.js b/packages/auth/src/middleware/tenancy.js
index b80b9a6763..adfd36a503 100644
--- a/packages/auth/src/middleware/tenancy.js
+++ b/packages/auth/src/middleware/tenancy.js
@@ -2,12 +2,17 @@ const { setTenantId } = require("../tenancy")
const ContextFactory = require("../tenancy/FunctionContext")
const { buildMatcherRegex, matches } = require("./matchers")
-module.exports = (allowQueryStringPatterns, noTenancyPatterns) => {
+module.exports = (
+ allowQueryStringPatterns,
+ noTenancyPatterns,
+ opts = { noTenancyRequired: false }
+) => {
const allowQsOptions = buildMatcherRegex(allowQueryStringPatterns)
const noTenancyOptions = buildMatcherRegex(noTenancyPatterns)
return ContextFactory.getMiddleware(ctx => {
- const allowNoTenant = !!matches(ctx, noTenancyOptions)
+ const allowNoTenant =
+ opts.noTenancyRequired || !!matches(ctx, noTenancyOptions)
const allowQs = !!matches(ctx, allowQsOptions)
setTenantId(ctx, { allowQs, allowNoTenant })
})
diff --git a/packages/auth/src/redis/index.js b/packages/auth/src/redis/index.js
index 4f2b5288ea..94b453c8f6 100644
--- a/packages/auth/src/redis/index.js
+++ b/packages/auth/src/redis/index.js
@@ -56,9 +56,13 @@ function init() {
if (CLIENT) {
CLIENT.disconnect()
}
- const { opts, host, port } = getRedisOptions(CLUSTERED)
+
+ const { redisProtocolUrl, opts, host, port } = getRedisOptions(CLUSTERED)
+
if (CLUSTERED) {
CLIENT = new Redis.Cluster([{ host, port }], opts)
+ } else if (redisProtocolUrl) {
+ CLIENT = new Redis(redisProtocolUrl)
} else {
CLIENT = new Redis(opts)
}
diff --git a/packages/auth/src/redis/utils.js b/packages/auth/src/redis/utils.js
index 415dcbf463..6befecd9ba 100644
--- a/packages/auth/src/redis/utils.js
+++ b/packages/auth/src/redis/utils.js
@@ -8,17 +8,27 @@ const REDIS_PASSWORD = !env.REDIS_PASSWORD ? "budibase" : env.REDIS_PASSWORD
exports.Databases = {
PW_RESETS: "pwReset",
+ VERIFICATIONS: "verification",
INVITATIONS: "invitation",
DEV_LOCKS: "devLocks",
DEBOUNCE: "debounce",
SESSIONS: "session",
USER_CACHE: "users",
+ FLAGS: "flags",
}
exports.SEPARATOR = SEPARATOR
exports.getRedisOptions = (clustered = false) => {
- const [host, port] = REDIS_URL.split(":")
+ const [host, port, ...rest] = REDIS_URL.split(":")
+
+ let redisProtocolUrl
+
+ // fully qualified redis URL
+ if (rest.length && /rediss?/.test(host)) {
+ redisProtocolUrl = REDIS_URL
+ }
+
const opts = {
connectTimeout: CONNECT_TIMEOUT_MS,
}
@@ -33,7 +43,7 @@ exports.getRedisOptions = (clustered = false) => {
opts.port = port
opts.password = REDIS_PASSWORD
}
- return { opts, host, port }
+ return { opts, host, port, redisProtocolUrl }
}
exports.addDbPrefix = (db, key) => {
diff --git a/packages/auth/src/tenancy/context.js b/packages/auth/src/tenancy/context.js
index f3f1f541e9..b1ef5a5807 100644
--- a/packages/auth/src/tenancy/context.js
+++ b/packages/auth/src/tenancy/context.js
@@ -21,9 +21,7 @@ exports.doInTenant = (tenantId, task) => {
cls.setOnContext(TENANT_ID, tenantId)
// invoke the task
- const result = task()
-
- return result
+ return task()
})
}
diff --git a/packages/bbui/package.json b/packages/bbui/package.json
index f6275b7fe3..6b4b21e61b 100644
--- a/packages/bbui/package.json
+++ b/packages/bbui/package.json
@@ -1,7 +1,7 @@
{
"name": "@budibase/bbui",
"description": "A UI solution used in the different Budibase projects.",
- "version": "0.9.105-alpha.31",
+ "version": "0.9.125-alpha.13",
"license": "AGPL-3.0",
"svelte": "src/index.js",
"module": "dist/bbui.es.js",
diff --git a/packages/bbui/src/ActionButton/ActionButton.svelte b/packages/bbui/src/ActionButton/ActionButton.svelte
index 83f71d385b..b518ac3d92 100644
--- a/packages/bbui/src/ActionButton/ActionButton.svelte
+++ b/packages/bbui/src/ActionButton/ActionButton.svelte
@@ -12,6 +12,7 @@
export let dataCy = null
export let size = "M"
export let active = false
+ export let fullWidth = false
function longPress(element) {
if (!longPressable) return
@@ -40,6 +41,7 @@
class:spectrum-ActionButton--quiet={quiet}
class:spectrum-ActionButton--emphasized={emphasized}
class:is-selected={selected}
+ class:fullWidth
class="spectrum-ActionButton spectrum-ActionButton--size{size}"
class:active
{disabled}
@@ -71,6 +73,9 @@
diff --git a/packages/bbui/src/Modal/ModalContent.svelte b/packages/bbui/src/Modal/ModalContent.svelte
index bba72e6093..678a813a61 100644
--- a/packages/bbui/src/Modal/ModalContent.svelte
+++ b/packages/bbui/src/Modal/ModalContent.svelte
@@ -46,8 +46,10 @@
{#if showDivider}
@@ -120,4 +122,9 @@
.close-icon :global(svg) {
margin-right: 0;
}
+
+ .header-spacing {
+ display: flex;
+ justify-content: space-between;
+ }
diff --git a/packages/bbui/src/ProgressCircle/ProgressCircle.svelte b/packages/bbui/src/ProgressCircle/ProgressCircle.svelte
index 9c8181ec7c..0428263346 100644
--- a/packages/bbui/src/ProgressCircle/ProgressCircle.svelte
+++ b/packages/bbui/src/ProgressCircle/ProgressCircle.svelte
@@ -13,7 +13,7 @@
}
}
- export let value = false
+ export let value = null
export let minValue = 0
export let maxValue = 100
@@ -42,7 +42,7 @@
diff --git a/packages/bbui/src/SideNavigation/Item.svelte b/packages/bbui/src/SideNavigation/Item.svelte
index f50270dfbd..dfebdb46a6 100644
--- a/packages/bbui/src/SideNavigation/Item.svelte
+++ b/packages/bbui/src/SideNavigation/Item.svelte
@@ -13,6 +13,7 @@
class="spectrum-SideNav-item"
class:is-selected={selected}
class:is-disabled={disabled}
+ on:click
>
{#if heading}
diff --git a/packages/bbui/src/Table/ArrayRenderer.svelte b/packages/bbui/src/Table/ArrayRenderer.svelte
new file mode 100644
index 0000000000..679973a03a
--- /dev/null
+++ b/packages/bbui/src/Table/ArrayRenderer.svelte
@@ -0,0 +1,17 @@
+
+
+{#each badges as badge}
+ {badge}
+{/each}
+{#if leftover}
+
+{leftover} more
+{/if}
diff --git a/packages/bbui/src/Table/CellRenderer.svelte b/packages/bbui/src/Table/CellRenderer.svelte
index 2d073b7782..d6a2f3196d 100644
--- a/packages/bbui/src/Table/CellRenderer.svelte
+++ b/packages/bbui/src/Table/CellRenderer.svelte
@@ -4,7 +4,7 @@
import DateTimeRenderer from "./DateTimeRenderer.svelte"
import RelationshipRenderer from "./RelationshipRenderer.svelte"
import AttachmentRenderer from "./AttachmentRenderer.svelte"
-
+ import ArrayRenderer from "./ArrayRenderer.svelte"
export let row
export let schema
export let value
@@ -19,6 +19,7 @@
options: StringRenderer,
number: StringRenderer,
longform: StringRenderer,
+ array: ArrayRenderer,
}
$: type = schema?.type ?? "string"
$: customRenderer = customRenderers?.find(x => x.column === schema?.name)
diff --git a/packages/bbui/src/Table/Table.svelte b/packages/bbui/src/Table/Table.svelte
index b6b61a8d04..11284b8917 100644
--- a/packages/bbui/src/Table/Table.svelte
+++ b/packages/bbui/src/Table/Table.svelte
@@ -3,6 +3,7 @@
import "@spectrum-css/table/dist/index-vars.css"
import CellRenderer from "./CellRenderer.svelte"
import SelectEditRenderer from "./SelectEditRenderer.svelte"
+ import { cloneDeep } from "lodash"
/**
* The expected schema is our normal couch schemas for our tables.
@@ -197,7 +198,7 @@
const editRow = (e, row) => {
e.stopPropagation()
- dispatch("editrow", row)
+ dispatch("editrow", cloneDeep(row))
}
const toggleSelectRow = row => {
diff --git a/packages/builder/assets/discord.svg b/packages/builder/assets/discord.svg
new file mode 100644
index 0000000000..3efe1ec110
--- /dev/null
+++ b/packages/builder/assets/discord.svg
@@ -0,0 +1,10 @@
+
diff --git a/packages/builder/assets/integromat.png b/packages/builder/assets/integromat.png
new file mode 100644
index 0000000000..1fbbe63e74
Binary files /dev/null and b/packages/builder/assets/integromat.png differ
diff --git a/packages/builder/assets/n8n.png b/packages/builder/assets/n8n.png
new file mode 100644
index 0000000000..b9dad93e5a
Binary files /dev/null and b/packages/builder/assets/n8n.png differ
diff --git a/packages/builder/assets/slack.svg b/packages/builder/assets/slack.svg
new file mode 100644
index 0000000000..d0a7c176f9
--- /dev/null
+++ b/packages/builder/assets/slack.svg
@@ -0,0 +1,33 @@
+
+
+
diff --git a/packages/builder/assets/zapier.png b/packages/builder/assets/zapier.png
new file mode 100644
index 0000000000..3805331440
Binary files /dev/null and b/packages/builder/assets/zapier.png differ
diff --git a/packages/builder/cypress/integration/createAutomation.spec.js b/packages/builder/cypress/integration/createAutomation.spec.js
index e82eeff670..0620a15e25 100644
--- a/packages/builder/cypress/integration/createAutomation.spec.js
+++ b/packages/builder/cypress/integration/createAutomation.spec.js
@@ -7,26 +7,42 @@ context("Create a automation", () => {
// https://on.cypress.io/interacting-with-elements
it("should create a automation", () => {
cy.createTestTableWithData()
-
+ cy.wait(2000)
cy.contains("Automate").click()
cy.get("[data-cy='new-screen'] > .spectrum-Icon").click()
- cy.get(".spectrum-Dialog-grid").within(() => {
+ cy.get(".modal-inner-wrapper").within(() => {
cy.get("input").type("Add Row")
+ cy.contains("Row Created").click()
+ cy.wait(500)
cy.get(".spectrum-Button--cta").click()
})
- // Add trigger
- cy.contains("Trigger").click()
- cy.contains("Row Created").click()
- cy.get(".setup").within(() => {
- cy.get(".spectrum-Picker-label").click()
- cy.contains("dog").click()
- })
+ // Setup trigger
+ cy.contains("Setup").click()
+ cy.get(".spectrum-Picker-label").click()
+ cy.wait(500)
+ cy.contains("dog").click()
// Create action
- cy.contains("Action").click()
- cy.contains("Create Row").click()
- cy.get(".setup").within(() => {
+ cy.contains("Add Action").click()
+ cy.get(".modal-inner-wrapper").within(() => {
+ cy.wait(1000)
+ cy.contains("Create Row").trigger('mouseover').click().click()
+ cy.get(".spectrum-Button--cta").click()
+ })
+ cy.contains("Setup").click()
+ cy.get(".spectrum-Picker-label").click()
+ cy.contains("dog").click()
+ cy.get(".spectrum-Textfield-input")
+ .first()
+ .type("goodboy")
+ cy.get(".spectrum-Textfield-input")
+ .eq(1)
+ .type("11")
+
+ cy.contains("Run test").click()
+ cy.get(".modal-inner-wrapper").within(() => {
+ cy.wait(1000)
cy.get(".spectrum-Picker-label").click()
cy.contains("dog").click()
cy.get(".spectrum-Textfield-input")
@@ -35,19 +51,12 @@ context("Create a automation", () => {
cy.get(".spectrum-Textfield-input")
.eq(1)
.type("11")
+ cy.get(".spectrum-Textfield-input")
+ .eq(2)
+ .type("123456")
+ cy.get(".spectrum-Textfield-input")
+ .eq(3)
+ .type("123456")
})
-
- // Save
- cy.contains("Save Automation").click()
-
- // Activate Automation
- cy.get("[data-cy=activate-automation]").click()
- })
-
- it("should add row when a new row is added", () => {
- cy.contains("Data").click()
- cy.addRow(["Rover", 15])
- cy.reload()
- cy.contains("goodboy").should("have.text", "goodboy")
})
})
diff --git a/packages/builder/cypress/setup.js b/packages/builder/cypress/setup.js
index 4ad8e5287d..1a6f1d5b2b 100644
--- a/packages/builder/cypress/setup.js
+++ b/packages/builder/cypress/setup.js
@@ -23,6 +23,7 @@ process.env.MINIO_SECRET_KEY = "budibase"
process.env.COUCH_DB_USER = "budibase"
process.env.COUCH_DB_PASSWORD = "budibase"
process.env.INTERNAL_API_KEY = "budibase"
+process.env.ALLOW_DEV_AUTOMATIONS = 1
// Stop info logs polluting test outputs
process.env.LOG_LEVEL = "error"
diff --git a/packages/builder/cypress/support/commands.js b/packages/builder/cypress/support/commands.js
index 261b840577..ea6ca81e66 100644
--- a/packages/builder/cypress/support/commands.js
+++ b/packages/builder/cypress/support/commands.js
@@ -34,12 +34,11 @@ Cypress.Commands.add("createApp", name => {
cy.get(".spectrum-Modal")
.within(() => {
cy.get("input").eq(0).type(name).should("have.value", name).blur()
- cy.contains("Create app").click()
+ cy.get(".spectrum-ButtonGroup").contains("Create app").click()
})
.then(() => {
- cy.get(".selected > .content", {
- timeout: 20000,
- }).should("be.visible")
+ cy.expandBudibaseConnection()
+ cy.get(".nav-item.selected > .content").should("be.visible")
})
})
@@ -83,6 +82,7 @@ Cypress.Commands.add("createTable", tableName => {
Cypress.Commands.add("addColumn", (tableName, columnName, type) => {
// Select Table
+ cy.selectTable(tableName)
cy.contains(".nav-item", tableName).click()
cy.contains("Create column").click()
@@ -161,3 +161,15 @@ Cypress.Commands.add("createScreen", (screenName, route) => {
cy.get(".spectrum-Button--cta").click()
})
})
+
+Cypress.Commands.add("expandBudibaseConnection", () => {
+ if (Cypress.$(".nav-item > .content > .opened").length === 0) {
+ // expand the Budibase DB connection string
+ cy.get(".icon.arrow").eq(0).click()
+ }
+})
+
+Cypress.Commands.add("selectTable", tableName => {
+ cy.expandBudibaseConnection()
+ cy.contains(".nav-item", tableName).click()
+})
diff --git a/packages/builder/package.json b/packages/builder/package.json
index bc7758a8c2..9c6c62aa8a 100644
--- a/packages/builder/package.json
+++ b/packages/builder/package.json
@@ -1,6 +1,6 @@
{
"name": "@budibase/builder",
- "version": "0.9.105-alpha.31",
+ "version": "0.9.125-alpha.13",
"license": "AGPL-3.0",
"private": true,
"scripts": {
@@ -65,10 +65,10 @@
}
},
"dependencies": {
- "@budibase/bbui": "^0.9.105-alpha.31",
- "@budibase/client": "^0.9.105-alpha.31",
+ "@budibase/bbui": "^0.9.125-alpha.13",
+ "@budibase/client": "^0.9.125-alpha.13",
"@budibase/colorpicker": "1.1.2",
- "@budibase/string-templates": "^0.9.105-alpha.31",
+ "@budibase/string-templates": "^0.9.125-alpha.13",
"@sentry/browser": "5.19.1",
"@spectrum-css/page": "^3.0.1",
"@spectrum-css/vars": "^3.0.1",
diff --git a/packages/builder/src/analytics.js b/packages/builder/src/analytics.js
index bcc59eb311..5b130a8e6b 100644
--- a/packages/builder/src/analytics.js
+++ b/packages/builder/src/analytics.js
@@ -23,6 +23,7 @@ async function activate() {
if (posthogConfigured) {
posthog.init(process.env.POSTHOG_TOKEN, {
autocapture: false,
+ capture_pageview: false,
api_host: process.env.POSTHOG_URL,
})
posthog.set_config({ persistence: "cookie" })
@@ -79,6 +80,7 @@ const isFeedbackTimeElapsed = sinceDateStr => {
const feedbackMilliseconds = feedbackHours * 60 * 60 * 1000
return Date.now() > sinceDate + feedbackMilliseconds
}
+
function submitFeedback(values) {
if (!analyticsEnabled || !process.env.POSTHOG_TOKEN) return
localStorage.setItem(FEEDBACK_SUBMITTED_KEY, Date.now())
diff --git a/packages/builder/src/builderStore/dataBinding.js b/packages/builder/src/builderStore/dataBinding.js
index 00eaaf0249..903512d0fb 100644
--- a/packages/builder/src/builderStore/dataBinding.js
+++ b/packages/builder/src/builderStore/dataBinding.js
@@ -1,6 +1,10 @@
import { cloneDeep } from "lodash/fp"
import { get } from "svelte/store"
-import { findComponent, findComponentPath } from "./storeUtils"
+import {
+ findComponent,
+ findComponentPath,
+ findAllMatchingComponents,
+} from "./storeUtils"
import { store } from "builderStore"
import { tables as tablesStore, queries as queriesStores } from "stores/backend"
import { makePropSafe } from "@budibase/string-templates"
@@ -18,7 +22,9 @@ export const getBindableProperties = (asset, componentId) => {
const userBindings = getUserBindings()
const urlBindings = getUrlBindings(asset)
const deviceBindings = getDeviceBindings()
+ const stateBindings = getStateBindings()
return [
+ ...stateBindings,
...deviceBindings,
...urlBindings,
...contextBindings,
@@ -256,6 +262,22 @@ const getDeviceBindings = () => {
return bindings
}
+/**
+ * Gets all state bindings that are globally available.
+ */
+const getStateBindings = () => {
+ let bindings = []
+ if (get(store).clientFeatures?.state) {
+ const safeState = makePropSafe("state")
+ bindings = getAllStateVariables().map(key => ({
+ type: "context",
+ runtimeBinding: `${safeState}.${makePropSafe(key)}`,
+ readableBinding: `State.${key}`,
+ }))
+ }
+ return bindings
+}
+
/**
* Gets all bindable properties from URL parameters.
*/
@@ -458,3 +480,49 @@ export function runtimeToReadableBinding(bindableProperties, textWithBindings) {
"readableBinding"
)
}
+
+/**
+ * Returns an array of the keys of any state variables which are set anywhere
+ * in the app.
+ */
+export const getAllStateVariables = () => {
+ let allComponents = []
+
+ // Find all onClick settings in all layouts
+ get(store).layouts.forEach(layout => {
+ const components = findAllMatchingComponents(
+ layout.props,
+ c => c.onClick != null
+ )
+ allComponents = allComponents.concat(components || [])
+ })
+
+ // Find all onClick settings in all screens
+ get(store).screens.forEach(screen => {
+ const components = findAllMatchingComponents(
+ screen.props,
+ c => c.onClick != null
+ )
+ allComponents = allComponents.concat(components || [])
+ })
+
+ // Add state bindings for all state actions
+ let bindingSet = new Set()
+ allComponents.forEach(component => {
+ if (!Array.isArray(component.onClick)) {
+ return
+ }
+ component.onClick.forEach(action => {
+ if (
+ action["##eventHandlerType"] === "Update State" &&
+ action.parameters?.type === "set" &&
+ action.parameters?.key &&
+ action.parameters?.value
+ ) {
+ bindingSet.add(action.parameters.key)
+ }
+ })
+ })
+
+ return Array.from(bindingSet)
+}
diff --git a/packages/builder/src/builderStore/store/automation/Automation.js b/packages/builder/src/builderStore/store/automation/Automation.js
index a9dce88258..dcbb747e38 100644
--- a/packages/builder/src/builderStore/store/automation/Automation.js
+++ b/packages/builder/src/builderStore/store/automation/Automation.js
@@ -13,6 +13,10 @@ export default class Automation {
return this.automation.definition.trigger
}
+ addTestData(data) {
+ this.automation.testData = data
+ }
+
addBlock(block) {
// Make sure to add trigger if doesn't exist
if (!this.hasTrigger() && block.type === "TRIGGER") {
diff --git a/packages/builder/src/builderStore/store/automation/index.js b/packages/builder/src/builderStore/store/automation/index.js
index f522c7bba1..b816a2c26d 100644
--- a/packages/builder/src/builderStore/store/automation/index.js
+++ b/packages/builder/src/builderStore/store/automation/index.js
@@ -17,7 +17,6 @@ const automationActions = store => ({
state.blockDefinitions = {
TRIGGER: jsonResponses[1].trigger,
ACTION: jsonResponses[1].action,
- LOGIC: jsonResponses[1].logic,
}
// if previously selected find the new obj and select it
if (selected) {
@@ -81,8 +80,16 @@ const automationActions = store => ({
},
trigger: async ({ automation }) => {
const { _id } = automation
- const TRIGGER_AUTOMATION_URL = `/api/automations/${_id}/trigger`
- return await api.post(TRIGGER_AUTOMATION_URL)
+ return await api.post(`/api/automations/${_id}/trigger`)
+ },
+ test: async ({ automation }, testData) => {
+ const { _id } = automation
+ const response = await api.post(`/api/automations/${_id}/test`, testData)
+ const json = await response.json()
+ store.update(state => {
+ state.selectedAutomation.testResults = json
+ return state
+ })
},
select: automation => {
store.update(state => {
@@ -91,6 +98,12 @@ const automationActions = store => ({
return state
})
},
+ addTestDataToAutomation: data => {
+ store.update(state => {
+ state.selectedAutomation.addTestData(data)
+ return state
+ })
+ },
addBlockToAutomation: block => {
store.update(state => {
const newBlock = state.selectedAutomation.addBlock(cloneDeep(block))
@@ -132,7 +145,6 @@ export const getAutomationStore = () => {
blockDefinitions: {
TRIGGER: [],
ACTION: [],
- LOGIC: [],
},
selectedAutomation: null,
}
diff --git a/packages/builder/src/builderStore/store/frontend.js b/packages/builder/src/builderStore/store/frontend.js
index 192ade9e5d..603fa88b09 100644
--- a/packages/builder/src/builderStore/store/frontend.js
+++ b/packages/builder/src/builderStore/store/frontend.js
@@ -41,6 +41,9 @@ const INITIAL_FRONTEND_STATE = {
spectrumThemes: false,
intelligentLoading: false,
deviceAwareness: false,
+ state: false,
+ customThemes: false,
+ devicePreview: false,
},
currentFrontEndType: "none",
selectedScreenId: "",
@@ -53,6 +56,8 @@ const INITIAL_FRONTEND_STATE = {
routes: {},
clientLibPath: "",
theme: "",
+ customTheme: {},
+ previewDevice: "desktop",
}
export const getFrontendStore = () => {
@@ -77,6 +82,7 @@ export const getFrontendStore = () => {
layouts,
screens,
theme: application.theme || "spectrum--light",
+ customTheme: application.customTheme,
hasAppPackage: true,
appInstance: application.instance,
clientLibPath,
@@ -110,6 +116,22 @@ export const getFrontendStore = () => {
}
},
},
+ customTheme: {
+ save: async customTheme => {
+ const appId = get(store).appId
+ const response = await api.put(`/api/applications/${appId}`, {
+ customTheme,
+ })
+ if (response.status === 200) {
+ store.update(state => {
+ state.customTheme = customTheme
+ return state
+ })
+ } else {
+ throw new Error("Error updating theme")
+ }
+ },
+ },
routing: {
fetch: async () => {
const response = await api.get("/api/routing")
@@ -210,6 +232,12 @@ export const getFrontendStore = () => {
await store.actions.layouts.save(selectedAsset)
}
},
+ setDevice: device => {
+ store.update(state => {
+ state.previewDevice = device
+ return state
+ })
+ },
},
layouts: {
select: layoutId => {
diff --git a/packages/builder/src/builderStore/store/screenTemplates/newRowScreen.js b/packages/builder/src/builderStore/store/screenTemplates/newRowScreen.js
index b890d42d54..c2dffef4b6 100644
--- a/packages/builder/src/builderStore/store/screenTemplates/newRowScreen.js
+++ b/packages/builder/src/builderStore/store/screenTemplates/newRowScreen.js
@@ -43,6 +43,7 @@ const createScreen = table => {
tableId: table._id,
type: "table",
},
+ size: "spectrum--medium",
})
const fieldGroup = new Component("@budibase/standard-components/fieldgroup")
diff --git a/packages/builder/src/builderStore/store/screenTemplates/rowDetailScreen.js b/packages/builder/src/builderStore/store/screenTemplates/rowDetailScreen.js
index ec737fe36b..3ba8be10b5 100644
--- a/packages/builder/src/builderStore/store/screenTemplates/rowDetailScreen.js
+++ b/packages/builder/src/builderStore/store/screenTemplates/rowDetailScreen.js
@@ -101,7 +101,6 @@ const createScreen = table => {
.instanceName("Form")
.customProps({
actionType: "Update",
- theme: "spectrum--lightest",
size: "spectrum--medium",
dataSource: {
label: table.name,
diff --git a/packages/builder/src/builderStore/store/screenTemplates/rowListScreen.js b/packages/builder/src/builderStore/store/screenTemplates/rowListScreen.js
index ccf1fceb29..188682ed3f 100644
--- a/packages/builder/src/builderStore/store/screenTemplates/rowListScreen.js
+++ b/packages/builder/src/builderStore/store/screenTemplates/rowListScreen.js
@@ -65,6 +65,7 @@ const createScreen = table => {
tableId: table._id,
type: "table",
},
+ size: "spectrum--medium",
paginate: true,
limit: 8,
})
diff --git a/packages/builder/src/builderStore/store/screenTemplates/utils/commonComponents.js b/packages/builder/src/builderStore/store/screenTemplates/utils/commonComponents.js
index 1a64a8958f..5b3bc041ff 100644
--- a/packages/builder/src/builderStore/store/screenTemplates/utils/commonComponents.js
+++ b/packages/builder/src/builderStore/store/screenTemplates/utils/commonComponents.js
@@ -131,6 +131,7 @@ const fieldTypeToComponentMap = {
string: "stringfield",
number: "numberfield",
options: "optionsfield",
+ array: "multifieldselect",
boolean: "booleanfield",
longform: "longformfield",
datetime: "datetimefield",
@@ -167,6 +168,13 @@ export function makeDatasourceFormComponents(datasource) {
optionsSource: "schema",
})
}
+ if (fieldType === "array") {
+ component.customProps({
+ placeholder: "Choose an option",
+ optionsSource: "schema",
+ })
+ }
+
if (fieldType === "link") {
let placeholder =
fieldSchema.relationshipType === "one-to-many"
diff --git a/packages/builder/src/components/automation/AutomationBuilder/AutomationBuilder.svelte b/packages/builder/src/components/automation/AutomationBuilder/AutomationBuilder.svelte
index 62faff3caa..7ce77a58e3 100644
--- a/packages/builder/src/components/automation/AutomationBuilder/AutomationBuilder.svelte
+++ b/packages/builder/src/components/automation/AutomationBuilder/AutomationBuilder.svelte
@@ -1,10 +1,8 @@
{#if automation}
-
{/if}
diff --git a/packages/builder/src/components/automation/AutomationBuilder/BlockList.svelte b/packages/builder/src/components/automation/AutomationBuilder/BlockList.svelte
deleted file mode 100644
index af5c9e449e..0000000000
--- a/packages/builder/src/components/automation/AutomationBuilder/BlockList.svelte
+++ /dev/null
@@ -1,108 +0,0 @@
-
-
-
- {#each tabs as tab, idx}
-
-
onChangeTab(idx)}
- >
- {tab.label}
-
-
- {/each}
-
- (selectedIndex = null)}
- bind:this={popover}
- {anchor}
- align="left"
->
-
- {#each blocks as [stepId, blockDefinition]}
- addBlockToAutomation(stepId, blockDefinition)}
- />
- {/each}
-
-
-
-
-
-
-
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ActionModal.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ActionModal.svelte
new file mode 100644
index 0000000000..8820259e90
--- /dev/null
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ActionModal.svelte
@@ -0,0 +1,136 @@
+
+
+ {
+ blockComplete = true
+ addBlockToAutomation()
+ }}
+>
+ Select an app or event.
+
+ Apps
+
+
+ {#each Object.entries(external) as [idx, action]}
+
selectAction(action)}
+ >
+
+
+
+ {idx.charAt(0).toUpperCase() + idx.slice(1)}
+
+
+ {/each}
+
+
+ Actions
+
+
+ {#each Object.entries(internal) as [idx, action]}
+
selectAction(action)}
+ >
+
+
+
+ {action.name}
+
+
+ {/each}
+
+
+
+
+
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/Arrow.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/Arrow.svelte
index 931490b197..5aec39abeb 100644
--- a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/Arrow.svelte
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/Arrow.svelte
@@ -6,6 +6,7 @@
xmlns="http://www.w3.org/2000/svg"
>
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ExternalActions.js b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ExternalActions.js
new file mode 100644
index 0000000000..843445a3c2
--- /dev/null
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ExternalActions.js
@@ -0,0 +1,11 @@
+import DiscordLogo from "assets/discord.svg"
+import ZapierLogo from "assets/zapier.png"
+import IntegromatLogo from "assets/integromat.png"
+import SlackLogo from "assets/slack.svg"
+
+export const externalActions = {
+ zapier: { name: "zapier", icon: ZapierLogo },
+ discord: { name: "discord", icon: DiscordLogo },
+ slack: { name: "slack", icon: SlackLogo },
+ integromat: { name: "integromat", icon: IntegromatLogo },
+}
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowChart.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowChart.svelte
index e960271b87..53a5de3b51 100644
--- a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowChart.svelte
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowChart.svelte
@@ -1,12 +1,25 @@
-{#if !blocks.length}Add a trigger to your automation to get started{/if}
-
- {#each blocks as block, idx (block.id)}
-
-
- {#if idx !== blocks.length - 1}
-
- {/if}
+
+
+
+
+
{automation.name}
+
+
deleteAutomation()} class="iconPadding">
+
+
+
{
+ testDataModal.show()
+ }}
+ icon="MultipleCheck"
+ size="S">Run test
+
+
- {/each}
-
+ {#each blocks as block, idx (block.id)}
+
+
+ {#if idx !== blocks.length - 1}
+
+
+
+ {/if}
+
+ {/each}
+
+
+
+
+
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowItem.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowItem.svelte
index 439db62639..5898537dae 100644
--- a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowItem.svelte
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/FlowItem.svelte
@@ -1,86 +1,204 @@
onSelect(block)}
+ on:click={() => {
+ onSelect(block)
+ }}
>
-
- {#if block.type === "TRIGGER"}
-
- When this happens...
- {:else if block.type === "ACTION"}
-
- Do this...
- {:else if block.type === "LOGIC"}
-
- Only continue if...
- {/if}
-
- {#if block.type === "TRIGGER"}Trigger{:else}Step {blockIdx + 1}{/if}
+
+
{
+ blockComplete = !blockComplete
+ }}
+ class="splitHeader"
+ >
+
+ {#if externalActions[block.stepId]}
+
+ {:else}
+
+ {/if}
+
+ {#if isTrigger}
+ When this happens:
+ {:else}
+ Do this:
+ {/if}
+
+ {block?.name?.toUpperCase() || ""}
+
+
+ {#if testResult}
+
resultsModal.show()}>
+ View response
+
+ {/if}
- {#if block.type !== "TRIGGER" || allowDeleteTrigger}
-
- {/if}
-
-
-
-
-
+
+ {#if !blockComplete}
+
+
+
+
+
+ {#if setupToggled}
+
+ {#if lastStep}
+
+ {/if}
+
+ {/if}
+
+
+ {/if}
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ResultsModal.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ResultsModal.svelte
new file mode 100644
index 0000000000..7dfdff20a7
--- /dev/null
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/ResultsModal.svelte
@@ -0,0 +1,114 @@
+
+
+
+
+
+ {#if isTrigger || testResult[0].outputs.success}
+
+
+
+ {:else}
+
+
+
+ {/if}
+
+
+
+ {
+ inputToggled = !inputToggled
+ }}
+ class="toggle splitHeader"
+ >
+
+
+ {#if inputToggled}
+
+ {:else}
+
+ {/if}
+
+
+ {#if inputToggled}
+
+
+
+ {/if}
+
+ {
+ outputToggled = !outputToggled
+ }}
+ class="toggle splitHeader"
+ >
+
+
+ {#if outputToggled}
+
+ {:else}
+
+ {/if}
+
+
+ {#if outputToggled}
+
+
+
+ {/if}
+
+
+
diff --git a/packages/builder/src/components/automation/AutomationBuilder/FlowChart/TestDataModal.svelte b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/TestDataModal.svelte
new file mode 100644
index 0000000000..d05c8fa326
--- /dev/null
+++ b/packages/builder/src/components/automation/AutomationBuilder/FlowChart/TestDataModal.svelte
@@ -0,0 +1,88 @@
+
+
+ {
+ automationStore.actions.addTestDataToAutomation(testData)
+ automationStore.actions.test($automationStore.selectedAutomation, testData)
+ }}
+ cancelText="Cancel"
+>
+
+
+
+
+
+
+
+
+
diff --git a/packages/builder/src/components/automation/AutomationPanel/AutomationList.svelte b/packages/builder/src/components/automation/AutomationPanel/AutomationList.svelte
index f70366773a..79de4bfbe6 100644
--- a/packages/builder/src/components/automation/AutomationPanel/AutomationList.svelte
+++ b/packages/builder/src/components/automation/AutomationPanel/AutomationList.svelte
@@ -6,7 +6,6 @@
import EditAutomationPopover from "./EditAutomationPopover.svelte"
$: selectedAutomationId = $automationStore.selectedAutomation?.automation?._id
-
onMount(() => {
automationStore.actions.fetch()
})
diff --git a/packages/builder/src/components/automation/AutomationPanel/AutomationPanel.svelte b/packages/builder/src/components/automation/AutomationPanel/AutomationPanel.svelte
index dbcbc33db0..0c975eab18 100644
--- a/packages/builder/src/components/automation/AutomationPanel/AutomationPanel.svelte
+++ b/packages/builder/src/components/automation/AutomationPanel/AutomationPanel.svelte
@@ -2,7 +2,9 @@
import AutomationList from "./AutomationList.svelte"
import CreateAutomationModal from "./CreateAutomationModal.svelte"
import { Icon, Modal, Tabs, Tab } from "@budibase/bbui"
- let modal
+
+ export let modal
+ export let webhookModal
@@ -11,7 +13,7 @@
diff --git a/packages/builder/src/components/automation/AutomationPanel/CreateAutomationModal.svelte b/packages/builder/src/components/automation/AutomationPanel/CreateAutomationModal.svelte
index dfcfc2ab95..e774c366a5 100644
--- a/packages/builder/src/components/automation/AutomationPanel/CreateAutomationModal.svelte
+++ b/packages/builder/src/components/automation/AutomationPanel/CreateAutomationModal.svelte
@@ -3,12 +3,14 @@
import { database } from "stores/backend"
import { automationStore } from "builderStore"
import { notifications } from "@budibase/bbui"
- import { Icon, Input, ModalContent } from "@budibase/bbui"
+ import { Input, ModalContent, Layout, Body, Icon } from "@budibase/bbui"
import analytics from "analytics"
let name
+ let selectedTrigger
+ let triggerVal
+ export let webhookModal
- $: valid = !!name
$: instanceId = $database._id
async function createAutomation() {
@@ -16,41 +18,97 @@
name,
instanceId,
})
+ const newBlock = $automationStore.selectedAutomation.constructBlock(
+ "TRIGGER",
+ triggerVal.stepId,
+ triggerVal
+ )
+
+ automationStore.actions.addBlockToAutomation(newBlock)
+ if (triggerVal.stepId === "WEBHOOK") {
+ webhookModal.show
+ }
+
+ await automationStore.actions.save({
+ instanceId,
+ automation: $automationStore.selectedAutomation?.automation,
+ })
+
notifications.success(`Automation ${name} created.`)
+
$goto(`./${$automationStore.selectedAutomation.automation._id}`)
analytics.captureEvent("Automation Created", { name })
}
+ $: triggers = Object.entries($automationStore.blockDefinitions.TRIGGER)
+
+ const selectTrigger = trigger => {
+ triggerVal = trigger
+ selectedTrigger = trigger.name
+ }
+ Please name your automation, then select a trigger. Every automation must
+ start with a trigger.
+
-
-
- Learn about automations
-
+
+
+ Triggers
+
+
+ {#each triggers as [idx, trigger]}
+
selectTrigger(trigger)}
+ >
+
+
+
+ {trigger.name}
+
+
+ {/each}
+
+
diff --git a/packages/builder/src/components/automation/SetupPanel/AutomationBlockSetup.svelte b/packages/builder/src/components/automation/SetupPanel/AutomationBlockSetup.svelte
index d1c5d104d3..8e6cb42ee2 100644
--- a/packages/builder/src/components/automation/SetupPanel/AutomationBlockSetup.svelte
+++ b/packages/builder/src/components/automation/SetupPanel/AutomationBlockSetup.svelte
@@ -2,8 +2,16 @@
import TableSelector from "./TableSelector.svelte"
import RowSelector from "./RowSelector.svelte"
import SchemaSetup from "./SchemaSetup.svelte"
- import { Button, Input, Select, Label } from "@budibase/bbui"
+ import {
+ Button,
+ Input,
+ Select,
+ Label,
+ ActionButton,
+ Drawer,
+ } from "@budibase/bbui"
import { automationStore } from "builderStore"
+ import { tables } from "stores/backend"
import WebhookDisplay from "../Shared/WebhookDisplay.svelte"
import DrawerBindableInput from "../../common/bindings/DrawerBindableInput.svelte"
import AutomationBindingPanel from "../../common/bindings/ServerBindingPanel.svelte"
@@ -12,15 +20,50 @@
import QueryParamSelector from "./QueryParamSelector.svelte"
import CronBuilder from "./CronBuilder.svelte"
import Editor from "components/integration/QueryEditor.svelte"
+ import { database } from "stores/backend"
+ import { debounce } from "lodash"
+ import ModalBindableInput from "components/common/bindings/ModalBindableInput.svelte"
+ import FilterDrawer from "components/design/PropertiesPanel/PropertyControls/FilterEditor/FilterDrawer.svelte"
+ // need the client lucene builder to convert to the structure API expects
+ import { buildLuceneQuery } from "../../../../../client/src/utils/lucene"
export let block
export let webhookModal
- $: inputs = Object.entries(block.schema?.inputs?.properties || {})
+ export let testData
+ export let schemaProperties
+ export let isTestModal = false
+ let drawer
+ let tempFilters = lookForFilters(schemaProperties) || []
+ let fillWidth = true
+
$: stepId = block.stepId
$: bindings = getAvailableBindings(
- block,
+ block || $automationStore.selectedBlock,
$automationStore.selectedAutomation?.automation?.definition
)
+ $: instanceId = $database._id
+
+ $: inputData = testData ? testData : block.inputs
+ $: tableId = inputData ? inputData.tableId : null
+ $: table = tableId
+ ? $tables.list.find(table => table._id === inputData.tableId)
+ : { schema: {} }
+ $: schemaFields = table ? Object.values(table.schema) : []
+
+ const onChange = debounce(
+ async function (e, key) {
+ if (isTestModal) {
+ testData[key] = e.detail
+ } else {
+ block.inputs[key] = e.detail
+ await automationStore.actions.save({
+ instanceId,
+ automation: $automationStore.selectedAutomation?.automation,
+ })
+ }
+ },
+ isTestModal ? 0 : 800
+ )
function getAvailableBindings(block, automation) {
if (!block || !automation) {
@@ -52,64 +95,158 @@
}
return bindings
}
+
+ function lookForFilters(properties) {
+ if (!properties) {
+ return []
+ }
+ let filters
+ const inputs = testData ? testData : block.inputs
+ for (let [key, field] of properties) {
+ // need to look for the builder definition (keyed separately, see saveFilters)
+ const defKey = `${key}-def`
+ if (field.customType === "filters" && inputs?.[defKey]) {
+ filters = inputs[defKey]
+ break
+ }
+ }
+ return filters || []
+ }
+
+ function saveFilters(key) {
+ const filters = buildLuceneQuery(tempFilters)
+ const defKey = `${key}-def`
+ inputData[key] = filters
+ inputData[defKey] = tempFilters
+ onChange({ detail: filters }, key)
+ // need to store the builder definition in the automation
+ onChange({ detail: tempFilters }, defKey)
+ drawer.hide()
+ }
-
{block.name}
- {#each inputs as [key, value]}
+ {#each schemaProperties as [key, value]}
-
+
{#if value.type === "string" && value.enum}
{/each}
@@ -132,9 +269,7 @@
grid-gap: 5px;
}
- .block-label {
- font-weight: 600;
- font-size: var(--font-size-s);
- color: var(--grey-7);
+ .test :global(.drawer) {
+ width: 10000px !important;
}
diff --git a/packages/builder/src/components/automation/SetupPanel/CronBuilder.svelte b/packages/builder/src/components/automation/SetupPanel/CronBuilder.svelte
index 810e452742..1b410cd86a 100644
--- a/packages/builder/src/components/automation/SetupPanel/CronBuilder.svelte
+++ b/packages/builder/src/components/automation/SetupPanel/CronBuilder.svelte
@@ -1,7 +1,13 @@
-
+
diff --git a/packages/builder/src/components/automation/SetupPanel/RowSelector.svelte b/packages/builder/src/components/automation/SetupPanel/RowSelector.svelte
index 99f41908e7..3f390e0a4f 100644
--- a/packages/builder/src/components/automation/SetupPanel/RowSelector.svelte
+++ b/packages/builder/src/components/automation/SetupPanel/RowSelector.svelte
@@ -3,12 +3,25 @@
import { Select } from "@budibase/bbui"
import DrawerBindableInput from "../../common/bindings/DrawerBindableInput.svelte"
import AutomationBindingPanel from "../../common/bindings/ServerBindingPanel.svelte"
+ import { createEventDispatcher } from "svelte"
+ import ModalBindableInput from "components/common/bindings/ModalBindableInput.svelte"
+ import { automationStore } from "builderStore"
+
+ const dispatch = createEventDispatcher()
export let value
export let bindings
-
$: table = $tables.list.find(table => table._id === value?.tableId)
$: schemaFields = Object.entries(table?.schema ?? {})
+ const onChangeTable = e => {
+ value = { tableId: e.detail }
+ dispatch("change", value)
+ }
+
+ const onChange = (e, field) => {
+ value[field] = e.detail
+ dispatch("change", value)
+ }
// Ensure any nullish tableId values get set to empty string so
// that the select works
@@ -20,7 +33,8 @@
table.name}
getOptionValue={table => table._id}
@@ -32,19 +46,32 @@
{#if !schema.autocolumn}
{#if schemaHasOptions(schema)}
onChange(e, field)}
label={field}
- bind:value={value[field]}
+ value={value[field]}
options={schema.constraints.inclusion}
/>
{:else if schema.type === "string" || schema.type === "number"}
- (value[field] = e.detail)}
- label={field}
- type="string"
- {bindings}
- />
+ {#if $automationStore.selectedAutomation.automation.testData}
+ onChange(e, field)}
+ {bindings}
+ />
+ {:else}
+ onChange(e, field)}
+ label={field}
+ type="string"
+ {bindings}
+ fillWidth={true}
+ />
+ {/if}
{/if}
{/if}
{/each}
diff --git a/packages/builder/src/components/automation/SetupPanel/SchemaSetup.svelte b/packages/builder/src/components/automation/SetupPanel/SchemaSetup.svelte
index 1257563ff8..54f5b90164 100644
--- a/packages/builder/src/components/automation/SetupPanel/SchemaSetup.svelte
+++ b/packages/builder/src/components/automation/SetupPanel/SchemaSetup.svelte
@@ -1,5 +1,8 @@
@@ -68,7 +72,10 @@
/>
(value[field.name] = e.target.value)}
+ on:change={e => {
+ value[field.name] = e.target.value
+ dispatch("change", value)
+ }}
options={typeOptions}
/>
- import { automationStore } from "builderStore"
- import { notifications, Button, Modal, Heading, Toggle } from "@budibase/bbui"
- import AutomationBlockSetup from "./AutomationBlockSetup.svelte"
- import CreateWebookModal from "../Shared/CreateWebhookModal.svelte"
-
- let webhookModal
-
- $: automation = $automationStore.selectedAutomation?.automation
- $: automationLive = automation?.live
-
- function setAutomationLive(live) {
- if (automationLive === live) {
- return
- }
- automation.live = live
- automationStore.actions.save(automation)
- if (live) {
- notifications.info(`Automation ${automation.name} enabled.`)
- } else {
- notifications.error(`Automation ${automation.name} disabled.`)
- }
- }
-
- async function testAutomation() {
- const result = await automationStore.actions.trigger({
- automation: $automationStore.selectedAutomation.automation,
- })
- if (result.status === 200) {
- notifications.success(
- `Automation ${automation.name} triggered successfully.`
- )
- } else {
- notifications.error(`Failed to trigger automation ${automation.name}.`)
- }
- }
-
- async function saveAutomation() {
- await automationStore.actions.save(automation)
- notifications.success(`Automation ${automation.name} saved.`)
- }
-
-
-
- Setup
- setAutomationLive(!automationLive)}
- dataCy="activate-automation"
- text="Live"
- />
-
-{#if $automationStore.selectedBlock}
-
-{:else if automation}
- {automation.name}
-
-{/if}
-
-
-
-
-
-
diff --git a/packages/builder/src/components/automation/SetupPanel/TableSelector.svelte b/packages/builder/src/components/automation/SetupPanel/TableSelector.svelte
index 4f9ac05a06..ceb28a37ca 100644
--- a/packages/builder/src/components/automation/SetupPanel/TableSelector.svelte
+++ b/packages/builder/src/components/automation/SetupPanel/TableSelector.svelte
@@ -1,11 +1,20 @@
table.name}
diff --git a/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte b/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte
index 0724016679..e82c55679a 100644
--- a/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte
+++ b/packages/builder/src/components/backend/DataTable/RowFieldControl.svelte
@@ -1,5 +1,12 @@
diff --git a/packages/builder/src/components/backend/DatasourceNavigator/DatasourceNavigator.svelte b/packages/builder/src/components/backend/DatasourceNavigator/DatasourceNavigator.svelte
index 42ea30dbb0..84c737eb67 100644
--- a/packages/builder/src/components/backend/DatasourceNavigator/DatasourceNavigator.svelte
+++ b/packages/builder/src/components/backend/DatasourceNavigator/DatasourceNavigator.svelte
@@ -9,7 +9,10 @@
import TableNavigator from "components/backend/TableNavigator/TableNavigator.svelte"
import ICONS from "./icons"
+ let openDataSources = []
+
function selectDatasource(datasource) {
+ toggleNode(datasource)
datasources.select(datasource._id)
$goto(`./datasource/${datasource._id}`)
}
@@ -19,6 +22,15 @@
$goto(`./datasource/${query.datasourceId}/${query._id}`)
}
+ function toggleNode(datasource) {
+ const isOpen = openDataSources.includes(datasource._id)
+ if (isOpen) {
+ openDataSources = openDataSources.filter(id => datasource._id !== id)
+ } else {
+ openDataSources = [...openDataSources, datasource._id]
+ }
+ }
+
onMount(() => {
datasources.fetch()
queries.fetch()
@@ -31,8 +43,11 @@
0}
text={datasource.name}
+ opened={openDataSources.includes(datasource._id)}
selected={$datasources.selected === datasource._id}
+ withArrow={true}
on:click={() => selectDatasource(datasource)}
+ on:iconClick={() => toggleNode(datasource)}
>
-
+ {#if openDataSources.includes(datasource._id)}
+
+ {/if}
{#each $queries.list.filter(query => query.datasourceId === datasource._id) as query}
onClickQuery(query)}
>
diff --git a/packages/builder/src/components/backend/DatasourceNavigator/modals/UpdateDatasourceModal.svelte b/packages/builder/src/components/backend/DatasourceNavigator/modals/UpdateDatasourceModal.svelte
index 561548fb59..f93af59a38 100644
--- a/packages/builder/src/components/backend/DatasourceNavigator/modals/UpdateDatasourceModal.svelte
+++ b/packages/builder/src/components/backend/DatasourceNavigator/modals/UpdateDatasourceModal.svelte
@@ -42,9 +42,9 @@
diff --git a/packages/builder/src/components/backend/DatasourceNavigator/popovers/EditDatasourcePopover.svelte b/packages/builder/src/components/backend/DatasourceNavigator/popovers/EditDatasourcePopover.svelte
index 7f2f104278..d837535267 100644
--- a/packages/builder/src/components/backend/DatasourceNavigator/popovers/EditDatasourcePopover.svelte
+++ b/packages/builder/src/components/backend/DatasourceNavigator/popovers/EditDatasourcePopover.svelte
@@ -26,7 +26,7 @@
-
+
diff --git a/packages/builder/src/components/backend/TableNavigator/TableDataImport.svelte b/packages/builder/src/components/backend/TableNavigator/TableDataImport.svelte
index 2bbbb471c2..0a59988da6 100644
--- a/packages/builder/src/components/backend/TableNavigator/TableDataImport.svelte
+++ b/packages/builder/src/components/backend/TableNavigator/TableDataImport.svelte
@@ -113,6 +113,10 @@
label: "Options",
value: FIELDS.OPTIONS.type,
},
+ {
+ label: "Multi-select",
+ value: FIELDS.ARRAY.type,
+ },
]
diff --git a/packages/builder/src/components/common/ConfigChecklist.svelte b/packages/builder/src/components/common/ConfigChecklist.svelte
index 83e6a41063..05f3f7c719 100644
--- a/packages/builder/src/components/common/ConfigChecklist.svelte
+++ b/packages/builder/src/components/common/ConfigChecklist.svelte
@@ -7,16 +7,29 @@
ProgressCircle,
} from "@budibase/bbui"
import { admin } from "stores/portal"
+ import { goto } from "@roxi/routify"
+ import { onMount } from "svelte"
- const MESSAGES = {
- apps: "Create your first app",
- smtp: "Set up email",
- adminUser: "Create your first user",
- sso: "Set up single sign-on",
- }
+ let width = window.innerWidth
+ $: side = width < 500 ? "right" : "left"
+
+ const resizeObserver = new ResizeObserver(entries => {
+ if (entries?.[0]) {
+ width = entries[0].contentRect?.width
+ }
+ })
+
+ onMount(() => {
+ const doc = document.documentElement
+ resizeObserver.observe(doc)
+
+ return () => {
+ resizeObserver.unobserve(doc)
+ }
+ })
-
+
@@ -28,9 +41,12 @@
{#each Object.keys($admin.checklist) as checklistItem, idx}