Merge remote-tracking branch 'origin/master' into automation-branching-ux-updates
This commit is contained in:
commit
faf5335d1f
|
@ -19,7 +19,6 @@ MINIO_PORT=4004
|
||||||
COUCH_DB_PORT=4005
|
COUCH_DB_PORT=4005
|
||||||
COUCH_DB_SQS_PORT=4006
|
COUCH_DB_SQS_PORT=4006
|
||||||
REDIS_PORT=6379
|
REDIS_PORT=6379
|
||||||
WATCHTOWER_PORT=6161
|
|
||||||
BUDIBASE_ENVIRONMENT=PRODUCTION
|
BUDIBASE_ENVIRONMENT=PRODUCTION
|
||||||
SQL_MAX_ROWS=
|
SQL_MAX_ROWS=
|
||||||
|
|
||||||
|
|
|
@ -74,7 +74,6 @@ services:
|
||||||
- WORKER_UPSTREAM_URL=http://worker-service:4003
|
- WORKER_UPSTREAM_URL=http://worker-service:4003
|
||||||
- MINIO_UPSTREAM_URL=http://minio-service:9000
|
- MINIO_UPSTREAM_URL=http://minio-service:9000
|
||||||
- COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
- COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
||||||
- WATCHTOWER_UPSTREAM_URL=http://watchtower-service:8080
|
|
||||||
- RESOLVER=127.0.0.11
|
- RESOLVER=127.0.0.11
|
||||||
depends_on:
|
depends_on:
|
||||||
- minio-service
|
- minio-service
|
||||||
|
|
|
@ -87,7 +87,6 @@ services:
|
||||||
- WORKER_UPSTREAM_URL=http://worker-service:4003
|
- WORKER_UPSTREAM_URL=http://worker-service:4003
|
||||||
- MINIO_UPSTREAM_URL=http://minio-service:9000
|
- MINIO_UPSTREAM_URL=http://minio-service:9000
|
||||||
- COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
- COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
||||||
- WATCHTOWER_UPSTREAM_URL=http://watchtower-service:8080
|
|
||||||
- RESOLVER=127.0.0.11
|
- RESOLVER=127.0.0.11
|
||||||
depends_on:
|
depends_on:
|
||||||
- minio-service
|
- minio-service
|
||||||
|
@ -112,19 +111,6 @@ services:
|
||||||
volumes:
|
volumes:
|
||||||
- redis_data:/data
|
- redis_data:/data
|
||||||
|
|
||||||
watchtower-service:
|
|
||||||
restart: always
|
|
||||||
image: containrrr/watchtower
|
|
||||||
volumes:
|
|
||||||
- /var/run/docker.sock:/var/run/docker.sock
|
|
||||||
command: --debug --http-api-update bbapps bbworker bbproxy
|
|
||||||
environment:
|
|
||||||
- WATCHTOWER_HTTP_API=true
|
|
||||||
- WATCHTOWER_HTTP_API_TOKEN=budibase
|
|
||||||
- WATCHTOWER_CLEANUP=true
|
|
||||||
labels:
|
|
||||||
- "com.centurylinklabs.watchtower.enable=false"
|
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
couchdb3_data:
|
couchdb3_data:
|
||||||
driver: local
|
driver: local
|
||||||
|
|
|
@ -1,152 +0,0 @@
|
||||||
static_resources:
|
|
||||||
listeners:
|
|
||||||
- name: main_listener
|
|
||||||
address:
|
|
||||||
socket_address: { address: 0.0.0.0, port_value: 10000 }
|
|
||||||
filter_chains:
|
|
||||||
- filters:
|
|
||||||
- name: envoy.filters.network.http_connection_manager
|
|
||||||
typed_config:
|
|
||||||
"@type": type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager
|
|
||||||
stat_prefix: ingress
|
|
||||||
codec_type: auto
|
|
||||||
route_config:
|
|
||||||
name: local_route
|
|
||||||
virtual_hosts:
|
|
||||||
- name: local_services
|
|
||||||
domains: ["*"]
|
|
||||||
routes:
|
|
||||||
- match: { prefix: "/app/" }
|
|
||||||
route:
|
|
||||||
cluster: app-service
|
|
||||||
prefix_rewrite: "/"
|
|
||||||
|
|
||||||
- match: { path: "/v1/update" }
|
|
||||||
route:
|
|
||||||
cluster: watchtower-service
|
|
||||||
|
|
||||||
- match: { prefix: "/builder/" }
|
|
||||||
route:
|
|
||||||
cluster: app-service
|
|
||||||
|
|
||||||
- match: { prefix: "/builder" }
|
|
||||||
route:
|
|
||||||
cluster: app-service
|
|
||||||
|
|
||||||
- match: { prefix: "/app_" }
|
|
||||||
route:
|
|
||||||
cluster: app-service
|
|
||||||
|
|
||||||
# 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
|
|
||||||
- match: { prefix: "/api/" }
|
|
||||||
route:
|
|
||||||
cluster: app-service
|
|
||||||
timeout: 120s
|
|
||||||
|
|
||||||
- match: { prefix: "/worker/" }
|
|
||||||
route:
|
|
||||||
cluster: worker-service
|
|
||||||
prefix_rewrite: "/"
|
|
||||||
|
|
||||||
- match: { prefix: "/db/" }
|
|
||||||
route:
|
|
||||||
cluster: couchdb-service
|
|
||||||
prefix_rewrite: "/"
|
|
||||||
|
|
||||||
# minio is on the default route because this works
|
|
||||||
# best, minio + AWS SDK doesn't handle path proxy
|
|
||||||
- match: { prefix: "/" }
|
|
||||||
route:
|
|
||||||
cluster: minio-service
|
|
||||||
|
|
||||||
http_filters:
|
|
||||||
- name: envoy.filters.http.router
|
|
||||||
|
|
||||||
clusters:
|
|
||||||
- name: app-service
|
|
||||||
connect_timeout: 0.25s
|
|
||||||
type: strict_dns
|
|
||||||
lb_policy: round_robin
|
|
||||||
load_assignment:
|
|
||||||
cluster_name: app-service
|
|
||||||
endpoints:
|
|
||||||
- lb_endpoints:
|
|
||||||
- endpoint:
|
|
||||||
address:
|
|
||||||
socket_address:
|
|
||||||
address: app-service
|
|
||||||
port_value: 4002
|
|
||||||
|
|
||||||
- name: minio-service
|
|
||||||
connect_timeout: 0.25s
|
|
||||||
type: strict_dns
|
|
||||||
lb_policy: round_robin
|
|
||||||
load_assignment:
|
|
||||||
cluster_name: minio-service
|
|
||||||
endpoints:
|
|
||||||
- lb_endpoints:
|
|
||||||
- endpoint:
|
|
||||||
address:
|
|
||||||
socket_address:
|
|
||||||
address: minio-service
|
|
||||||
port_value: 9000
|
|
||||||
|
|
||||||
- name: worker-service
|
|
||||||
connect_timeout: 0.25s
|
|
||||||
type: strict_dns
|
|
||||||
lb_policy: round_robin
|
|
||||||
load_assignment:
|
|
||||||
cluster_name: worker-service
|
|
||||||
endpoints:
|
|
||||||
- lb_endpoints:
|
|
||||||
- endpoint:
|
|
||||||
address:
|
|
||||||
socket_address:
|
|
||||||
address: worker-service
|
|
||||||
port_value: 4003
|
|
||||||
|
|
||||||
- name: couchdb-service
|
|
||||||
connect_timeout: 0.25s
|
|
||||||
type: strict_dns
|
|
||||||
lb_policy: round_robin
|
|
||||||
load_assignment:
|
|
||||||
cluster_name: couchdb-service
|
|
||||||
endpoints:
|
|
||||||
- lb_endpoints:
|
|
||||||
- endpoint:
|
|
||||||
address:
|
|
||||||
socket_address:
|
|
||||||
address: couchdb-service
|
|
||||||
port_value: 5984
|
|
||||||
|
|
||||||
- name: watchtower-service
|
|
||||||
connect_timeout: 0.25s
|
|
||||||
type: strict_dns
|
|
||||||
lb_policy: round_robin
|
|
||||||
load_assignment:
|
|
||||||
cluster_name: watchtower-service
|
|
||||||
endpoints:
|
|
||||||
- lb_endpoints:
|
|
||||||
- endpoint:
|
|
||||||
address:
|
|
||||||
socket_address:
|
|
||||||
address: watchtower-service
|
|
||||||
port_value: 8080
|
|
||||||
|
|
|
@ -18,7 +18,6 @@ WORKER_PORT=4003
|
||||||
MINIO_PORT=4004
|
MINIO_PORT=4004
|
||||||
COUCH_DB_PORT=4005
|
COUCH_DB_PORT=4005
|
||||||
REDIS_PORT=6379
|
REDIS_PORT=6379
|
||||||
WATCHTOWER_PORT=6161
|
|
||||||
BUDIBASE_ENVIRONMENT=PRODUCTION
|
BUDIBASE_ENVIRONMENT=PRODUCTION
|
||||||
|
|
||||||
# An admin user can be automatically created initially if these are set
|
# An admin user can be automatically created initially if these are set
|
||||||
|
@ -26,4 +25,4 @@ BB_ADMIN_USER_EMAIL=
|
||||||
BB_ADMIN_USER_PASSWORD=
|
BB_ADMIN_USER_PASSWORD=
|
||||||
|
|
||||||
# A path that is watched for plugin bundles. Any bundles found are imported automatically/
|
# A path that is watched for plugin bundles. Any bundles found are imported automatically/
|
||||||
PLUGINS_DIR=
|
PLUGINS_DIR=
|
||||||
|
|
|
@ -78,11 +78,6 @@
|
||||||
"default": "6379",
|
"default": "6379",
|
||||||
"preset": true
|
"preset": true
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"name": "WATCHTOWER_PORT",
|
|
||||||
"default": "6161",
|
|
||||||
"preset": true
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"name": "BUDIBASE_ENVIRONMENT",
|
"name": "BUDIBASE_ENVIRONMENT",
|
||||||
"default": "PRODUCTION",
|
"default": "PRODUCTION",
|
||||||
|
|
|
@ -22,5 +22,4 @@ ENV APPS_UPSTREAM_URL=http://app-service:4002
|
||||||
ENV WORKER_UPSTREAM_URL=http://worker-service:4003
|
ENV WORKER_UPSTREAM_URL=http://worker-service:4003
|
||||||
ENV MINIO_UPSTREAM_URL=http://minio-service:9000
|
ENV MINIO_UPSTREAM_URL=http://minio-service:9000
|
||||||
ENV COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
ENV COUCHDB_UPSTREAM_URL=http://couchdb-service:5984
|
||||||
ENV WATCHTOWER_UPSTREAM_URL=http://watchtower-service:8080
|
|
||||||
ENV RESOLVER=127.0.0.11
|
ENV RESOLVER=127.0.0.11
|
||||||
|
|
|
@ -81,7 +81,6 @@ http {
|
||||||
set $worker ${WORKER_UPSTREAM_URL};
|
set $worker ${WORKER_UPSTREAM_URL};
|
||||||
set $minio ${MINIO_UPSTREAM_URL};
|
set $minio ${MINIO_UPSTREAM_URL};
|
||||||
set $couchdb ${COUCHDB_UPSTREAM_URL};
|
set $couchdb ${COUCHDB_UPSTREAM_URL};
|
||||||
set $watchtower ${WATCHTOWER_UPSTREAM_URL};
|
|
||||||
|
|
||||||
location /health {
|
location /health {
|
||||||
access_log off;
|
access_log off;
|
||||||
|
@ -107,10 +106,6 @@ http {
|
||||||
proxy_pass $apps;
|
proxy_pass $apps;
|
||||||
}
|
}
|
||||||
|
|
||||||
location = /v1/update {
|
|
||||||
proxy_pass $watchtower;
|
|
||||||
}
|
|
||||||
|
|
||||||
location ~ ^/(builder|app_) {
|
location ~ ^/(builder|app_) {
|
||||||
proxy_http_version 1.1;
|
proxy_http_version 1.1;
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,6 @@ let IMAGES = {
|
||||||
couch: "ibmcom/couchdb3",
|
couch: "ibmcom/couchdb3",
|
||||||
curl: "curlimages/curl",
|
curl: "curlimages/curl",
|
||||||
redis: "redis",
|
redis: "redis",
|
||||||
watchtower: "containrrr/watchtower",
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IS_SINGLE_IMAGE) {
|
if (IS_SINGLE_IMAGE) {
|
||||||
|
@ -53,4 +52,4 @@ if (!IS_SINGLE_IMAGE) {
|
||||||
copyFile(FILES.ENV)
|
copyFile(FILES.ENV)
|
||||||
|
|
||||||
// compress
|
// compress
|
||||||
execSync(`tar -czf bb-airgapped.tar.gz hosting/scripts/bb-airgapped`)
|
execSync(`tar -czf bb-airgapped.tar.gz hosting/scripts/bb-airgapped`)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"$schema": "node_modules/lerna/schemas/lerna-schema.json",
|
"$schema": "node_modules/lerna/schemas/lerna-schema.json",
|
||||||
"version": "3.1.0",
|
"version": "3.2.0",
|
||||||
"npmClient": "yarn",
|
"npmClient": "yarn",
|
||||||
"packages": [
|
"packages": [
|
||||||
"packages/*",
|
"packages/*",
|
||||||
|
|
|
@ -267,12 +267,12 @@ export class FlagSet<V extends Flag<any>, T extends { [key: string]: V }> {
|
||||||
// All of the machinery in this file is to make sure that flags have their
|
// All of the machinery in this file is to make sure that flags have their
|
||||||
// default values set correctly and their types flow through the system.
|
// default values set correctly and their types flow through the system.
|
||||||
export const flags = new FlagSet({
|
export const flags = new FlagSet({
|
||||||
[FeatureFlag.DEFAULT_VALUES]: Flag.boolean(env.isDev()),
|
[FeatureFlag.DEFAULT_VALUES]: Flag.boolean(true),
|
||||||
[FeatureFlag.AUTOMATION_BRANCHING]: Flag.boolean(env.isDev()),
|
[FeatureFlag.AUTOMATION_BRANCHING]: Flag.boolean(true),
|
||||||
[FeatureFlag.SQS]: Flag.boolean(true),
|
[FeatureFlag.SQS]: Flag.boolean(true),
|
||||||
[FeatureFlag.AI_CUSTOM_CONFIGS]: Flag.boolean(env.isDev()),
|
[FeatureFlag.ENRICHED_RELATIONSHIPS]: Flag.boolean(true),
|
||||||
[FeatureFlag.ENRICHED_RELATIONSHIPS]: Flag.boolean(env.isDev()),
|
[FeatureFlag.AI_CUSTOM_CONFIGS]: Flag.boolean(true),
|
||||||
[FeatureFlag.BUDIBASE_AI]: Flag.boolean(env.isDev()),
|
[FeatureFlag.BUDIBASE_AI]: Flag.boolean(true),
|
||||||
})
|
})
|
||||||
|
|
||||||
type UnwrapPromise<T> = T extends Promise<infer U> ? U : T
|
type UnwrapPromise<T> = T extends Promise<infer U> ? U : T
|
||||||
|
|
|
@ -4,12 +4,10 @@
|
||||||
Layout,
|
Layout,
|
||||||
Heading,
|
Heading,
|
||||||
Body,
|
Body,
|
||||||
Button,
|
|
||||||
Divider,
|
Divider,
|
||||||
notifications,
|
notifications,
|
||||||
Label,
|
Label,
|
||||||
Modal,
|
Link,
|
||||||
ModalContent,
|
|
||||||
} from "@budibase/bbui"
|
} from "@budibase/bbui"
|
||||||
import { API } from "api"
|
import { API } from "api"
|
||||||
import { auth, admin } from "stores/portal"
|
import { auth, admin } from "stores/portal"
|
||||||
|
@ -21,8 +19,6 @@
|
||||||
let githubVersion
|
let githubVersion
|
||||||
let githubPublishedDate
|
let githubPublishedDate
|
||||||
let githubPublishedTime
|
let githubPublishedTime
|
||||||
let needsUpdate = true
|
|
||||||
let updateModal
|
|
||||||
|
|
||||||
// Only admins allowed here
|
// Only admins allowed here
|
||||||
$: {
|
$: {
|
||||||
|
@ -31,21 +27,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function updateBudibase() {
|
|
||||||
try {
|
|
||||||
notifications.info("Updating budibase..")
|
|
||||||
await fetch("/v1/update", {
|
|
||||||
headers: {
|
|
||||||
Authorization: "Bearer budibase",
|
|
||||||
},
|
|
||||||
})
|
|
||||||
notifications.success("Your budibase installation is up to date.")
|
|
||||||
getVersion()
|
|
||||||
} catch (err) {
|
|
||||||
notifications.error(`Error installing budibase update ${err}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getVersion() {
|
async function getVersion() {
|
||||||
try {
|
try {
|
||||||
version = await API.getBudibaseVersion()
|
version = await API.getBudibaseVersion()
|
||||||
|
@ -69,13 +50,6 @@
|
||||||
githubPublishedDate = new Date(githubResponse.published_at)
|
githubPublishedDate = new Date(githubResponse.published_at)
|
||||||
githubPublishedTime = githubPublishedDate.toLocaleTimeString()
|
githubPublishedTime = githubPublishedDate.toLocaleTimeString()
|
||||||
githubPublishedDate = githubPublishedDate.toLocaleDateString()
|
githubPublishedDate = githubPublishedDate.toLocaleDateString()
|
||||||
|
|
||||||
//Does Budibase need to be updated?
|
|
||||||
if (githubVersion === version) {
|
|
||||||
needsUpdate = false
|
|
||||||
} else {
|
|
||||||
needsUpdate = true
|
|
||||||
}
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
notifications.error("Error getting the latest Budibase version")
|
notifications.error("Error getting the latest Budibase version")
|
||||||
githubVersion = null
|
githubVersion = null
|
||||||
|
@ -115,23 +89,15 @@
|
||||||
>
|
>
|
||||||
</Layout>
|
</Layout>
|
||||||
<Divider />
|
<Divider />
|
||||||
<div>
|
<Layout noPadding gap="XS">
|
||||||
<Button cta on:click={updateModal.show} disabled={!needsUpdate}
|
<Heading>Updating Budibase</Heading>
|
||||||
>Update Budibase</Button
|
<Body
|
||||||
|
>To update your self-host installation, follow the docs found <Link
|
||||||
|
size="L"
|
||||||
|
href="https://docs.budibase.com/docs/updating-budibase">here.</Link
|
||||||
|
></Body
|
||||||
>
|
>
|
||||||
<Modal bind:this={updateModal}>
|
</Layout>
|
||||||
<ModalContent
|
|
||||||
title="Update Budibase"
|
|
||||||
confirmText="Update"
|
|
||||||
onConfirm={updateBudibase}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
>Are you sure you want to update your budibase installation to the
|
|
||||||
latest version?</span
|
|
||||||
>
|
|
||||||
</ModalContent>
|
|
||||||
</Modal>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
</Layout>
|
</Layout>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
|
@ -355,6 +355,93 @@ describe("Loop automations", () => {
|
||||||
expect(results.steps[2].outputs.rows).toHaveLength(expectedRows.length)
|
expect(results.steps[2].outputs.rows).toHaveLength(expectedRows.length)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it("should run an automation with a loop and update row step using stepIds", async () => {
|
||||||
|
const table = await config.createTable({
|
||||||
|
name: "TestTable",
|
||||||
|
type: "table",
|
||||||
|
schema: {
|
||||||
|
name: {
|
||||||
|
name: "name",
|
||||||
|
type: FieldType.STRING,
|
||||||
|
constraints: {
|
||||||
|
presence: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
value: {
|
||||||
|
name: "value",
|
||||||
|
type: FieldType.NUMBER,
|
||||||
|
constraints: {
|
||||||
|
presence: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const rows = [
|
||||||
|
{ name: "Row 1", value: 1, tableId: table._id },
|
||||||
|
{ name: "Row 2", value: 2, tableId: table._id },
|
||||||
|
{ name: "Row 3", value: 3, tableId: table._id },
|
||||||
|
]
|
||||||
|
|
||||||
|
await config.api.row.bulkImport(table._id!, { rows })
|
||||||
|
|
||||||
|
const builder = createAutomationBuilder({
|
||||||
|
name: "Test Loop and Update Row",
|
||||||
|
})
|
||||||
|
|
||||||
|
const results = await builder
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.queryRows(
|
||||||
|
{
|
||||||
|
tableId: table._id!,
|
||||||
|
},
|
||||||
|
{ stepId: "abc123" }
|
||||||
|
)
|
||||||
|
.loop({
|
||||||
|
option: LoopStepType.ARRAY,
|
||||||
|
binding: "{{ steps.abc123.rows }}",
|
||||||
|
})
|
||||||
|
.updateRow({
|
||||||
|
rowId: "{{ loop.currentItem._id }}",
|
||||||
|
row: {
|
||||||
|
name: "Updated {{ loop.currentItem.name }}",
|
||||||
|
value: "{{ loop.currentItem.value }}",
|
||||||
|
tableId: table._id,
|
||||||
|
},
|
||||||
|
meta: {},
|
||||||
|
})
|
||||||
|
.queryRows({
|
||||||
|
tableId: table._id!,
|
||||||
|
})
|
||||||
|
.run()
|
||||||
|
|
||||||
|
const expectedRows = [
|
||||||
|
{ name: "Updated Row 1", value: 1 },
|
||||||
|
{ name: "Updated Row 2", value: 2 },
|
||||||
|
{ name: "Updated Row 3", value: 3 },
|
||||||
|
]
|
||||||
|
|
||||||
|
expect(results.steps[1].outputs.items).toEqual(
|
||||||
|
expect.arrayContaining(
|
||||||
|
expectedRows.map(row =>
|
||||||
|
expect.objectContaining({
|
||||||
|
success: true,
|
||||||
|
row: expect.objectContaining(row),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(results.steps[2].outputs.rows).toEqual(
|
||||||
|
expect.arrayContaining(
|
||||||
|
expectedRows.map(row => expect.objectContaining(row))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
expect(results.steps[1].outputs.items).toHaveLength(expectedRows.length)
|
||||||
|
expect(results.steps[2].outputs.rows).toHaveLength(expectedRows.length)
|
||||||
|
})
|
||||||
|
|
||||||
it("should run an automation with a loop and delete row step", async () => {
|
it("should run an automation with a loop and delete row step", async () => {
|
||||||
const table = await config.createTable({
|
const table = await config.createTable({
|
||||||
name: "TestTable",
|
name: "TestTable",
|
||||||
|
|
|
@ -385,7 +385,7 @@ class Orchestrator {
|
||||||
stepIdx: number,
|
stepIdx: number,
|
||||||
pathIdx?: number
|
pathIdx?: number
|
||||||
): Promise<number> {
|
): Promise<number> {
|
||||||
await processObject(loopStep.inputs, this.context)
|
await processObject(loopStep.inputs, this.processContext(this.context))
|
||||||
const iterations = getLoopIterations(loopStep)
|
const iterations = getLoopIterations(loopStep)
|
||||||
let stepToLoopIndex = stepIdx + 1
|
let stepToLoopIndex = stepIdx + 1
|
||||||
let pathStepIdx = (pathIdx || stepIdx) + 1
|
let pathStepIdx = (pathIdx || stepIdx) + 1
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
export async function wait(ms: number) {
|
||||||
|
return new Promise(resolve => setTimeout(resolve, ms))
|
||||||
|
}
|
|
@ -1,5 +1,7 @@
|
||||||
export * from "./helpers"
|
export * from "./helpers"
|
||||||
export * from "./integrations"
|
export * from "./integrations"
|
||||||
|
export * from "./async"
|
||||||
|
export * from "./retry"
|
||||||
export * as cron from "./cron"
|
export * as cron from "./cron"
|
||||||
export * as schema from "./schema"
|
export * as schema from "./schema"
|
||||||
export * as views from "./views"
|
export * as views from "./views"
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
import { wait } from "./async"
|
||||||
|
|
||||||
|
interface RetryOpts {
|
||||||
|
times?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function retry<T>(
|
||||||
|
fn: () => Promise<T>,
|
||||||
|
opts?: RetryOpts
|
||||||
|
): Promise<T> {
|
||||||
|
const { times = 3 } = opts || {}
|
||||||
|
if (times < 1) {
|
||||||
|
throw new Error(`invalid retry count: ${times}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
let lastError: any
|
||||||
|
for (let i = 0; i < times; i++) {
|
||||||
|
const backoff = 1.5 ** i * 1000 * (Math.random() + 0.5)
|
||||||
|
await wait(backoff)
|
||||||
|
|
||||||
|
try {
|
||||||
|
return await fn()
|
||||||
|
} catch (e) {
|
||||||
|
lastError = e
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw lastError
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ import { Ctx, MaintenanceType, FeatureFlag } from "@budibase/types"
|
||||||
import env from "../../../environment"
|
import env from "../../../environment"
|
||||||
import { env as coreEnv, db as dbCore, features } from "@budibase/backend-core"
|
import { env as coreEnv, db as dbCore, features } from "@budibase/backend-core"
|
||||||
import nodeFetch from "node-fetch"
|
import nodeFetch from "node-fetch"
|
||||||
|
import { helpers } from "@budibase/shared-core"
|
||||||
|
|
||||||
let sqsAvailable: boolean
|
let sqsAvailable: boolean
|
||||||
async function isSqsAvailable() {
|
async function isSqsAvailable() {
|
||||||
|
@ -12,17 +13,22 @@ async function isSqsAvailable() {
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const couchInfo = dbCore.getCouchInfo()
|
const { url } = dbCore.getCouchInfo()
|
||||||
if (!couchInfo.sqlUrl) {
|
if (!url) {
|
||||||
sqsAvailable = false
|
sqsAvailable = false
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
await nodeFetch(couchInfo.sqlUrl, {
|
await helpers.retry(
|
||||||
timeout: 1000,
|
async () => {
|
||||||
})
|
await nodeFetch(url, { timeout: 2000 })
|
||||||
|
},
|
||||||
|
{ times: 3 }
|
||||||
|
)
|
||||||
|
console.log("connected to SQS")
|
||||||
sqsAvailable = true
|
sqsAvailable = true
|
||||||
return true
|
return true
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
console.warn("failed to connect to SQS", e)
|
||||||
sqsAvailable = false
|
sqsAvailable = false
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue