2024-09-30 16:16:24 +02:00
|
|
|
import { ArrayOperator, BasicOperator, SearchFilters } from "@budibase/types"
|
2023-11-07 16:07:00 +01:00
|
|
|
import * as Constants from "./constants"
|
|
|
|
|
2023-03-09 09:50:26 +01:00
|
|
|
export function unreachable(
|
|
|
|
value: never,
|
|
|
|
message = `No such case in exhaustive switch: ${value}`
|
|
|
|
) {
|
|
|
|
throw new Error(message)
|
|
|
|
}
|
2023-06-07 13:29:36 +02:00
|
|
|
|
|
|
|
export async function parallelForeach<T>(
|
|
|
|
items: T[],
|
|
|
|
task: (item: T) => Promise<void>,
|
|
|
|
maxConcurrency: number
|
|
|
|
): Promise<void> {
|
|
|
|
const promises: Promise<void>[] = []
|
|
|
|
let index = 0
|
|
|
|
|
|
|
|
const processItem = async (item: T) => {
|
|
|
|
try {
|
|
|
|
await task(item)
|
|
|
|
} finally {
|
|
|
|
processNext()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const processNext = () => {
|
|
|
|
if (index >= items.length) {
|
|
|
|
// No more items to process
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
const item = items[index]
|
|
|
|
index++
|
|
|
|
|
|
|
|
const promise = processItem(item)
|
|
|
|
promises.push(promise)
|
|
|
|
|
|
|
|
if (promises.length >= maxConcurrency) {
|
|
|
|
Promise.race(promises).then(processNext)
|
|
|
|
} else {
|
|
|
|
processNext()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
processNext()
|
|
|
|
|
|
|
|
await Promise.all(promises)
|
|
|
|
}
|
2023-11-07 16:07:00 +01:00
|
|
|
|
|
|
|
export function filterValueToLabel() {
|
|
|
|
return Object.keys(Constants.OperatorOptions).reduce(
|
|
|
|
(acc: { [key: string]: string }, key: string) => {
|
|
|
|
const ops: { [key: string]: any } = Constants.OperatorOptions
|
|
|
|
const op: { [key: string]: string } = ops[key]
|
|
|
|
acc[op["value"]] = op.label
|
|
|
|
return acc
|
|
|
|
},
|
|
|
|
{}
|
|
|
|
)
|
|
|
|
}
|
2024-02-19 10:13:03 +01:00
|
|
|
|
|
|
|
export function hasSchema(test: any) {
|
|
|
|
return (
|
|
|
|
typeof test === "object" &&
|
|
|
|
!Array.isArray(test) &&
|
|
|
|
test !== null &&
|
|
|
|
!(test instanceof Date) &&
|
|
|
|
Object.keys(test).length > 0
|
|
|
|
)
|
|
|
|
}
|
2024-08-01 11:02:21 +02:00
|
|
|
|
|
|
|
export function trimOtherProps(object: any, allowedProps: string[]) {
|
|
|
|
const result = Object.keys(object)
|
|
|
|
.filter(key => allowedProps.includes(key))
|
|
|
|
.reduce<Record<string, any>>(
|
|
|
|
(acc, key) => ({ ...acc, [key]: object[key] }),
|
|
|
|
{}
|
|
|
|
)
|
|
|
|
return result
|
|
|
|
}
|
2024-09-30 16:16:24 +02:00
|
|
|
|
|
|
|
export function isSupportedUserSearch(query: SearchFilters) {
|
|
|
|
const allowed = [
|
|
|
|
{ op: BasicOperator.STRING, key: "email" },
|
|
|
|
{ op: BasicOperator.EQUAL, key: "_id" },
|
|
|
|
{ op: ArrayOperator.ONE_OF, key: "_id" },
|
|
|
|
]
|
|
|
|
for (let [key, operation] of Object.entries(query)) {
|
|
|
|
if (typeof operation !== "object") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
const fields = Object.keys(operation || {})
|
|
|
|
// this filter doesn't contain options - ignore
|
|
|
|
if (fields.length === 0) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
const allowedOperation = allowed.find(
|
|
|
|
allow =>
|
|
|
|
allow.op === key && fields.length === 1 && fields[0] === allow.key
|
|
|
|
)
|
|
|
|
if (!allowedOperation) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|