Merge branch 'master' of github.com:Budibase/budibase into chore/api-typing
This commit is contained in:
commit
467bdcf74c
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"$schema": "node_modules/lerna/schemas/lerna-schema.json",
|
"$schema": "node_modules/lerna/schemas/lerna-schema.json",
|
||||||
"version": "3.2.18",
|
"version": "3.2.19",
|
||||||
"npmClient": "yarn",
|
"npmClient": "yarn",
|
||||||
"concurrency": 20,
|
"concurrency": 20,
|
||||||
"command": {
|
"command": {
|
||||||
|
|
|
@ -96,6 +96,10 @@ if (env.SELF_HOSTED) {
|
||||||
ACTION_IMPLS["EXECUTE_BASH"] = bash.run
|
ACTION_IMPLS["EXECUTE_BASH"] = bash.run
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
BUILTIN_ACTION_DEFINITIONS["EXECUTE_BASH"] = bash.definition
|
BUILTIN_ACTION_DEFINITIONS["EXECUTE_BASH"] = bash.definition
|
||||||
|
|
||||||
|
if (env.isTest()) {
|
||||||
|
BUILTIN_ACTION_DEFINITIONS["OPENAI"] = openai.definition
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getActionDefinitions(): Promise<
|
export async function getActionDefinitions(): Promise<
|
||||||
|
|
|
@ -1,26 +1,148 @@
|
||||||
import { getConfig, afterAll as _afterAll, runStep } from "./utilities"
|
import { createAutomationBuilder } from "./utilities/AutomationTestBuilder"
|
||||||
|
import * as automation from "../index"
|
||||||
|
import * as setup from "./utilities"
|
||||||
|
import { Table } from "@budibase/types"
|
||||||
|
|
||||||
describe("test the bash action", () => {
|
describe("Execute Bash Automations", () => {
|
||||||
let config = getConfig()
|
let config = setup.getConfig(),
|
||||||
|
table: Table
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
|
await automation.init()
|
||||||
await config.init()
|
await config.init()
|
||||||
})
|
table = await config.createTable()
|
||||||
afterAll(_afterAll)
|
await config.createRow({
|
||||||
|
name: "test row",
|
||||||
it("should be able to execute a script", async () => {
|
description: "test description",
|
||||||
let res = await runStep(config, "EXECUTE_BASH", {
|
tableId: table._id!,
|
||||||
code: "echo 'test'",
|
|
||||||
})
|
})
|
||||||
expect(res.stdout).toEqual("test\n")
|
|
||||||
expect(res.success).toEqual(true)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should handle a null value", async () => {
|
afterAll(setup.afterAll)
|
||||||
let res = await runStep(config, "EXECUTE_BASH", {
|
|
||||||
code: null,
|
it("should use trigger data in bash command and pass output to subsequent steps", async () => {
|
||||||
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Bash with Trigger Data",
|
||||||
|
config,
|
||||||
})
|
})
|
||||||
expect(res.stdout).toEqual(
|
.appAction({ fields: { command: "hello world" } })
|
||||||
|
.bash(
|
||||||
|
{ code: "echo '{{ trigger.fields.command }}'" },
|
||||||
|
{ stepName: "Echo Command" }
|
||||||
|
)
|
||||||
|
.serverLog(
|
||||||
|
{ text: "Bash output was: {{ steps.[Echo Command].stdout }}" },
|
||||||
|
{ stepName: "Log Output" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.stdout).toEqual("hello world\n")
|
||||||
|
expect(result.steps[1].outputs.message).toContain(
|
||||||
|
"Bash output was: hello world"
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it("should chain multiple bash commands using previous outputs", async () => {
|
||||||
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Chained Bash Commands",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: { filename: "testfile.txt" } })
|
||||||
|
.bash(
|
||||||
|
{ code: "echo 'initial content' > {{ trigger.fields.filename }}" },
|
||||||
|
{ stepName: "Create File" }
|
||||||
|
)
|
||||||
|
.bash(
|
||||||
|
{ code: "cat {{ trigger.fields.filename }} | tr '[a-z]' '[A-Z]'" },
|
||||||
|
{ stepName: "Transform Content" }
|
||||||
|
)
|
||||||
|
.bash(
|
||||||
|
{ code: "rm {{ trigger.fields.filename }}" },
|
||||||
|
{ stepName: "Cleanup" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[1].outputs.stdout).toEqual("INITIAL CONTENT\n")
|
||||||
|
expect(result.steps[1].outputs.success).toEqual(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it("should integrate bash output with row operations", async () => {
|
||||||
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Bash with Row Operations",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.queryRows(
|
||||||
|
{
|
||||||
|
tableId: table._id!,
|
||||||
|
filters: {},
|
||||||
|
},
|
||||||
|
{ stepName: "Get Row" }
|
||||||
|
)
|
||||||
|
.bash(
|
||||||
|
{
|
||||||
|
code: "echo Row data: {{ steps.[Get Row].rows.[0].name }} - {{ steps.[Get Row].rows.[0].description }}",
|
||||||
|
},
|
||||||
|
{ stepName: "Process Row Data" }
|
||||||
|
)
|
||||||
|
.serverLog(
|
||||||
|
{ text: "{{ steps.[Process Row Data].stdout }}" },
|
||||||
|
{ stepName: "Log Result" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[1].outputs.stdout).toContain(
|
||||||
|
"Row data: test row - test description"
|
||||||
|
)
|
||||||
|
expect(result.steps[2].outputs.message).toContain(
|
||||||
|
"Row data: test row - test description"
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it("should handle bash output in conditional logic", async () => {
|
||||||
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Bash with Conditional",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: { threshold: "5" } })
|
||||||
|
.bash(
|
||||||
|
{ code: "echo $(( {{ trigger.fields.threshold }} + 5 ))" },
|
||||||
|
{ stepName: "Calculate Value" }
|
||||||
|
)
|
||||||
|
.executeScript(
|
||||||
|
{
|
||||||
|
code: `
|
||||||
|
const value = parseInt(steps["Calculate Value"].stdout);
|
||||||
|
return value > 8 ? "high" : "low";
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{ stepName: "Check Value" }
|
||||||
|
)
|
||||||
|
.serverLog(
|
||||||
|
{ text: "Value was {{ steps.[Check Value].value }}" },
|
||||||
|
{ stepName: "Log Result" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.stdout).toEqual("10\n")
|
||||||
|
expect(result.steps[1].outputs.value).toEqual("high")
|
||||||
|
expect(result.steps[2].outputs.message).toContain("Value was high")
|
||||||
|
})
|
||||||
|
|
||||||
|
it("should handle null values gracefully", async () => {
|
||||||
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Null Bash Input",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.bash(
|
||||||
|
//@ts-ignore
|
||||||
|
{ code: null },
|
||||||
|
{ stepName: "Null Command" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.stdout).toBe(
|
||||||
"Budibase bash automation failed: Invalid inputs"
|
"Budibase bash automation failed: Invalid inputs"
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import { getConfig, runStep, afterAll as _afterAll } from "./utilities"
|
import { getConfig, afterAll as _afterAll } from "./utilities"
|
||||||
|
import { createAutomationBuilder } from "./utilities/AutomationTestBuilder"
|
||||||
import { OpenAI } from "openai"
|
import { OpenAI } from "openai"
|
||||||
import { setEnv as setCoreEnv } from "@budibase/backend-core"
|
import { setEnv as setCoreEnv } from "@budibase/backend-core"
|
||||||
import * as pro from "@budibase/pro"
|
import * as pro from "@budibase/pro"
|
||||||
|
import { Model } from "@budibase/types"
|
||||||
|
|
||||||
jest.mock("openai", () => ({
|
jest.mock("openai", () => ({
|
||||||
OpenAI: jest.fn().mockImplementation(() => ({
|
OpenAI: jest.fn().mockImplementation(() => ({
|
||||||
|
@ -47,6 +49,7 @@ describe("test the openai action", () => {
|
||||||
let resetEnv: () => void | undefined
|
let resetEnv: () => void | undefined
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
|
setCoreEnv({ SELF_HOSTED: true })
|
||||||
await config.init()
|
await config.init()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -62,17 +65,39 @@ describe("test the openai action", () => {
|
||||||
afterAll(_afterAll)
|
afterAll(_afterAll)
|
||||||
|
|
||||||
it("should be able to receive a response from ChatGPT given a prompt", async () => {
|
it("should be able to receive a response from ChatGPT given a prompt", async () => {
|
||||||
const res = await runStep(config, "OPENAI", { prompt: OPENAI_PROMPT })
|
setCoreEnv({ SELF_HOSTED: true })
|
||||||
expect(res.response).toEqual("This is a test")
|
|
||||||
expect(res.success).toBeTruthy()
|
const result = await createAutomationBuilder({
|
||||||
|
name: "Test OpenAI Response",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.openai(
|
||||||
|
{ prompt: OPENAI_PROMPT, model: Model.GPT_4O_MINI },
|
||||||
|
{ stepName: "Basic OpenAI Query" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.response).toEqual("This is a test")
|
||||||
|
expect(result.steps[0].outputs.success).toBeTruthy()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should present the correct error message when a prompt is not provided", async () => {
|
it("should present the correct error message when a prompt is not provided", async () => {
|
||||||
const res = await runStep(config, "OPENAI", { prompt: null })
|
const result = await createAutomationBuilder({
|
||||||
expect(res.response).toEqual(
|
name: "Test OpenAI No Prompt",
|
||||||
|
config,
|
||||||
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.openai(
|
||||||
|
{ prompt: "", model: Model.GPT_4O_MINI },
|
||||||
|
{ stepName: "Empty Prompt Query" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.response).toEqual(
|
||||||
"Budibase OpenAI Automation Failed: No prompt supplied"
|
"Budibase OpenAI Automation Failed: No prompt supplied"
|
||||||
)
|
)
|
||||||
expect(res.success).toBeFalsy()
|
expect(result.steps[0].outputs.success).toBeFalsy()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should present the correct error message when an error is thrown from the createChatCompletion call", async () => {
|
it("should present the correct error message when an error is thrown from the createChatCompletion call", async () => {
|
||||||
|
@ -91,14 +116,21 @@ describe("test the openai action", () => {
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
const res = await runStep(config, "OPENAI", {
|
const result = await createAutomationBuilder({
|
||||||
prompt: OPENAI_PROMPT,
|
name: "Test OpenAI Error",
|
||||||
|
config,
|
||||||
})
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.openai(
|
||||||
|
{ prompt: OPENAI_PROMPT, model: Model.GPT_4O_MINI },
|
||||||
|
{ stepName: "Error Producing Query" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
expect(res.response).toEqual(
|
expect(result.steps[0].outputs.response).toEqual(
|
||||||
"Error: An error occurred while calling createChatCompletion"
|
"Error: An error occurred while calling createChatCompletion"
|
||||||
)
|
)
|
||||||
expect(res.success).toBeFalsy()
|
expect(result.steps[0].outputs.success).toBeFalsy()
|
||||||
})
|
})
|
||||||
|
|
||||||
it("should ensure that the pro AI module is called when the budibase AI features are enabled", async () => {
|
it("should ensure that the pro AI module is called when the budibase AI features are enabled", async () => {
|
||||||
|
@ -106,10 +138,19 @@ describe("test the openai action", () => {
|
||||||
jest.spyOn(pro.features, "isAICustomConfigsEnabled").mockResolvedValue(true)
|
jest.spyOn(pro.features, "isAICustomConfigsEnabled").mockResolvedValue(true)
|
||||||
|
|
||||||
const prompt = "What is the meaning of life?"
|
const prompt = "What is the meaning of life?"
|
||||||
await runStep(config, "OPENAI", {
|
await createAutomationBuilder({
|
||||||
model: "gpt-4o-mini",
|
name: "Test OpenAI Pro Features",
|
||||||
prompt,
|
config,
|
||||||
})
|
})
|
||||||
|
.appAction({ fields: {} })
|
||||||
|
.openai(
|
||||||
|
{
|
||||||
|
model: Model.GPT_4O_MINI,
|
||||||
|
prompt,
|
||||||
|
},
|
||||||
|
{ stepName: "Pro Features Query" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
expect(pro.ai.LargeLanguageModel.forCurrentTenant).toHaveBeenCalledWith(
|
expect(pro.ai.LargeLanguageModel.forCurrentTenant).toHaveBeenCalledWith(
|
||||||
"gpt-4o-mini"
|
"gpt-4o-mini"
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
import { Table } from "@budibase/types"
|
import { EmptyFilterOption, SortOrder, Table } from "@budibase/types"
|
||||||
import * as setup from "./utilities"
|
import * as setup from "./utilities"
|
||||||
|
import { createAutomationBuilder } from "./utilities/AutomationTestBuilder"
|
||||||
|
import * as automation from "../index"
|
||||||
|
|
||||||
const NAME = "Test"
|
const NAME = "Test"
|
||||||
|
|
||||||
|
@ -8,6 +10,7 @@ describe("Test a query step automation", () => {
|
||||||
let config = setup.getConfig()
|
let config = setup.getConfig()
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
|
await automation.init()
|
||||||
await config.init()
|
await config.init()
|
||||||
table = await config.createTable()
|
table = await config.createTable()
|
||||||
const row = {
|
const row = {
|
||||||
|
@ -22,107 +25,132 @@ describe("Test a query step automation", () => {
|
||||||
afterAll(setup.afterAll)
|
afterAll(setup.afterAll)
|
||||||
|
|
||||||
it("should be able to run the query step", async () => {
|
it("should be able to run the query step", async () => {
|
||||||
const inputs = {
|
const result = await createAutomationBuilder({
|
||||||
tableId: table._id,
|
name: "Basic Query Test",
|
||||||
filters: {
|
|
||||||
equal: {
|
|
||||||
name: NAME,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
sortColumn: "name",
|
|
||||||
sortOrder: "ascending",
|
|
||||||
limit: 10,
|
|
||||||
}
|
|
||||||
const res = await setup.runStep(
|
|
||||||
config,
|
config,
|
||||||
setup.actions.QUERY_ROWS.stepId,
|
})
|
||||||
inputs
|
.appAction({ fields: {} })
|
||||||
)
|
.queryRows(
|
||||||
expect(res.success).toBe(true)
|
{
|
||||||
expect(res.rows).toBeDefined()
|
tableId: table._id!,
|
||||||
expect(res.rows.length).toBe(2)
|
filters: {
|
||||||
expect(res.rows[0].name).toBe(NAME)
|
equal: {
|
||||||
|
name: NAME,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
sortColumn: "name",
|
||||||
|
sortOrder: SortOrder.ASCENDING,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
{ stepName: "Query All Rows" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.success).toBe(true)
|
||||||
|
expect(result.steps[0].outputs.rows).toBeDefined()
|
||||||
|
expect(result.steps[0].outputs.rows.length).toBe(2)
|
||||||
|
expect(result.steps[0].outputs.rows[0].name).toBe(NAME)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("Returns all rows when onEmptyFilter has no value and no filters are passed", async () => {
|
it("Returns all rows when onEmptyFilter has no value and no filters are passed", async () => {
|
||||||
const inputs = {
|
const result = await createAutomationBuilder({
|
||||||
tableId: table._id,
|
name: "Empty Filter Test",
|
||||||
filters: {},
|
|
||||||
sortColumn: "name",
|
|
||||||
sortOrder: "ascending",
|
|
||||||
limit: 10,
|
|
||||||
}
|
|
||||||
const res = await setup.runStep(
|
|
||||||
config,
|
config,
|
||||||
setup.actions.QUERY_ROWS.stepId,
|
})
|
||||||
inputs
|
.appAction({ fields: {} })
|
||||||
)
|
.queryRows(
|
||||||
expect(res.success).toBe(true)
|
{
|
||||||
expect(res.rows).toBeDefined()
|
tableId: table._id!,
|
||||||
expect(res.rows.length).toBe(2)
|
filters: {},
|
||||||
expect(res.rows[0].name).toBe(NAME)
|
sortColumn: "name",
|
||||||
|
sortOrder: SortOrder.ASCENDING,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
{ stepName: "Query With Empty Filter" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.success).toBe(true)
|
||||||
|
expect(result.steps[0].outputs.rows).toBeDefined()
|
||||||
|
expect(result.steps[0].outputs.rows.length).toBe(2)
|
||||||
|
expect(result.steps[0].outputs.rows[0].name).toBe(NAME)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("Returns no rows when onEmptyFilter is RETURN_NONE and theres no filters", async () => {
|
it("Returns no rows when onEmptyFilter is RETURN_NONE and theres no filters", async () => {
|
||||||
const inputs = {
|
const result = await createAutomationBuilder({
|
||||||
tableId: table._id,
|
name: "Return None Test",
|
||||||
filters: {},
|
|
||||||
"filters-def": [],
|
|
||||||
sortColumn: "name",
|
|
||||||
sortOrder: "ascending",
|
|
||||||
limit: 10,
|
|
||||||
onEmptyFilter: "none",
|
|
||||||
}
|
|
||||||
const res = await setup.runStep(
|
|
||||||
config,
|
config,
|
||||||
setup.actions.QUERY_ROWS.stepId,
|
})
|
||||||
inputs
|
.appAction({ fields: {} })
|
||||||
)
|
.queryRows(
|
||||||
expect(res.success).toBe(false)
|
{
|
||||||
expect(res.rows).toBeDefined()
|
tableId: table._id!,
|
||||||
expect(res.rows.length).toBe(0)
|
filters: {},
|
||||||
|
"filters-def": [],
|
||||||
|
sortColumn: "name",
|
||||||
|
sortOrder: SortOrder.ASCENDING,
|
||||||
|
limit: 10,
|
||||||
|
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
|
||||||
|
},
|
||||||
|
{ stepName: "Query With Return None" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.success).toBe(false)
|
||||||
|
expect(result.steps[0].outputs.rows).toBeDefined()
|
||||||
|
expect(result.steps[0].outputs.rows.length).toBe(0)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("Returns no rows when onEmptyFilters RETURN_NONE and a filter is passed with a null value", async () => {
|
it("Returns no rows when onEmptyFilters RETURN_NONE and a filter is passed with a null value", async () => {
|
||||||
const inputs = {
|
const result = await createAutomationBuilder({
|
||||||
tableId: table._id,
|
name: "Null Filter Test",
|
||||||
onEmptyFilter: "none",
|
|
||||||
filters: {},
|
|
||||||
"filters-def": [
|
|
||||||
{
|
|
||||||
value: null,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
sortColumn: "name",
|
|
||||||
sortOrder: "ascending",
|
|
||||||
limit: 10,
|
|
||||||
}
|
|
||||||
const res = await setup.runStep(
|
|
||||||
config,
|
config,
|
||||||
setup.actions.QUERY_ROWS.stepId,
|
})
|
||||||
inputs
|
.appAction({ fields: {} })
|
||||||
)
|
.queryRows(
|
||||||
expect(res.success).toBe(false)
|
{
|
||||||
expect(res.rows).toBeDefined()
|
tableId: table._id!,
|
||||||
expect(res.rows.length).toBe(0)
|
onEmptyFilter: EmptyFilterOption.RETURN_NONE,
|
||||||
|
filters: {},
|
||||||
|
"filters-def": [
|
||||||
|
{
|
||||||
|
value: null,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
sortColumn: "name",
|
||||||
|
sortOrder: SortOrder.ASCENDING,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
{ stepName: "Query With Null Filter" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.success).toBe(false)
|
||||||
|
expect(result.steps[0].outputs.rows).toBeDefined()
|
||||||
|
expect(result.steps[0].outputs.rows.length).toBe(0)
|
||||||
})
|
})
|
||||||
|
|
||||||
it("Returns rows when onEmptyFilter is RETURN_ALL and no filter is passed", async () => {
|
it("Returns rows when onEmptyFilter is RETURN_ALL and no filter is passed", async () => {
|
||||||
const inputs = {
|
const result = await createAutomationBuilder({
|
||||||
tableId: table._id,
|
name: "Return All Test",
|
||||||
onEmptyFilter: "all",
|
|
||||||
filters: {},
|
|
||||||
sortColumn: "name",
|
|
||||||
sortOrder: "ascending",
|
|
||||||
limit: 10,
|
|
||||||
}
|
|
||||||
const res = await setup.runStep(
|
|
||||||
config,
|
config,
|
||||||
setup.actions.QUERY_ROWS.stepId,
|
})
|
||||||
inputs
|
.appAction({ fields: {} })
|
||||||
)
|
.queryRows(
|
||||||
expect(res.success).toBe(true)
|
{
|
||||||
expect(res.rows).toBeDefined()
|
tableId: table._id!,
|
||||||
expect(res.rows.length).toBe(2)
|
onEmptyFilter: EmptyFilterOption.RETURN_ALL,
|
||||||
|
filters: {},
|
||||||
|
sortColumn: "name",
|
||||||
|
sortOrder: SortOrder.ASCENDING,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
{ stepName: "Query With Return All" }
|
||||||
|
)
|
||||||
|
.run()
|
||||||
|
|
||||||
|
expect(result.steps[0].outputs.success).toBe(true)
|
||||||
|
expect(result.steps[0].outputs.rows).toBeDefined()
|
||||||
|
expect(result.steps[0].outputs.rows.length).toBe(2)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -35,6 +35,8 @@ import {
|
||||||
Branch,
|
Branch,
|
||||||
FilterStepInputs,
|
FilterStepInputs,
|
||||||
ExecuteScriptStepInputs,
|
ExecuteScriptStepInputs,
|
||||||
|
OpenAIStepInputs,
|
||||||
|
BashStepInputs,
|
||||||
} from "@budibase/types"
|
} from "@budibase/types"
|
||||||
import TestConfiguration from "../../../tests/utilities/TestConfiguration"
|
import TestConfiguration from "../../../tests/utilities/TestConfiguration"
|
||||||
import * as setup from "../utilities"
|
import * as setup from "../utilities"
|
||||||
|
@ -221,6 +223,30 @@ class BaseStepBuilder {
|
||||||
input
|
input
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bash(
|
||||||
|
input: BashStepInputs,
|
||||||
|
opts?: { stepName?: string; stepId?: string }
|
||||||
|
): this {
|
||||||
|
return this.step(
|
||||||
|
AutomationActionStepId.EXECUTE_BASH,
|
||||||
|
BUILTIN_ACTION_DEFINITIONS.EXECUTE_BASH,
|
||||||
|
input,
|
||||||
|
opts
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
openai(
|
||||||
|
input: OpenAIStepInputs,
|
||||||
|
opts?: { stepName?: string; stepId?: string }
|
||||||
|
): this {
|
||||||
|
return this.step(
|
||||||
|
AutomationActionStepId.OPENAI,
|
||||||
|
BUILTIN_ACTION_DEFINITIONS.OPENAI,
|
||||||
|
input,
|
||||||
|
opts
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
class StepBuilder extends BaseStepBuilder {
|
class StepBuilder extends BaseStepBuilder {
|
||||||
build(): AutomationStep[] {
|
build(): AutomationStep[] {
|
||||||
|
|
|
@ -150,7 +150,7 @@ export type OpenAIStepInputs = {
|
||||||
prompt: string
|
prompt: string
|
||||||
model: Model
|
model: Model
|
||||||
}
|
}
|
||||||
enum Model {
|
export enum Model {
|
||||||
GPT_35_TURBO = "gpt-3.5-turbo",
|
GPT_35_TURBO = "gpt-3.5-turbo",
|
||||||
// will only work with api keys that have access to the GPT4 API
|
// will only work with api keys that have access to the GPT4 API
|
||||||
GPT_4 = "gpt-4",
|
GPT_4 = "gpt-4",
|
||||||
|
|
Loading…
Reference in New Issue