Merge branch 'master' into BUDI-8428/row-action-crud

This commit is contained in:
Adria Navarro 2024-07-12 11:53:12 +02:00 committed by GitHub
commit 99b4aae7de
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
17 changed files with 158 additions and 118 deletions

View File

@ -1,6 +1,6 @@
{ {
"$schema": "node_modules/lerna/schemas/lerna-schema.json", "$schema": "node_modules/lerna/schemas/lerna-schema.json",
"version": "2.29.15", "version": "2.29.19",
"npmClient": "yarn", "npmClient": "yarn",
"packages": [ "packages": [
"packages/*", "packages/*",

View File

@ -25,7 +25,10 @@ export const getCouchInfo = (connection?: string) => {
} }
const authCookie = Buffer.from(`${username}:${password}`).toString("base64") const authCookie = Buffer.from(`${username}:${password}`).toString("base64")
let sqlUrl = env.COUCH_DB_SQL_URL let sqlUrl = env.COUCH_DB_SQL_URL
if (!sqlUrl && urlInfo.url) { // default for dev
if (env.isDev() && !sqlUrl) {
sqlUrl = "http://localhost:4006"
} else if (!sqlUrl && urlInfo.url) {
const parsed = new URL(urlInfo.url) const parsed = new URL(urlInfo.url)
// attempt to connect on default port // attempt to connect on default port
sqlUrl = urlInfo.url.replace(parsed.port, "4984") sqlUrl = urlInfo.url.replace(parsed.port, "4984")

View File

@ -1,6 +1,6 @@
import env from "../environment" import env from "../environment"
import { DEFAULT_TENANT_ID, SEPARATOR, DocumentType } from "../constants" import { DEFAULT_TENANT_ID, SEPARATOR, DocumentType } from "../constants"
import { getTenantId, getGlobalDBName } from "../context" import { getTenantId, getGlobalDBName, isMultiTenant } from "../context"
import { doWithDB, directCouchAllDbs } from "./db" import { doWithDB, directCouchAllDbs } from "./db"
import { AppState, DeletedApp, getAppMetadata } from "../cache/appMetadata" import { AppState, DeletedApp, getAppMetadata } from "../cache/appMetadata"
import { isDevApp, isDevAppID, getProdAppID } from "../docIds/conversions" import { isDevApp, isDevAppID, getProdAppID } from "../docIds/conversions"
@ -213,6 +213,11 @@ export function isSqsEnabledForTenant(): boolean {
return false return false
} }
// single tenant (self host and dev) always enabled if flag set
if (!isMultiTenant()) {
return true
}
// This is to guard against the situation in tests where tests pass because // This is to guard against the situation in tests where tests pass because
// we're not actually using SQS, we're using Lucene and the tests pass due to // we're not actually using SQS, we're using Lucene and the tests pass due to
// parity. // parity.
@ -222,5 +227,13 @@ export function isSqsEnabledForTenant(): boolean {
) )
} }
// Special case to enable all tenants, for testing in QA.
if (
env.SQS_SEARCH_ENABLE_TENANTS.length === 1 &&
env.SQS_SEARCH_ENABLE_TENANTS[0] === "*"
) {
return true
}
return env.SQS_SEARCH_ENABLE_TENANTS.includes(tenantId) return env.SQS_SEARCH_ENABLE_TENANTS.includes(tenantId)
} }

View File

@ -114,7 +114,7 @@ const environment = {
ENCRYPTION_KEY: process.env.ENCRYPTION_KEY, ENCRYPTION_KEY: process.env.ENCRYPTION_KEY,
API_ENCRYPTION_KEY: getAPIEncryptionKey(), API_ENCRYPTION_KEY: getAPIEncryptionKey(),
COUCH_DB_URL: process.env.COUCH_DB_URL || "http://localhost:4005", COUCH_DB_URL: process.env.COUCH_DB_URL || "http://localhost:4005",
COUCH_DB_SQL_URL: process.env.COUCH_DB_SQL_URL || "http://localhost:4006", COUCH_DB_SQL_URL: process.env.COUCH_DB_SQL_URL,
SQS_SEARCH_ENABLE: process.env.SQS_SEARCH_ENABLE, SQS_SEARCH_ENABLE: process.env.SQS_SEARCH_ENABLE,
SQS_SEARCH_ENABLE_TENANTS: SQS_SEARCH_ENABLE_TENANTS:
process.env.SQS_SEARCH_ENABLE_TENANTS?.split(",") || [], process.env.SQS_SEARCH_ENABLE_TENANTS?.split(",") || [],

View File

@ -2,6 +2,7 @@ import { GenericContainer, StartedTestContainer } from "testcontainers"
import { generator, structures } from "../../../tests" import { generator, structures } from "../../../tests"
import RedisWrapper from "../redis" import RedisWrapper from "../redis"
import { env } from "../.." import { env } from "../.."
import { randomUUID } from "crypto"
jest.setTimeout(30000) jest.setTimeout(30000)
@ -52,10 +53,10 @@ describe("redis", () => {
describe("bulkStore", () => { describe("bulkStore", () => {
function createRandomObject( function createRandomObject(
keyLength: number, keyLength: number,
valueGenerator: () => any = () => generator.word() valueGenerator: () => any = () => randomUUID()
) { ) {
return generator return generator
.unique(() => generator.word(), keyLength) .unique(() => randomUUID(), keyLength)
.reduce((acc, key) => { .reduce((acc, key) => {
acc[key] = valueGenerator() acc[key] = valueGenerator()
return acc return acc

View File

@ -30,6 +30,16 @@
return lowerA > lowerB ? 1 : -1 return lowerA > lowerB ? 1 : -1
}) })
$: groupedAutomations = filteredAutomations.reduce((acc, auto) => {
acc[auto.definition.trigger.event] ??= {
icon: auto.definition.trigger.icon,
name: (auto.definition.trigger?.name || "").toUpperCase(),
entries: [],
}
acc[auto.definition.trigger.event].entries.push(auto)
return acc
}, {})
$: showNoResults = searchString && !filteredAutomations.length $: showNoResults = searchString && !filteredAutomations.length
onMount(async () => { onMount(async () => {
@ -55,16 +65,25 @@
/> />
</div> </div>
<div class="side-bar-nav"> <div class="side-bar-nav">
{#each filteredAutomations as automation} {#each Object.values(groupedAutomations || {}) as triggerGroup}
<NavItem <div class="nav-group">
text={automation.name} <div class="nav-group-header" title={triggerGroup?.name}>
selected={automation._id === selectedAutomationId} {triggerGroup?.name}
on:click={() => selectAutomation(automation._id)} </div>
selectedBy={$userSelectedResourceMap[automation._id]} {#each triggerGroup.entries as automation}
disabled={automation.disabled} <NavItem
> icon={triggerGroup.icon}
<EditAutomationPopover {automation} /> iconColor={"var(--spectrum-global-color-gray-900)"}
</NavItem> text={automation.name}
selected={automation._id === selectedAutomationId}
on:click={() => selectAutomation(automation._id)}
selectedBy={$userSelectedResourceMap[automation._id]}
disabled={automation.disabled}
>
<EditAutomationPopover {automation} />
</NavItem>
{/each}
</div>
{/each} {/each}
{#if showNoResults} {#if showNoResults}
@ -82,6 +101,17 @@
</Modal> </Modal>
<style> <style>
.nav-group {
padding-top: var(--spacing-l);
}
.nav-group-header {
color: var(--spectrum-global-color-gray-600);
padding: 0px calc(var(--spacing-l) + 4px);
padding-bottom: var(--spacing-l);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.side-bar { .side-bar {
flex: 0 0 260px; flex: 0 0 260px;
display: flex; display: flex;
@ -104,7 +134,7 @@
justify-content: flex-start; justify-content: flex-start;
align-items: center; align-items: center;
gap: var(--spacing-l); gap: var(--spacing-l);
padding: 0 var(--spacing-l); padding: 0 calc(var(--spacing-l) + 4px);
} }
.side-bar-nav { .side-bar-nav {
flex: 1 1 auto; flex: 1 1 auto;

View File

@ -250,6 +250,7 @@
}, },
})) }))
}, },
placeholder: false,
getOptionLabel: type => type.name, getOptionLabel: type => type.name,
getOptionValue: type => type.id, getOptionValue: type => type.id,
options: [ options: [
@ -280,14 +281,14 @@
tableId: inputData["row"].tableId, tableId: inputData["row"].tableId,
}, },
meta: { meta: {
fields: inputData["meta"].oldFields || {}, fields: inputData["meta"]?.oldFields || {},
}, },
onChange: e => { onChange: e => {
onChange({ onChange({
oldRow: e.detail.row, oldRow: e.detail.row,
meta: { meta: {
fields: inputData["meta"].fields, fields: inputData["meta"]?.fields || {},
oldFields: e.detail.meta.fields, oldFields: e.detail?.meta?.fields || {},
}, },
}) })
}, },
@ -304,7 +305,15 @@
row: inputData["row"], row: inputData["row"],
meta: inputData["meta"] || {}, meta: inputData["meta"] || {},
onChange: e => { onChange: e => {
onChange(e.detail) onChange({
row: e.detail.row,
meta: {
fields: e.detail?.meta?.fields || {},
...(isTestModal
? { oldFields: inputData["meta"]?.oldFields || {} }
: {}),
},
})
}, },
...baseProps, ...baseProps,
}, },
@ -452,7 +461,6 @@
*/ */
const onChange = Utils.sequential(async update => { const onChange = Utils.sequential(async update => {
const request = cloneDeep(update) const request = cloneDeep(update)
// Process app trigger updates // Process app trigger updates
if (isTrigger && !isTestModal) { if (isTrigger && !isTestModal) {
// Row trigger // Row trigger

View File

@ -288,29 +288,27 @@
{/each} {/each}
{#if table && schemaFields} {#if table && schemaFields}
{#key editableFields} <div
<div class="add-fields-btn"
class="add-fields-btn" class:empty={Object.is(editableFields, {})}
class:empty={Object.is(editableFields, {})} bind:this={popoverAnchor}
bind:this={popoverAnchor} >
> <ActionButton
<ActionButton icon="Add"
icon="Add" fullWidth
fullWidth on:click={() => {
on:click={() => { customPopover.show()
customPopover.show() }}
}} disabled={!schemaFields}
disabled={!schemaFields} >Add fields
>Add fields </ActionButton>
</ActionButton> </div>
</div>
{/key}
{/if} {/if}
<Popover <Popover
align="center" align="center"
bind:this={customPopover} bind:this={customPopover}
anchor={popoverAnchor} anchor={editableFields ? popoverAnchor : null}
useAnchorWidth useAnchorWidth
maxHeight={300} maxHeight={300}
resizable={false} resizable={false}

View File

@ -819,7 +819,10 @@ describe.each([
const table = await config.api.table.save(tableRequest) const table = await config.api.table.save(tableRequest)
const stringValue = generator.word() const stringValue = generator.word()
const naturalValue = generator.integer({ min: 0, max: 1000 })
// MySQL and MariaDB auto-increment fields have a minimum value of 1. If
// you try to save a row with a value of 0 it will use 1 instead.
const naturalValue = generator.integer({ min: 1, max: 1000 })
const existing = await config.api.row.save(table._id!, { const existing = await config.api.row.save(table._id!, {
string: stringValue, string: stringValue,
@ -1457,17 +1460,12 @@ describe.each([
delete tableRequest.schema.id delete tableRequest.schema.id
const table = await config.api.table.save(tableRequest) const table = await config.api.table.save(tableRequest)
const toCreate = generator
.unique(() => generator.integer({ min: 0, max: 10000 }), 10)
.map(number => ({ number, string: generator.word({ length: 30 }) }))
const rows = await Promise.all( const rows = await Promise.all(
generator toCreate.map(d => config.api.row.save(table._id!, d))
.unique(
() => ({
string: generator.word({ length: 30 }),
number: generator.integer({ min: 0, max: 10000 }),
}),
10
)
.map(d => config.api.row.save(table._id!, d))
) )
const res = await config.api.row.exportRows(table._id!, { const res = await config.api.row.exportRows(table._id!, {

View File

@ -809,6 +809,20 @@ describe.each([
}, },
}).toContainExactly([{ name: "foo" }, { name: "bar" }]) }).toContainExactly([{ name: "foo" }, { name: "bar" }])
}) })
it("empty arrays returns all when onEmptyFilter is set to return 'all'", async () => {
await expectQuery({
onEmptyFilter: EmptyFilterOption.RETURN_ALL,
oneOf: { name: [] },
}).toContainExactly([{ name: "foo" }, { name: "bar" }])
})
it("empty arrays returns all when onEmptyFilter is set to return 'none'", async () => {
await expectQuery({
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
oneOf: { name: [] },
}).toContainExactly([])
})
}) })
describe("fuzzy", () => { describe("fuzzy", () => {

View File

@ -151,7 +151,7 @@ export const checkPermissionsEndpoint = async ({
await exports await exports
.createRequest(config.request, method, url, body) .createRequest(config.request, method, url, body)
.set(failHeader) .set(failHeader)
.expect(403) .expect(401)
} }
export const getDB = () => { export const getDB = () => {

View File

@ -24,16 +24,6 @@ export enum FilterTypes {
ONE_OF = "oneOf", ONE_OF = "oneOf",
} }
export const NoEmptyFilterStrings = [
FilterTypes.STRING,
FilterTypes.FUZZY,
FilterTypes.EQUAL,
FilterTypes.NOT_EQUAL,
FilterTypes.CONTAINS,
FilterTypes.NOT_CONTAINS,
FilterTypes.CONTAINS_ANY,
]
export const CanSwitchTypes = [ export const CanSwitchTypes = [
[FieldType.JSON, FieldType.ARRAY], [FieldType.JSON, FieldType.ARRAY],
[ [

View File

@ -28,7 +28,6 @@ const DEFAULTS = {
PLUGINS_DIR: "/plugins", PLUGINS_DIR: "/plugins",
FORKED_PROCESS_NAME: "main", FORKED_PROCESS_NAME: "main",
JS_RUNNER_MEMORY_LIMIT: 64, JS_RUNNER_MEMORY_LIMIT: 64,
COUCH_DB_SQL_URL: "http://localhost:4006",
} }
const QUERY_THREAD_TIMEOUT = const QUERY_THREAD_TIMEOUT =
@ -44,7 +43,7 @@ const environment = {
// important - prefer app port to generic port // important - prefer app port to generic port
PORT: process.env.APP_PORT || process.env.PORT, PORT: process.env.APP_PORT || process.env.PORT,
COUCH_DB_URL: process.env.COUCH_DB_URL, COUCH_DB_URL: process.env.COUCH_DB_URL,
COUCH_DB_SQL_URL: process.env.COUCH_DB_SQL_URL || DEFAULTS.COUCH_DB_SQL_URL, COUCH_DB_SQL_URL: process.env.COUCH_DB_SQL_URL,
MINIO_URL: process.env.MINIO_URL, MINIO_URL: process.env.MINIO_URL,
WORKER_URL: process.env.WORKER_URL, WORKER_URL: process.env.WORKER_URL,
AWS_REGION: process.env.AWS_REGION, AWS_REGION: process.env.AWS_REGION,

View File

@ -2,14 +2,12 @@ import {
EmptyFilterOption, EmptyFilterOption,
Row, Row,
RowSearchParams, RowSearchParams,
SearchFilters,
SearchResponse, SearchResponse,
SortOrder, SortOrder,
} from "@budibase/types" } from "@budibase/types"
import { isExternalTableID } from "../../../integrations/utils" import { isExternalTableID } from "../../../integrations/utils"
import * as internal from "./search/internal" import * as internal from "./search/internal"
import * as external from "./search/external" import * as external from "./search/external"
import { NoEmptyFilterStrings } from "../../../constants"
import * as sqs from "./search/sqs" import * as sqs from "./search/sqs"
import { ExportRowsParams, ExportRowsResult } from "./search/types" import { ExportRowsParams, ExportRowsResult } from "./search/types"
import { dataFilters } from "@budibase/shared-core" import { dataFilters } from "@budibase/shared-core"
@ -32,44 +30,11 @@ function pickApi(tableId: any) {
return internal return internal
} }
function isEmptyArray(value: any) {
return Array.isArray(value) && value.length === 0
}
// don't do a pure falsy check, as 0 is included
// https://github.com/Budibase/budibase/issues/10118
export function removeEmptyFilters(filters: SearchFilters) {
for (let filterField of NoEmptyFilterStrings) {
if (!filters[filterField]) {
continue
}
for (let filterType of Object.keys(filters)) {
if (filterType !== filterField) {
continue
}
// don't know which one we're checking, type could be anything
const value = filters[filterType] as unknown
if (typeof value === "object") {
for (let [key, value] of Object.entries(
filters[filterType] as object
)) {
if (value == null || value === "" || isEmptyArray(value)) {
// @ts-ignore
delete filters[filterField][key]
}
}
}
}
}
return filters
}
export async function search( export async function search(
options: RowSearchParams options: RowSearchParams
): Promise<SearchResponse<Row>> { ): Promise<SearchResponse<Row>> {
const isExternalTable = isExternalTableID(options.tableId) const isExternalTable = isExternalTableID(options.tableId)
options.query = removeEmptyFilters(options.query || {}) options.query = dataFilters.cleanupQuery(options.query || {})
options.query = dataFilters.fixupFilterArrays(options.query) options.query = dataFilters.fixupFilterArrays(options.query)
if ( if (
!dataFilters.hasFilters(options.query) && !dataFilters.hasFilters(options.query) &&

View File

@ -45,13 +45,10 @@ import {
getTableIDList, getTableIDList,
} from "./filters" } from "./filters"
import { dataFilters } from "@budibase/shared-core" import { dataFilters } from "@budibase/shared-core"
import { DEFAULT_TABLE_IDS } from "../../../../constants"
const builder = new sql.Sql(SqlClient.SQL_LITE) const builder = new sql.Sql(SqlClient.SQL_LITE)
const MISSING_COLUMN_REGEX = new RegExp(`no such column: .+`) const MISSING_COLUMN_REGEX = new RegExp(`no such column: .+`)
const USER_COLUMN_PREFIX_REGEX = new RegExp( const MISSING_TABLE_REGX = new RegExp(`no such table: .+`)
`no such column: .+${USER_COLUMN_PREFIX}`
)
function buildInternalFieldList( function buildInternalFieldList(
table: Table, table: Table,
@ -240,10 +237,10 @@ async function runSqlQuery(
function resyncDefinitionsRequired(status: number, message: string) { function resyncDefinitionsRequired(status: number, message: string) {
// pre data_ prefix on column names, need to resync // pre data_ prefix on column names, need to resync
return ( return (
(status === 400 && message?.match(USER_COLUMN_PREFIX_REGEX)) || // there are tables missing - try a resync
// default tables aren't included in definition (status === 400 && message.match(MISSING_TABLE_REGX)) ||
(status === 400 && // there are columns missing - try a resync
DEFAULT_TABLE_IDS.find(tableId => message?.includes(tableId))) || (status === 400 && message.match(MISSING_COLUMN_REGEX)) ||
// no design document found, needs a full sync // no design document found, needs a full sync
(status === 404 && message?.includes(SQLITE_DESIGN_DOC_ID)) (status === 404 && message?.includes(SQLITE_DESIGN_DOC_ID))
) )
@ -251,7 +248,8 @@ function resyncDefinitionsRequired(status: number, message: string) {
export async function search( export async function search(
options: RowSearchParams, options: RowSearchParams,
table: Table table: Table,
opts?: { retrying?: boolean }
): Promise<SearchResponse<Row>> { ): Promise<SearchResponse<Row>> {
let { paginate, query, ...params } = options let { paginate, query, ...params } = options
@ -376,9 +374,9 @@ export async function search(
return response return response
} catch (err: any) { } catch (err: any) {
const msg = typeof err === "string" ? err : err.message const msg = typeof err === "string" ? err : err.message
if (resyncDefinitionsRequired(err.status, msg)) { if (!opts?.retrying && resyncDefinitionsRequired(err.status, msg)) {
await sdk.tables.sqs.syncDefinition() await sdk.tables.sqs.syncDefinition()
return search(options, table) return search(options, table, { retrying: true })
} }
// previously the internal table didn't error when a column didn't exist in search // previously the internal table didn't error when a column didn't exist in search
if (err.status === 400 && msg?.match(MISSING_COLUMN_REGEX)) { if (err.status === 400 && msg?.match(MISSING_COLUMN_REGEX)) {

View File

@ -106,31 +106,49 @@ export const NoEmptyFilterStrings = [
OperatorOptions.NotEquals.value, OperatorOptions.NotEquals.value,
OperatorOptions.Contains.value, OperatorOptions.Contains.value,
OperatorOptions.NotContains.value, OperatorOptions.NotContains.value,
OperatorOptions.ContainsAny.value,
OperatorOptions.In.value,
] as (keyof SearchQueryFields)[] ] as (keyof SearchQueryFields)[]
/** /**
* Removes any fields that contain empty strings that would cause inconsistent * Removes any fields that contain empty strings that would cause inconsistent
* behaviour with how backend tables are filtered (no value means no filter). * behaviour with how backend tables are filtered (no value means no filter).
*
* don't do a pure falsy check, as 0 is included
* https://github.com/Budibase/budibase/issues/10118
*/ */
const cleanupQuery = (query: SearchFilters) => { export const cleanupQuery = (query: SearchFilters) => {
if (!query) { if (!query) {
return query return query
} }
for (let filterField of NoEmptyFilterStrings) { for (let filterField of NoEmptyFilterStrings) {
const operator = filterField as SearchFilterOperator if (!query[filterField]) {
if (!query[operator]) {
continue continue
} }
for (let [key, value] of Object.entries(query[operator]!)) { for (let filterType of Object.keys(query)) {
if (value == null || value === "") { if (filterType !== filterField) {
delete query[operator]![key] continue
}
// don't know which one we're checking, type could be anything
const value = query[filterType] as unknown
if (typeof value === "object") {
for (let [key, value] of Object.entries(query[filterType] as object)) {
if (value == null || value === "" || isEmptyArray(value)) {
// @ts-ignore
delete query[filterField][key]
}
}
} }
} }
} }
return query return query
} }
function isEmptyArray(value: any) {
return Array.isArray(value) && value.length === 0
}
/** /**
* Removes a numeric prefix on field names designed to give fields uniqueness * Removes a numeric prefix on field names designed to give fields uniqueness
*/ */

View File

@ -1,6 +1,6 @@
import { Ctx, MaintenanceType } from "@budibase/types" import { Ctx, MaintenanceType } from "@budibase/types"
import env from "../../../environment" import env from "../../../environment"
import { env as coreEnv } from "@budibase/backend-core" import { env as coreEnv, db as dbCore } from "@budibase/backend-core"
import nodeFetch from "node-fetch" import nodeFetch from "node-fetch"
let sqsAvailable: boolean let sqsAvailable: boolean
@ -12,7 +12,12 @@ async function isSqsAvailable() {
} }
try { try {
await nodeFetch(coreEnv.COUCH_DB_SQL_URL, { const couchInfo = dbCore.getCouchInfo()
if (!couchInfo.sqlUrl) {
sqsAvailable = false
return false
}
await nodeFetch(couchInfo.sqlUrl, {
timeout: 1000, timeout: 1000,
}) })
sqsAvailable = true sqsAvailable = true