Merge branch 'master' into BUDI-8428/row-action-crud
This commit is contained in:
commit
99b4aae7de
|
@ -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/*",
|
||||||
|
|
|
@ -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")
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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(",") || [],
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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}
|
||||||
|
|
|
@ -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!, {
|
||||||
|
|
|
@ -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", () => {
|
||||||
|
|
|
@ -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 = () => {
|
||||||
|
|
|
@ -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],
|
||||||
[
|
[
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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) &&
|
||||||
|
|
|
@ -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)) {
|
||||||
|
|
|
@ -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
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue