Merge branch 'master' into fix/got-version
This commit is contained in:
commit
196038368e
|
@ -1,5 +1,5 @@
|
||||||
{
|
{
|
||||||
"version": "2.19.0",
|
"version": "2.19.1",
|
||||||
"npmClient": "yarn",
|
"npmClient": "yarn",
|
||||||
"packages": [
|
"packages": [
|
||||||
"packages/*",
|
"packages/*",
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit aaf64dd002a7d2b9ff37f989e03d444c9b9f2b36
|
Subproject commit 8cbe73c9dc39d0bda3dbca3dac345a4a1f916f18
|
|
@ -617,6 +617,7 @@ const getDeviceBindings = () => {
|
||||||
/**
|
/**
|
||||||
* Gets all selected rows bindings for tables in the current asset.
|
* Gets all selected rows bindings for tables in the current asset.
|
||||||
* TODO: remove in future because we don't need a separate store for this
|
* TODO: remove in future because we don't need a separate store for this
|
||||||
|
* DEPRECATED
|
||||||
*/
|
*/
|
||||||
const getSelectedRowsBindings = asset => {
|
const getSelectedRowsBindings = asset => {
|
||||||
let bindings = []
|
let bindings = []
|
||||||
|
@ -632,8 +633,8 @@ const getSelectedRowsBindings = asset => {
|
||||||
runtimeBinding: `${safeState}.${makePropSafe(table._id)}.${makePropSafe(
|
runtimeBinding: `${safeState}.${makePropSafe(table._id)}.${makePropSafe(
|
||||||
"selectedRows"
|
"selectedRows"
|
||||||
)}`,
|
)}`,
|
||||||
readableBinding: `${table._instanceName}.Selected rows`,
|
readableBinding: `${table._instanceName}.Selected Row IDs (deprecated)`,
|
||||||
category: "Selected rows",
|
category: "Selected Row IDs (deprecated)",
|
||||||
icon: "ViewRow",
|
icon: "ViewRow",
|
||||||
display: { name: table._instanceName },
|
display: { name: table._instanceName },
|
||||||
}))
|
}))
|
||||||
|
@ -649,8 +650,8 @@ const getSelectedRowsBindings = asset => {
|
||||||
runtimeBinding: `${safeState}.${makePropSafe(
|
runtimeBinding: `${safeState}.${makePropSafe(
|
||||||
block._id + "-table"
|
block._id + "-table"
|
||||||
)}.${makePropSafe("selectedRows")}`,
|
)}.${makePropSafe("selectedRows")}`,
|
||||||
readableBinding: `${block._instanceName}.Selected rows`,
|
readableBinding: `${block._instanceName}.Selected Row IDs (deprecated)`,
|
||||||
category: "Selected rows",
|
category: "Selected Row IDs (deprecated)",
|
||||||
icon: "ViewRow",
|
icon: "ViewRow",
|
||||||
display: { name: block._instanceName },
|
display: { name: block._instanceName },
|
||||||
}))
|
}))
|
||||||
|
|
|
@ -2,15 +2,34 @@
|
||||||
import { Button, Select, Input, Label } from "@budibase/bbui"
|
import { Button, Select, Input, Label } from "@budibase/bbui"
|
||||||
import { onMount, createEventDispatcher } from "svelte"
|
import { onMount, createEventDispatcher } from "svelte"
|
||||||
import { flags } from "stores/backend"
|
import { flags } from "stores/backend"
|
||||||
|
import { helpers, REBOOT_CRON } from "@budibase/shared-core"
|
||||||
|
|
||||||
const dispatch = createEventDispatcher()
|
const dispatch = createEventDispatcher()
|
||||||
|
|
||||||
export let value
|
export let value
|
||||||
|
let error
|
||||||
|
|
||||||
|
$: {
|
||||||
|
const exists = CRON_EXPRESSIONS.some(cron => cron.value === value)
|
||||||
|
const customIndex = CRON_EXPRESSIONS.findIndex(
|
||||||
|
cron => cron.label === "Custom"
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!exists && customIndex === -1) {
|
||||||
|
CRON_EXPRESSIONS[0] = { label: "Custom", value: value }
|
||||||
|
} else if (exists && customIndex !== -1) {
|
||||||
|
CRON_EXPRESSIONS.splice(customIndex, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const onChange = e => {
|
const onChange = e => {
|
||||||
if (e.detail === value) {
|
if (value !== REBOOT_CRON) {
|
||||||
|
error = helpers.cron.validate(e.detail).err
|
||||||
|
}
|
||||||
|
if (e.detail === value || error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
value = e.detail
|
value = e.detail
|
||||||
dispatch("change", e.detail)
|
dispatch("change", e.detail)
|
||||||
}
|
}
|
||||||
|
@ -41,7 +60,7 @@
|
||||||
if (!$flags.cloud) {
|
if (!$flags.cloud) {
|
||||||
CRON_EXPRESSIONS.push({
|
CRON_EXPRESSIONS.push({
|
||||||
label: "Every Budibase Reboot",
|
label: "Every Budibase Reboot",
|
||||||
value: "@reboot",
|
value: REBOOT_CRON,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -49,6 +68,7 @@
|
||||||
|
|
||||||
<div class="block-field">
|
<div class="block-field">
|
||||||
<Input
|
<Input
|
||||||
|
{error}
|
||||||
on:change={onChange}
|
on:change={onChange}
|
||||||
{value}
|
{value}
|
||||||
on:blur={() => (touched = true)}
|
on:blur={() => (touched = true)}
|
||||||
|
@ -64,7 +84,7 @@
|
||||||
{#if presets}
|
{#if presets}
|
||||||
<Select
|
<Select
|
||||||
on:change={onChange}
|
on:change={onChange}
|
||||||
{value}
|
value={value || "Custom"}
|
||||||
secondary
|
secondary
|
||||||
extraThin
|
extraThin
|
||||||
label="Presets"
|
label="Presets"
|
||||||
|
|
|
@ -101,7 +101,13 @@
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error)
|
console.error(error)
|
||||||
analytics.captureException(error)
|
analytics.captureException(error)
|
||||||
notifications.error("Error publishing app")
|
const baseMsg = "Error publishing app"
|
||||||
|
const message = error.message
|
||||||
|
if (message) {
|
||||||
|
notifications.error(`${baseMsg} - ${message}`)
|
||||||
|
} else {
|
||||||
|
notifications.error(baseMsg)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
publishing = false
|
publishing = false
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
|
|
||||||
$: editing = app.sessions?.length
|
$: editing = app.sessions?.length
|
||||||
$: isBuilder = sdk.users.isBuilder($auth.user, app?.devId)
|
$: isBuilder = sdk.users.isBuilder($auth.user, app?.devId)
|
||||||
|
$: unclickable = !isBuilder && !app.deployed
|
||||||
|
|
||||||
const handleDefaultClick = () => {
|
const handleDefaultClick = () => {
|
||||||
if (!isBuilder) {
|
if (!isBuilder) {
|
||||||
|
@ -31,11 +32,17 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
const goToApp = () => {
|
const goToApp = () => {
|
||||||
window.open(`/app/${app.name}`, "_blank")
|
if (app.deployed && app.url) {
|
||||||
|
window.open(`/app${app.url}`, "_blank")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="app-row" on:click={lockedAction || handleDefaultClick}>
|
<div
|
||||||
|
class="app-row"
|
||||||
|
on:click={lockedAction || handleDefaultClick}
|
||||||
|
class:unclickable
|
||||||
|
>
|
||||||
<div class="title">
|
<div class="title">
|
||||||
<div class="app-icon">
|
<div class="app-icon">
|
||||||
<Icon size="L" name={app.icon?.name || "Apps"} color={app.icon?.color} />
|
<Icon size="L" name={app.icon?.name || "Apps"} color={app.icon?.color} />
|
||||||
|
@ -74,7 +81,7 @@
|
||||||
Edit
|
Edit
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else if app.deployed}
|
||||||
<!-- this can happen if an app builder has app user access to an app -->
|
<!-- this can happen if an app builder has app user access to an app -->
|
||||||
<div class="app-row-actions">
|
<div class="app-row-actions">
|
||||||
<Button size="S" secondary>View</Button>
|
<Button size="S" secondary>View</Button>
|
||||||
|
@ -94,7 +101,7 @@
|
||||||
transition: border 130ms ease-out;
|
transition: border 130ms ease-out;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
}
|
}
|
||||||
.app-row:hover {
|
.app-row:not(.unclickable):hover {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
border-color: var(--spectrum-global-color-gray-300);
|
border-color: var(--spectrum-global-color-gray-300);
|
||||||
}
|
}
|
||||||
|
|
|
@ -4719,10 +4719,22 @@
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"context": {
|
"context": [
|
||||||
"type": "schema",
|
{
|
||||||
"scope": "local"
|
"type": "schema",
|
||||||
}
|
"scope": "local"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "static",
|
||||||
|
"values": [
|
||||||
|
{
|
||||||
|
"label": "Selected Rows",
|
||||||
|
"key": "selectedRows",
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"daterangepicker": {
|
"daterangepicker": {
|
||||||
"name": "Date Range",
|
"name": "Date Range",
|
||||||
|
@ -5610,37 +5622,50 @@
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"context": {
|
"context": [
|
||||||
"type": "static",
|
{
|
||||||
"suffix": "provider",
|
"type": "static",
|
||||||
"values": [
|
"suffix": "provider",
|
||||||
{
|
"values": [
|
||||||
"label": "Rows",
|
{
|
||||||
"key": "rows",
|
"label": "Rows",
|
||||||
"type": "array"
|
"key": "rows",
|
||||||
},
|
"type": "array"
|
||||||
{
|
},
|
||||||
"label": "Extra Info",
|
{
|
||||||
"key": "info",
|
"label": "Extra Info",
|
||||||
"type": "string"
|
"key": "info",
|
||||||
},
|
"type": "string"
|
||||||
{
|
},
|
||||||
"label": "Rows Length",
|
{
|
||||||
"key": "rowsLength",
|
"label": "Rows Length",
|
||||||
"type": "number"
|
"key": "rowsLength",
|
||||||
},
|
"type": "number"
|
||||||
{
|
},
|
||||||
"label": "Schema",
|
{
|
||||||
"key": "schema",
|
"label": "Schema",
|
||||||
"type": "object"
|
"key": "schema",
|
||||||
},
|
"type": "object"
|
||||||
{
|
},
|
||||||
"label": "Page Number",
|
{
|
||||||
"key": "pageNumber",
|
"label": "Page Number",
|
||||||
"type": "number"
|
"key": "pageNumber",
|
||||||
}
|
"type": "number"
|
||||||
]
|
}
|
||||||
}
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "static",
|
||||||
|
"suffix": "table",
|
||||||
|
"values": [
|
||||||
|
{
|
||||||
|
"label": "Selected Rows",
|
||||||
|
"key": "selectedRows",
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"cardsblock": {
|
"cardsblock": {
|
||||||
"block": true,
|
"block": true,
|
||||||
|
|
|
@ -113,6 +113,9 @@
|
||||||
// List of context keys which we use inside bindings
|
// List of context keys which we use inside bindings
|
||||||
let knownContextKeyMap = {}
|
let knownContextKeyMap = {}
|
||||||
|
|
||||||
|
// Cleanup function to stop observing context changes when unmounting
|
||||||
|
let unobserve
|
||||||
|
|
||||||
// Set up initial state for each new component instance
|
// Set up initial state for each new component instance
|
||||||
$: initialise(instance)
|
$: initialise(instance)
|
||||||
|
|
||||||
|
@ -311,6 +314,11 @@
|
||||||
|
|
||||||
// Force an initial enrichment of the new settings
|
// Force an initial enrichment of the new settings
|
||||||
enrichComponentSettings(get(context), settingsDefinitionMap)
|
enrichComponentSettings(get(context), settingsDefinitionMap)
|
||||||
|
|
||||||
|
// Start observing changes in context now that we are initialised
|
||||||
|
if (!unobserve) {
|
||||||
|
unobserve = context.actions.observeChanges(handleContextChange)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extracts a map of all context keys which are required by action settings
|
// Extracts a map of all context keys which are required by action settings
|
||||||
|
@ -567,8 +575,8 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register an unregister component instance
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
|
// Register this component instance for external access
|
||||||
if ($appStore.isDevApp) {
|
if ($appStore.isDevApp) {
|
||||||
if (!componentStore.actions.isComponentRegistered(id)) {
|
if (!componentStore.actions.isComponentRegistered(id)) {
|
||||||
componentStore.actions.registerInstance(id, {
|
componentStore.actions.registerInstance(id, {
|
||||||
|
@ -581,16 +589,17 @@
|
||||||
state: store,
|
state: store,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return () => {
|
}
|
||||||
if (componentStore.actions.isComponentRegistered(id)) {
|
return () => {
|
||||||
componentStore.actions.unregisterInstance(id)
|
// Unregister component
|
||||||
}
|
if (componentStore.actions.isComponentRegistered(id)) {
|
||||||
|
componentStore.actions.unregisterInstance(id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Stop observing context changes
|
||||||
|
unobserve?.()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// Observe changes to context
|
|
||||||
onMount(() => context.actions.observeChanges(handleContextChange))
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if constructor && initialSettings && (visible || inSelectedPath) && !builderHidden}
|
{#if constructor && initialSettings && (visible || inSelectedPath) && !builderHidden}
|
||||||
|
|
|
@ -69,7 +69,10 @@
|
||||||
}
|
}
|
||||||
const col = linkColumn || "_id"
|
const col = linkColumn || "_id"
|
||||||
const split = url.split("/:")
|
const split = url.split("/:")
|
||||||
return `${split[0]}/{{ ${safe(repeaterId)}.${safe(col)} }}`
|
if (split.length > 1) {
|
||||||
|
return `${split[0]}/{{ ${safe(repeaterId)}.${safe(col)} }}`
|
||||||
|
}
|
||||||
|
return url
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load the datasource schema so we can determine column types
|
// Load the datasource schema so we can determine column types
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
import { Table } from "@budibase/bbui"
|
import { Table } from "@budibase/bbui"
|
||||||
import SlotRenderer from "./SlotRenderer.svelte"
|
import SlotRenderer from "./SlotRenderer.svelte"
|
||||||
import { canBeSortColumn } from "@budibase/shared-core"
|
import { canBeSortColumn } from "@budibase/shared-core"
|
||||||
|
import Provider from "../../context/Provider.svelte"
|
||||||
|
|
||||||
export let dataProvider
|
export let dataProvider
|
||||||
export let columns
|
export let columns
|
||||||
|
@ -55,6 +56,11 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Build our data context
|
||||||
|
$: dataContext = {
|
||||||
|
selectedRows,
|
||||||
|
}
|
||||||
|
|
||||||
const getFields = (
|
const getFields = (
|
||||||
schema,
|
schema,
|
||||||
customColumns,
|
customColumns,
|
||||||
|
@ -156,27 +162,29 @@
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div use:styleable={$component.styles} class={size}>
|
<div use:styleable={$component.styles} class={size}>
|
||||||
<Table
|
<Provider data={dataContext}>
|
||||||
{data}
|
<Table
|
||||||
{schema}
|
{data}
|
||||||
{loading}
|
{schema}
|
||||||
{rowCount}
|
{loading}
|
||||||
{quiet}
|
{rowCount}
|
||||||
{compact}
|
{quiet}
|
||||||
{customRenderers}
|
{compact}
|
||||||
allowSelectRows={allowSelectRows && table}
|
{customRenderers}
|
||||||
bind:selectedRows
|
allowSelectRows={allowSelectRows && table}
|
||||||
allowEditRows={false}
|
bind:selectedRows
|
||||||
allowEditColumns={false}
|
allowEditRows={false}
|
||||||
showAutoColumns={true}
|
allowEditColumns={false}
|
||||||
disableSorting
|
showAutoColumns={true}
|
||||||
autoSortColumns={!columns?.length}
|
disableSorting
|
||||||
on:sort={onSort}
|
autoSortColumns={!columns?.length}
|
||||||
on:click={handleClick}
|
on:sort={onSort}
|
||||||
placeholderText={noRowsMessage || "No rows found"}
|
on:click={handleClick}
|
||||||
>
|
placeholderText={noRowsMessage || "No rows found"}
|
||||||
<slot />
|
>
|
||||||
</Table>
|
<slot />
|
||||||
|
</Table>
|
||||||
|
</Provider>
|
||||||
{#if allowSelectRows && selectedRows.length}
|
{#if allowSelectRows && selectedRows.length}
|
||||||
<div class="row-count">
|
<div class="row-count">
|
||||||
{selectedRows.length} row{selectedRows.length === 1 ? "" : "s"} selected
|
{selectedRows.length} row{selectedRows.length === 1 ? "" : "s"} selected
|
||||||
|
|
|
@ -5,7 +5,7 @@ const createRowSelectionStore = () => {
|
||||||
|
|
||||||
function updateSelection(componentId, tableId, selectedRows) {
|
function updateSelection(componentId, tableId, selectedRows) {
|
||||||
store.update(state => {
|
store.update(state => {
|
||||||
state[componentId] = { tableId: tableId, selectedRows: selectedRows }
|
state[componentId] = { tableId, selectedRows }
|
||||||
return state
|
return state
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -341,7 +341,10 @@ const exportDataHandler = async action => {
|
||||||
format: action.parameters.type,
|
format: action.parameters.type,
|
||||||
columns: action.parameters.columns,
|
columns: action.parameters.columns,
|
||||||
})
|
})
|
||||||
download(data, `${selection.tableId}.${action.parameters.type}`)
|
download(
|
||||||
|
new Blob([data], { type: "text/plain" }),
|
||||||
|
`${selection.tableId}.${action.parameters.type}`
|
||||||
|
)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
notificationStore.actions.error("There was an error exporting the data")
|
notificationStore.actions.error("There was an error exporting the data")
|
||||||
}
|
}
|
||||||
|
|
|
@ -103,6 +103,7 @@
|
||||||
"pouchdb-all-dbs": "1.1.1",
|
"pouchdb-all-dbs": "1.1.1",
|
||||||
"pouchdb-find": "7.2.2",
|
"pouchdb-find": "7.2.2",
|
||||||
"redis": "4",
|
"redis": "4",
|
||||||
|
"serialize-error": "^7.0.1",
|
||||||
"server-destroy": "1.0.1",
|
"server-destroy": "1.0.1",
|
||||||
"snowflake-promise": "^4.5.0",
|
"snowflake-promise": "^4.5.0",
|
||||||
"socket.io": "4.6.1",
|
"socket.io": "4.6.1",
|
||||||
|
|
|
@ -38,7 +38,6 @@ async function init() {
|
||||||
SELF_HOSTED: "1",
|
SELF_HOSTED: "1",
|
||||||
DISABLE_ACCOUNT_PORTAL: "1",
|
DISABLE_ACCOUNT_PORTAL: "1",
|
||||||
MULTI_TENANCY: "",
|
MULTI_TENANCY: "",
|
||||||
DISABLE_THREADING: "1",
|
|
||||||
SERVICE: "app-service",
|
SERVICE: "app-service",
|
||||||
DEPLOYMENT_ENVIRONMENT: "development",
|
DEPLOYMENT_ENVIRONMENT: "development",
|
||||||
BB_ADMIN_USER_EMAIL: "",
|
BB_ADMIN_USER_EMAIL: "",
|
||||||
|
|
|
@ -16,9 +16,9 @@ import {
|
||||||
} from "@budibase/types"
|
} from "@budibase/types"
|
||||||
import sdk from "../sdk"
|
import sdk from "../sdk"
|
||||||
import { automationsEnabled } from "../features"
|
import { automationsEnabled } from "../features"
|
||||||
|
import { helpers, REBOOT_CRON } from "@budibase/shared-core"
|
||||||
import tracer from "dd-trace"
|
import tracer from "dd-trace"
|
||||||
|
|
||||||
const REBOOT_CRON = "@reboot"
|
|
||||||
const WH_STEP_ID = definitions.WEBHOOK.stepId
|
const WH_STEP_ID = definitions.WEBHOOK.stepId
|
||||||
const CRON_STEP_ID = definitions.CRON.stepId
|
const CRON_STEP_ID = definitions.CRON.stepId
|
||||||
let Runner: Thread
|
let Runner: Thread
|
||||||
|
@ -198,6 +198,13 @@ export async function enableCronTrigger(appId: any, automation: Automation) {
|
||||||
!isRebootTrigger(automation) &&
|
!isRebootTrigger(automation) &&
|
||||||
trigger?.inputs.cron
|
trigger?.inputs.cron
|
||||||
) {
|
) {
|
||||||
|
const cronExp = trigger.inputs.cron
|
||||||
|
const validation = helpers.cron.validate(cronExp)
|
||||||
|
if (!validation.valid) {
|
||||||
|
throw new Error(
|
||||||
|
`Invalid automation CRON "${cronExp}" - ${validation.err.join(", ")}`
|
||||||
|
)
|
||||||
|
}
|
||||||
// make a job id rather than letting Bull decide, makes it easier to handle on way out
|
// make a job id rather than letting Bull decide, makes it easier to handle on way out
|
||||||
const jobId = `${appId}_cron_${newid()}`
|
const jobId = `${appId}_cron_${newid()}`
|
||||||
const job: any = await automationQueue.add(
|
const job: any = await automationQueue.add(
|
||||||
|
@ -205,7 +212,7 @@ export async function enableCronTrigger(appId: any, automation: Automation) {
|
||||||
automation,
|
automation,
|
||||||
event: { appId, timestamp: Date.now() },
|
event: { appId, timestamp: Date.now() },
|
||||||
},
|
},
|
||||||
{ repeat: { cron: trigger.inputs.cron }, jobId }
|
{ repeat: { cron: cronExp }, jobId }
|
||||||
)
|
)
|
||||||
// Assign cron job ID from bull so we can remove it later if the cron trigger is removed
|
// Assign cron job ID from bull so we can remove it later if the cron trigger is removed
|
||||||
trigger.cronJobId = job.id
|
trigger.cronJobId = job.id
|
||||||
|
|
|
@ -97,11 +97,7 @@ const environment = {
|
||||||
APP_MIGRATION_TIMEOUT: parseIntSafe(process.env.APP_MIGRATION_TIMEOUT),
|
APP_MIGRATION_TIMEOUT: parseIntSafe(process.env.APP_MIGRATION_TIMEOUT),
|
||||||
JS_RUNNER_MEMORY_LIMIT:
|
JS_RUNNER_MEMORY_LIMIT:
|
||||||
parseIntSafe(process.env.JS_RUNNER_MEMORY_LIMIT) || 64,
|
parseIntSafe(process.env.JS_RUNNER_MEMORY_LIMIT) || 64,
|
||||||
}
|
LOG_JS_ERRORS: process.env.LOG_JS_ERRORS,
|
||||||
|
|
||||||
// threading can cause memory issues with node-ts in development
|
|
||||||
if (coreEnv.isDev() && environment.DISABLE_THREADING == null) {
|
|
||||||
environment._set("DISABLE_THREADING", "1")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean up any environment variable edge cases
|
// clean up any environment variable edge cases
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import vm from "vm"
|
import vm from "vm"
|
||||||
import env from "../environment"
|
import env from "../environment"
|
||||||
import { setJSRunner } from "@budibase/string-templates"
|
import { setJSRunner, setOnErrorLog } from "@budibase/string-templates"
|
||||||
import { context, timers } from "@budibase/backend-core"
|
import { context, logging, timers } from "@budibase/backend-core"
|
||||||
import tracer from "dd-trace"
|
import tracer from "dd-trace"
|
||||||
|
import { serializeError } from "serialize-error"
|
||||||
|
|
||||||
type TrackerFn = <T>(f: () => T) => T
|
type TrackerFn = <T>(f: () => T) => T
|
||||||
|
|
||||||
|
@ -58,4 +59,10 @@ export function init() {
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if (env.LOG_JS_ERRORS) {
|
||||||
|
setOnErrorLog((error: Error) => {
|
||||||
|
logging.logWarn(JSON.stringify(serializeError(error)))
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ import { QueryVariable } from "./definitions"
|
||||||
import env from "../environment"
|
import env from "../environment"
|
||||||
import * as db from "../db"
|
import * as db from "../db"
|
||||||
import { redis, db as dbCore } from "@budibase/backend-core"
|
import { redis, db as dbCore } from "@budibase/backend-core"
|
||||||
|
import * as jsRunner from "../jsRunner"
|
||||||
|
|
||||||
const VARIABLE_TTL_SECONDS = 3600
|
const VARIABLE_TTL_SECONDS = 3600
|
||||||
let client: any
|
let client: any
|
||||||
|
@ -29,7 +30,9 @@ export function threadSetup() {
|
||||||
console.debug(`[${env.FORKED_PROCESS_NAME}] thread setup skipped`)
|
console.debug(`[${env.FORKED_PROCESS_NAME}] thread setup skipped`)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
console.debug(`[${env.FORKED_PROCESS_NAME}] thread setup running`)
|
console.debug(`[${env.FORKED_PROCESS_NAME}] thread setup running`)
|
||||||
|
jsRunner.init()
|
||||||
db.init()
|
db.init()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,8 @@
|
||||||
"check:types": "tsc -p tsconfig.json --noEmit --paths null"
|
"check:types": "tsc -p tsconfig.json --noEmit --paths null"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@budibase/types": "0.0.0"
|
"@budibase/types": "0.0.0",
|
||||||
|
"cron-validate": "1.4.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"rimraf": "3.0.2",
|
"rimraf": "3.0.2",
|
||||||
|
|
|
@ -99,6 +99,8 @@ export const SocketSessionTTL = 60
|
||||||
export const ValidQueryNameRegex = /^[^()]*$/
|
export const ValidQueryNameRegex = /^[^()]*$/
|
||||||
export const ValidColumnNameRegex = /^[_a-zA-Z0-9\s]*$/g
|
export const ValidColumnNameRegex = /^[_a-zA-Z0-9\s]*$/g
|
||||||
|
|
||||||
|
export const REBOOT_CRON = "@reboot"
|
||||||
|
|
||||||
export const InvalidFileExtensions = [
|
export const InvalidFileExtensions = [
|
||||||
"7z",
|
"7z",
|
||||||
"action",
|
"action",
|
||||||
|
|
|
@ -0,0 +1,47 @@
|
||||||
|
import cronValidate from "cron-validate"
|
||||||
|
|
||||||
|
const INPUT_CRON_START = "(Input cron: "
|
||||||
|
const ERROR_SWAPS = {
|
||||||
|
"smaller than lower limit": "less than",
|
||||||
|
"bigger than upper limit": "greater than",
|
||||||
|
daysOfMonth: "'days of the month'",
|
||||||
|
daysOfWeek: "'days of the week'",
|
||||||
|
years: "'years'",
|
||||||
|
months: "'months'",
|
||||||
|
hours: "'hours'",
|
||||||
|
minutes: "'minutes'",
|
||||||
|
seconds: "'seconds'",
|
||||||
|
}
|
||||||
|
|
||||||
|
function improveErrors(errors: string[]): string[] {
|
||||||
|
const finalErrors: string[] = []
|
||||||
|
|
||||||
|
for (let error of errors) {
|
||||||
|
if (error.includes(INPUT_CRON_START)) {
|
||||||
|
error = error.split(INPUT_CRON_START)[0].trim()
|
||||||
|
}
|
||||||
|
for (let [oldErr, newErr] of Object.entries(ERROR_SWAPS)) {
|
||||||
|
if (error.includes(oldErr)) {
|
||||||
|
error = error.replace(new RegExp(oldErr, "g"), newErr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finalErrors.push(error)
|
||||||
|
}
|
||||||
|
return finalErrors
|
||||||
|
}
|
||||||
|
|
||||||
|
export function validate(
|
||||||
|
cronExpression: string
|
||||||
|
): { valid: false; err: string[] } | { valid: true } {
|
||||||
|
const result = cronValidate(cronExpression, {
|
||||||
|
preset: "npm-node-cron",
|
||||||
|
override: {
|
||||||
|
useSeconds: false,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
if (!result.isValid()) {
|
||||||
|
return { valid: false, err: improveErrors(result.getError()) }
|
||||||
|
} else {
|
||||||
|
return { valid: true }
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,2 +1,3 @@
|
||||||
export * from "./helpers"
|
export * from "./helpers"
|
||||||
export * from "./integrations"
|
export * from "./integrations"
|
||||||
|
export * as cron from "./cron"
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
import { expect, describe, it } from "vitest"
|
||||||
|
import { cron } from "../helpers"
|
||||||
|
|
||||||
|
describe("check valid and invalid crons", () => {
|
||||||
|
it("invalid - 0 0 0 11 *", () => {
|
||||||
|
expect(cron.validate("0 0 0 11 *")).toStrictEqual({
|
||||||
|
valid: false,
|
||||||
|
err: [expect.stringContaining("less than '1'")],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("invalid - 5 4 32 1 1", () => {
|
||||||
|
expect(cron.validate("5 4 32 1 1")).toStrictEqual({
|
||||||
|
valid: false,
|
||||||
|
err: [expect.stringContaining("greater than '31'")],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it("valid - * * * * *", () => {
|
||||||
|
expect(cron.validate("* * * * *")).toStrictEqual({ valid: true })
|
||||||
|
})
|
||||||
|
})
|
|
@ -8,6 +8,9 @@ const { getJsHelperList } = require("./list")
|
||||||
let runJS
|
let runJS
|
||||||
module.exports.setJSRunner = runner => (runJS = runner)
|
module.exports.setJSRunner = runner => (runJS = runner)
|
||||||
|
|
||||||
|
let onErrorLog
|
||||||
|
module.exports.setOnErrorLog = delegate => (onErrorLog = delegate)
|
||||||
|
|
||||||
// Helper utility to strip square brackets from a value
|
// Helper utility to strip square brackets from a value
|
||||||
const removeSquareBrackets = value => {
|
const removeSquareBrackets = value => {
|
||||||
if (!value || typeof value !== "string") {
|
if (!value || typeof value !== "string") {
|
||||||
|
@ -56,6 +59,8 @@ module.exports.processJS = (handlebars, context) => {
|
||||||
const res = { data: runJS(js, sandboxContext) }
|
const res = { data: runJS(js, sandboxContext) }
|
||||||
return `{{${LITERAL_MARKER} js_result-${JSON.stringify(res)}}}`
|
return `{{${LITERAL_MARKER} js_result-${JSON.stringify(res)}}}`
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
onErrorLog && onErrorLog(error)
|
||||||
|
|
||||||
if (error.code === "ERR_SCRIPT_EXECUTION_TIMEOUT") {
|
if (error.code === "ERR_SCRIPT_EXECUTION_TIMEOUT") {
|
||||||
return "Timed out while executing JS"
|
return "Timed out while executing JS"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
const vm = require("vm")
|
||||||
const handlebars = require("handlebars")
|
const handlebars = require("handlebars")
|
||||||
const { registerAll, registerMinimum } = require("./helpers/index")
|
const { registerAll, registerMinimum } = require("./helpers/index")
|
||||||
const processors = require("./processors")
|
const processors = require("./processors")
|
||||||
|
@ -365,6 +366,7 @@ module.exports.doesContainString = (template, string) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports.setJSRunner = javascript.setJSRunner
|
module.exports.setJSRunner = javascript.setJSRunner
|
||||||
|
module.exports.setOnErrorLog = javascript.setOnErrorLog
|
||||||
|
|
||||||
module.exports.convertToJS = hbs => {
|
module.exports.convertToJS = hbs => {
|
||||||
const blocks = exports.findHBSBlocks(hbs)
|
const blocks = exports.findHBSBlocks(hbs)
|
||||||
|
@ -401,3 +403,19 @@ const errors = require("./errors")
|
||||||
module.exports.JsErrorTimeout = errors.JsErrorTimeout
|
module.exports.JsErrorTimeout = errors.JsErrorTimeout
|
||||||
|
|
||||||
module.exports.helpersToRemoveForJs = helpersToRemoveForJs
|
module.exports.helpersToRemoveForJs = helpersToRemoveForJs
|
||||||
|
|
||||||
|
if (process && !process.env.NO_JS) {
|
||||||
|
/**
|
||||||
|
* Use polyfilled vm to run JS scripts in a browser Env
|
||||||
|
*/
|
||||||
|
javascript.setJSRunner((js, context) => {
|
||||||
|
context = {
|
||||||
|
...context,
|
||||||
|
alert: undefined,
|
||||||
|
setInterval: undefined,
|
||||||
|
setTimeout: undefined,
|
||||||
|
}
|
||||||
|
vm.createContext(context)
|
||||||
|
return vm.runInNewContext(js, context, { timeout: 1000 })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
import vm from "vm"
|
|
||||||
import templates from "./index.js"
|
import templates from "./index.js"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -20,23 +19,8 @@ export const disableEscaping = templates.disableEscaping
|
||||||
export const findHBSBlocks = templates.findHBSBlocks
|
export const findHBSBlocks = templates.findHBSBlocks
|
||||||
export const convertToJS = templates.convertToJS
|
export const convertToJS = templates.convertToJS
|
||||||
export const setJSRunner = templates.setJSRunner
|
export const setJSRunner = templates.setJSRunner
|
||||||
|
export const setOnErrorLog = templates.setOnErrorLog
|
||||||
export const FIND_ANY_HBS_REGEX = templates.FIND_ANY_HBS_REGEX
|
export const FIND_ANY_HBS_REGEX = templates.FIND_ANY_HBS_REGEX
|
||||||
export const helpersToRemoveForJs = templates.helpersToRemoveForJs
|
export const helpersToRemoveForJs = templates.helpersToRemoveForJs
|
||||||
|
|
||||||
if (process && !process.env.NO_JS) {
|
|
||||||
/**
|
|
||||||
* Use polyfilled vm to run JS scripts in a browser Env
|
|
||||||
*/
|
|
||||||
setJSRunner((js, context) => {
|
|
||||||
context = {
|
|
||||||
...context,
|
|
||||||
alert: undefined,
|
|
||||||
setInterval: undefined,
|
|
||||||
setTimeout: undefined,
|
|
||||||
}
|
|
||||||
vm.createContext(context)
|
|
||||||
return vm.runInNewContext(js, context, { timeout: 1000 })
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
export * from "./errors.js"
|
export * from "./errors.js"
|
||||||
|
|
36
yarn.lock
36
yarn.lock
|
@ -1980,6 +1980,13 @@
|
||||||
resolved "https://registry.yarnpkg.com/@babel/regjsgen/-/regjsgen-0.8.0.tgz#f0ba69b075e1f05fb2825b7fad991e7adbb18310"
|
resolved "https://registry.yarnpkg.com/@babel/regjsgen/-/regjsgen-0.8.0.tgz#f0ba69b075e1f05fb2825b7fad991e7adbb18310"
|
||||||
integrity sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==
|
integrity sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==
|
||||||
|
|
||||||
|
"@babel/runtime@^7.10.5":
|
||||||
|
version "7.23.9"
|
||||||
|
resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.23.9.tgz#47791a15e4603bb5f905bc0753801cf21d6345f7"
|
||||||
|
integrity sha512-0CX6F+BI2s9dkUqr08KFrAIZgNFj75rdBU/DjCyYLIaV/quFjkk6T+EJ2LkZHyZTbEV4L5p97mNkUsHl2wLFAw==
|
||||||
|
dependencies:
|
||||||
|
regenerator-runtime "^0.14.0"
|
||||||
|
|
||||||
"@babel/runtime@^7.12.5", "@babel/runtime@^7.13.10", "@babel/runtime@^7.15.4", "@babel/runtime@^7.21.0", "@babel/runtime@^7.8.4", "@babel/runtime@^7.9.2":
|
"@babel/runtime@^7.12.5", "@babel/runtime@^7.13.10", "@babel/runtime@^7.15.4", "@babel/runtime@^7.21.0", "@babel/runtime@^7.8.4", "@babel/runtime@^7.9.2":
|
||||||
version "7.23.8"
|
version "7.23.8"
|
||||||
resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.23.8.tgz#8ee6fe1ac47add7122902f257b8ddf55c898f650"
|
resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.23.8.tgz#8ee6fe1ac47add7122902f257b8ddf55c898f650"
|
||||||
|
@ -5465,6 +5472,11 @@
|
||||||
resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.200.tgz#435b6035c7eba9cdf1e039af8212c9e9281e7149"
|
resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.200.tgz#435b6035c7eba9cdf1e039af8212c9e9281e7149"
|
||||||
integrity sha512-YI/M/4HRImtNf3pJgbF+W6FrXovqj+T+/HpENLTooK9PnkacBsDpeP3IpHab40CClUfhNmdM2WTNP2sa2dni5Q==
|
integrity sha512-YI/M/4HRImtNf3pJgbF+W6FrXovqj+T+/HpENLTooK9PnkacBsDpeP3IpHab40CClUfhNmdM2WTNP2sa2dni5Q==
|
||||||
|
|
||||||
|
"@types/lodash@^4.14.165":
|
||||||
|
version "4.14.202"
|
||||||
|
resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.202.tgz#f09dbd2fb082d507178b2f2a5c7e74bd72ff98f8"
|
||||||
|
integrity sha512-OvlIYQK9tNneDlS0VN54LLd5uiPCBOp7gS5Z0f1mjoJYBrtStzgmJBxONW3U6OZqdtNzZPmn9BS/7WI7BFFcFQ==
|
||||||
|
|
||||||
"@types/long@^4.0.0":
|
"@types/long@^4.0.0":
|
||||||
version "4.0.2"
|
version "4.0.2"
|
||||||
resolved "https://registry.yarnpkg.com/@types/long/-/long-4.0.2.tgz#b74129719fc8d11c01868010082d483b7545591a"
|
resolved "https://registry.yarnpkg.com/@types/long/-/long-4.0.2.tgz#b74129719fc8d11c01868010082d483b7545591a"
|
||||||
|
@ -8622,6 +8634,13 @@ cron-parser@^4.2.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
luxon "^3.2.1"
|
luxon "^3.2.1"
|
||||||
|
|
||||||
|
cron-validate@1.4.5:
|
||||||
|
version "1.4.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/cron-validate/-/cron-validate-1.4.5.tgz#eceb221f7558e6302e5f84c7b3a454fdf4d064c3"
|
||||||
|
integrity sha512-nKlOJEnYKudMn/aNyNH8xxWczlfpaazfWV32Pcx/2St51r2bxWbGhZD7uwzMcRhunA/ZNL+Htm/i0792Z59UMQ==
|
||||||
|
dependencies:
|
||||||
|
yup "0.32.9"
|
||||||
|
|
||||||
cross-spawn@^6.0.0:
|
cross-spawn@^6.0.0:
|
||||||
version "6.0.5"
|
version "6.0.5"
|
||||||
resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4"
|
resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4"
|
||||||
|
@ -14650,7 +14669,7 @@ lock@^1.1.0:
|
||||||
resolved "https://registry.yarnpkg.com/lock/-/lock-1.1.0.tgz#53157499d1653b136ca66451071fca615703fa55"
|
resolved "https://registry.yarnpkg.com/lock/-/lock-1.1.0.tgz#53157499d1653b136ca66451071fca615703fa55"
|
||||||
integrity sha512-NZQIJJL5Rb9lMJ0Yl1JoVr9GSdo4HTPsUEWsSFzB8dE8DSoiLCVavWZPi7Rnlv/o73u6I24S/XYc/NmG4l8EKA==
|
integrity sha512-NZQIJJL5Rb9lMJ0Yl1JoVr9GSdo4HTPsUEWsSFzB8dE8DSoiLCVavWZPi7Rnlv/o73u6I24S/XYc/NmG4l8EKA==
|
||||||
|
|
||||||
lodash-es@^4.17.21:
|
lodash-es@^4.17.15, lodash-es@^4.17.21:
|
||||||
version "4.17.21"
|
version "4.17.21"
|
||||||
resolved "https://registry.yarnpkg.com/lodash-es/-/lodash-es-4.17.21.tgz#43e626c46e6591b7750beb2b50117390c609e3ee"
|
resolved "https://registry.yarnpkg.com/lodash-es/-/lodash-es-4.17.21.tgz#43e626c46e6591b7750beb2b50117390c609e3ee"
|
||||||
integrity sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==
|
integrity sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==
|
||||||
|
@ -14800,7 +14819,7 @@ lodash.xor@^4.5.0:
|
||||||
resolved "https://registry.yarnpkg.com/lodash.xor/-/lodash.xor-4.5.0.tgz#4d48ed7e98095b0632582ba714d3ff8ae8fb1db6"
|
resolved "https://registry.yarnpkg.com/lodash.xor/-/lodash.xor-4.5.0.tgz#4d48ed7e98095b0632582ba714d3ff8ae8fb1db6"
|
||||||
integrity sha512-sVN2zimthq7aZ5sPGXnSz32rZPuqcparVW50chJQe+mzTYV+IsxSsl/2gnkWWE2Of7K3myBQBqtLKOUEHJKRsQ==
|
integrity sha512-sVN2zimthq7aZ5sPGXnSz32rZPuqcparVW50chJQe+mzTYV+IsxSsl/2gnkWWE2Of7K3myBQBqtLKOUEHJKRsQ==
|
||||||
|
|
||||||
lodash@4.17.21, lodash@^4.17.11, lodash@^4.17.14, lodash@^4.17.15, lodash@^4.17.19, lodash@^4.17.21, lodash@^4.17.3, lodash@^4.7.0:
|
lodash@4.17.21, lodash@^4.17.11, lodash@^4.17.14, lodash@^4.17.15, lodash@^4.17.19, lodash@^4.17.20, lodash@^4.17.21, lodash@^4.17.3, lodash@^4.7.0:
|
||||||
version "4.17.21"
|
version "4.17.21"
|
||||||
resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
|
resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
|
||||||
integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
|
integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
|
||||||
|
@ -22312,6 +22331,19 @@ yocto-queue@^1.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-1.0.0.tgz#7f816433fb2cbc511ec8bf7d263c3b58a1a3c251"
|
resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-1.0.0.tgz#7f816433fb2cbc511ec8bf7d263c3b58a1a3c251"
|
||||||
integrity sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==
|
integrity sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==
|
||||||
|
|
||||||
|
yup@0.32.9:
|
||||||
|
version "0.32.9"
|
||||||
|
resolved "https://registry.yarnpkg.com/yup/-/yup-0.32.9.tgz#9367bec6b1b0e39211ecbca598702e106019d872"
|
||||||
|
integrity sha512-Ci1qN+i2H0XpY7syDQ0k5zKQ/DoxO0LzPg8PAR/X4Mpj6DqaeCoIYEEjDJwhArh3Fa7GWbQQVDZKeXYlSH4JMg==
|
||||||
|
dependencies:
|
||||||
|
"@babel/runtime" "^7.10.5"
|
||||||
|
"@types/lodash" "^4.14.165"
|
||||||
|
lodash "^4.17.20"
|
||||||
|
lodash-es "^4.17.15"
|
||||||
|
nanoclone "^0.2.1"
|
||||||
|
property-expr "^2.0.4"
|
||||||
|
toposort "^2.0.2"
|
||||||
|
|
||||||
yup@^0.32.11:
|
yup@^0.32.11:
|
||||||
version "0.32.11"
|
version "0.32.11"
|
||||||
resolved "https://registry.yarnpkg.com/yup/-/yup-0.32.11.tgz#d67fb83eefa4698607982e63f7ca4c5ed3cf18c5"
|
resolved "https://registry.yarnpkg.com/yup/-/yup-0.32.11.tgz#d67fb83eefa4698607982e63f7ca4c5ed3cf18c5"
|
||||||
|
|
Loading…
Reference in New Issue