Add tests for executeScriptV2
This commit is contained in:
parent
94f4b8c43e
commit
910296e25c
|
@ -83,7 +83,7 @@ export async function run({
|
|||
try {
|
||||
return {
|
||||
success: true,
|
||||
value: processStringSync(inputs.code, context),
|
||||
value: processStringSync(inputs.code, context, { noThrow: false }),
|
||||
}
|
||||
} catch (err) {
|
||||
return {
|
||||
|
|
|
@ -0,0 +1,165 @@
|
|||
import { createAutomationBuilder } from "./utilities/AutomationTestBuilder"
|
||||
import * as automation from "../index"
|
||||
import * as setup from "./utilities"
|
||||
import { Table } from "@budibase/types"
|
||||
|
||||
function encodeJS(js: string): string {
|
||||
return `{{ js "${Buffer.from(js, "utf-8").toString("base64")}" }}`
|
||||
}
|
||||
|
||||
describe("Execute Script Automations", () => {
|
||||
let config = setup.getConfig(),
|
||||
table: Table
|
||||
|
||||
beforeEach(async () => {
|
||||
await automation.init()
|
||||
await config.init()
|
||||
table = await config.createTable()
|
||||
await config.createRow()
|
||||
})
|
||||
|
||||
afterAll(setup.afterAll)
|
||||
|
||||
it("should execute a basic script and return the result", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Basic Script Execution",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: {} })
|
||||
.executeScriptV2({ code: encodeJS("return 2 + 2") })
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.value).toEqual(4)
|
||||
})
|
||||
|
||||
it("should access bindings from previous steps", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Access Bindings",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: { data: [1, 2, 3] } })
|
||||
.executeScriptV2(
|
||||
{
|
||||
code: encodeJS(`return $("trigger.fields.data").map(x => x * 2)`),
|
||||
},
|
||||
{ stepId: "binding-script-step" }
|
||||
)
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.value).toEqual([2, 4, 6])
|
||||
})
|
||||
|
||||
it("should handle script execution errors gracefully", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Handle Script Errors",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: {} })
|
||||
.executeScriptV2({
|
||||
code: encodeJS("return nonexistentVariable.map(x => x)"),
|
||||
})
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.response).toContain(
|
||||
"ReferenceError: nonexistentVariable is not defined"
|
||||
)
|
||||
expect(results.steps[0].outputs.success).toEqual(false)
|
||||
})
|
||||
|
||||
it("should handle conditional logic in scripts", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Conditional Script Logic",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: { value: 10 } })
|
||||
.executeScriptV2({
|
||||
code: encodeJS(`
|
||||
if ($("trigger.fields.value") > 5) {
|
||||
return "Value is greater than 5";
|
||||
} else {
|
||||
return "Value is 5 or less";
|
||||
}
|
||||
`),
|
||||
})
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.value).toEqual("Value is greater than 5")
|
||||
})
|
||||
|
||||
it("should use multiple steps and validate script execution", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Multi-Step Script Execution",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: {} })
|
||||
.serverLog(
|
||||
{ text: "Starting multi-step automation" },
|
||||
{ stepId: "start-log-step" }
|
||||
)
|
||||
.createRow(
|
||||
{ row: { name: "Test Row", value: 42, tableId: table._id } },
|
||||
{ stepId: "abc123" }
|
||||
)
|
||||
.executeScriptV2(
|
||||
{
|
||||
code: encodeJS(`
|
||||
const createdRow = $("steps")['abc123'];
|
||||
return createdRow.row.value * 2;
|
||||
`),
|
||||
},
|
||||
{ stepId: "ScriptingStep1" }
|
||||
)
|
||||
.serverLog({
|
||||
text: `Final result is {{ steps.ScriptingStep1.value }}`,
|
||||
})
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.message).toContain(
|
||||
"Starting multi-step automation"
|
||||
)
|
||||
expect(results.steps[1].outputs.row.value).toEqual(42)
|
||||
expect(results.steps[2].outputs.value).toEqual(84)
|
||||
expect(results.steps[3].outputs.message).toContain("Final result is 84")
|
||||
})
|
||||
|
||||
it("should fail if the code has not been encoded as a handlebars template", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Invalid Code Encoding",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: {} })
|
||||
.executeScriptV2({
|
||||
code: "return 2 + 2",
|
||||
})
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.response.message).toEqual(
|
||||
"Expected code to be a {{ js }} template block"
|
||||
)
|
||||
expect(results.steps[0].outputs.success).toEqual(false)
|
||||
})
|
||||
|
||||
it("does not process embedded handlebars templates", async () => {
|
||||
const builder = createAutomationBuilder({
|
||||
name: "Embedded Handlebars",
|
||||
})
|
||||
|
||||
const results = await builder
|
||||
.appAction({ fields: {} })
|
||||
.executeScriptV2({
|
||||
code: encodeJS(`return "{{ triggers.row.whatever }}"`),
|
||||
})
|
||||
.run()
|
||||
|
||||
expect(results.steps[0].outputs.value).toEqual(
|
||||
"{{ triggers.row.whatever }}"
|
||||
)
|
||||
expect(results.steps[0].outputs.success).toEqual(true)
|
||||
})
|
||||
})
|
|
@ -633,10 +633,16 @@ class Orchestrator {
|
|||
}
|
||||
|
||||
const stepFn = await this.getStepFunctionality(step.stepId)
|
||||
let inputs = await processObject(
|
||||
let inputs = originalStepInput
|
||||
if (step.stepId !== AutomationActionStepId.EXECUTE_SCRIPT_V2) {
|
||||
// The EXECUTE_SCRIPT_V2 step saves its input.code value as a `{{ js
|
||||
// "..." }}` template, and expects to receive it that way in the
|
||||
// function that runs it. So we skip this next bit for that step.
|
||||
inputs = await processObject(
|
||||
originalStepInput,
|
||||
this.mergeContexts(this.context)
|
||||
)
|
||||
}
|
||||
inputs = automationUtils.cleanInputValues(inputs, step.schema.inputs)
|
||||
|
||||
const outputs = await stepFn({
|
||||
|
|
Loading…
Reference in New Issue