Merge branch 'v3-ui' of github.com:Budibase/budibase into view-calculation-ui

This commit is contained in:
Andrew Kingston 2024-10-23 14:18:33 +01:00
commit 8f80d8eb86
No known key found for this signature in database
12 changed files with 67 additions and 367 deletions

View File

@ -9,7 +9,6 @@
import { sdk } from "@budibase/shared-core" import { sdk } from "@budibase/shared-core"
import ConfirmDialog from "components/common/ConfirmDialog.svelte" import ConfirmDialog from "components/common/ConfirmDialog.svelte"
import UpdateAutomationModal from "components/automation/AutomationPanel/UpdateAutomationModal.svelte" import UpdateAutomationModal from "components/automation/AutomationPanel/UpdateAutomationModal.svelte"
import UpdateRowActionModal from "components/automation/AutomationPanel/UpdateRowActionModal.svelte"
import NavItem from "components/common/NavItem.svelte" import NavItem from "components/common/NavItem.svelte"
export let automation export let automation
@ -17,7 +16,6 @@
let confirmDeleteDialog let confirmDeleteDialog
let updateAutomationDialog let updateAutomationDialog
let updateRowActionDialog
$: isRowAction = sdk.automations.isRowAction(automation) $: isRowAction = sdk.automations.isRowAction(automation)
@ -92,7 +90,7 @@
name: "Edit", name: "Edit",
keyBind: null, keyBind: null,
visible: true, visible: true,
callback: updateRowActionDialog.show, callback: updateAutomationDialog.show,
}, },
del, del,
] ]
@ -135,8 +133,4 @@
This action cannot be undone. This action cannot be undone.
</ConfirmDialog> </ConfirmDialog>
{#if isRowAction}
<UpdateRowActionModal {automation} bind:this={updateRowActionDialog} />
{:else}
<UpdateAutomationModal {automation} bind:this={updateAutomationDialog} /> <UpdateAutomationModal {automation} bind:this={updateAutomationDialog} />
{/if}

View File

@ -1,83 +0,0 @@
<script>
import { rowActions } from "stores/builder"
import {
notifications,
Icon,
Input,
ModalContent,
Modal,
} from "@budibase/bbui"
export let automation
export let onCancel = undefined
let name
let error = ""
let modal
export const show = () => {
name = automation?.displayName
modal.show()
}
export const hide = () => {
modal.hide()
}
async function saveAutomation() {
try {
await rowActions.rename(
automation.definition.trigger.inputs.tableId,
automation.definition.trigger.inputs.rowActionId,
name
)
notifications.success(`Row action updated successfully`)
hide()
} catch (error) {
notifications.error("Error saving row action")
}
}
function checkValid(evt) {
name = evt.target.value
if (!name) {
error = "Name is required"
return
}
error = ""
}
</script>
<Modal bind:this={modal} on:hide={onCancel}>
<ModalContent
title="Edit Row Action"
confirmText="Save"
size="L"
onConfirm={saveAutomation}
disabled={error}
>
<Input bind:value={name} label="Name" on:input={checkValid} {error} />
<a
slot="footer"
target="_blank"
href="https://docs.budibase.com/docs/automation-steps"
>
<Icon name="InfoOutline" />
<span>Learn about automations</span>
</a>
</ModalContent>
</Modal>
<style>
a {
color: var(--ink);
font-size: 14px;
vertical-align: middle;
display: flex;
align-items: center;
text-decoration: none;
}
a span {
text-decoration: underline;
margin-left: var(--spectrum-alias-item-padding-s);
}
</style>

View File

@ -2,9 +2,9 @@ import {
CreateRowActionRequest, CreateRowActionRequest,
Ctx, Ctx,
RowActionPermissions, RowActionPermissions,
RowActionPermissionsResponse,
RowActionResponse, RowActionResponse,
RowActionsResponse, RowActionsResponse,
UpdateRowActionRequest,
} from "@budibase/types" } from "@budibase/types"
import sdk from "../../../sdk" import sdk from "../../../sdk"
@ -30,6 +30,7 @@ export async function find(ctx: Ctx<void, RowActionsResponse>) {
} }
const { actions } = rowActions const { actions } = rowActions
const automationNames = await sdk.rowActions.getNames(rowActions)
const result: RowActionsResponse = { const result: RowActionsResponse = {
actions: Object.entries(actions).reduce<Record<string, RowActionResponse>>( actions: Object.entries(actions).reduce<Record<string, RowActionResponse>>(
(acc, [key, action]) => ({ (acc, [key, action]) => ({
@ -37,7 +38,7 @@ export async function find(ctx: Ctx<void, RowActionsResponse>) {
[key]: { [key]: {
id: key, id: key,
tableId, tableId,
name: action.name, name: automationNames[action.automationId],
automationId: action.automationId, automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions), allowedSources: flattenAllowedSources(tableId, action.permissions),
}, },
@ -68,26 +69,6 @@ export async function create(
ctx.status = 201 ctx.status = 201
} }
export async function update(
ctx: Ctx<UpdateRowActionRequest, RowActionResponse>
) {
const table = await getTable(ctx)
const tableId = table._id!
const { actionId } = ctx.params
const action = await sdk.rowActions.update(tableId, actionId, {
name: ctx.request.body.name,
})
ctx.body = {
tableId,
id: action.id,
name: action.name,
automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions),
}
}
export async function remove(ctx: Ctx<void, void>) { export async function remove(ctx: Ctx<void, void>) {
const table = await getTable(ctx) const table = await getTable(ctx)
const { actionId } = ctx.params const { actionId } = ctx.params
@ -96,22 +77,22 @@ export async function remove(ctx: Ctx<void, void>) {
ctx.status = 204 ctx.status = 204
} }
export async function setTablePermission(ctx: Ctx<void, RowActionResponse>) { export async function setTablePermission(
ctx: Ctx<void, RowActionPermissionsResponse>
) {
const table = await getTable(ctx) const table = await getTable(ctx)
const tableId = table._id! const tableId = table._id!
const { actionId } = ctx.params const { actionId } = ctx.params
const action = await sdk.rowActions.setTablePermission(tableId, actionId) const action = await sdk.rowActions.setTablePermission(tableId, actionId)
ctx.body = { ctx.body = {
tableId,
id: action.id,
name: action.name,
automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions), allowedSources: flattenAllowedSources(tableId, action.permissions),
} }
} }
export async function unsetTablePermission(ctx: Ctx<void, RowActionResponse>) { export async function unsetTablePermission(
ctx: Ctx<void, RowActionPermissionsResponse>
) {
const table = await getTable(ctx) const table = await getTable(ctx)
const tableId = table._id! const tableId = table._id!
const { actionId } = ctx.params const { actionId } = ctx.params
@ -119,15 +100,13 @@ export async function unsetTablePermission(ctx: Ctx<void, RowActionResponse>) {
const action = await sdk.rowActions.unsetTablePermission(tableId, actionId) const action = await sdk.rowActions.unsetTablePermission(tableId, actionId)
ctx.body = { ctx.body = {
tableId,
id: action.id,
name: action.name,
automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions), allowedSources: flattenAllowedSources(tableId, action.permissions),
} }
} }
export async function setViewPermission(ctx: Ctx<void, RowActionResponse>) { export async function setViewPermission(
ctx: Ctx<void, RowActionPermissionsResponse>
) {
const table = await getTable(ctx) const table = await getTable(ctx)
const tableId = table._id! const tableId = table._id!
const { actionId, viewId } = ctx.params const { actionId, viewId } = ctx.params
@ -138,15 +117,13 @@ export async function setViewPermission(ctx: Ctx<void, RowActionResponse>) {
viewId viewId
) )
ctx.body = { ctx.body = {
tableId,
id: action.id,
name: action.name,
automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions), allowedSources: flattenAllowedSources(tableId, action.permissions),
} }
} }
export async function unsetViewPermission(ctx: Ctx<void, RowActionResponse>) { export async function unsetViewPermission(
ctx: Ctx<void, RowActionPermissionsResponse>
) {
const table = await getTable(ctx) const table = await getTable(ctx)
const tableId = table._id! const tableId = table._id!
const { actionId, viewId } = ctx.params const { actionId, viewId } = ctx.params
@ -158,10 +135,6 @@ export async function unsetViewPermission(ctx: Ctx<void, RowActionResponse>) {
) )
ctx.body = { ctx.body = {
tableId,
id: action.id,
name: action.name,
automationId: action.automationId,
allowedSources: flattenAllowedSources(tableId, action.permissions), allowedSources: flattenAllowedSources(tableId, action.permissions),
} }
} }

View File

@ -40,12 +40,6 @@ router
rowActionValidator(), rowActionValidator(),
rowActionController.create rowActionController.create
) )
.put(
"/api/tables/:tableId/actions/:actionId",
authorized(BUILDER),
rowActionValidator(),
rowActionController.update
)
.delete( .delete(
"/api/tables/:tableId/actions/:actionId", "/api/tables/:tableId/actions/:actionId",
authorized(BUILDER), authorized(BUILDER),

View File

@ -328,129 +328,6 @@ describe("/rowsActions", () => {
}) })
}) })
describe("update", () => {
unauthorisedTests((expectations, testConfig) =>
config.api.rowAction.update(
tableId,
generator.guid(),
createRowActionRequest(),
expectations,
testConfig
)
)
it("can update existing actions", async () => {
for (const rowAction of createRowActionRequests(3)) {
await createRowAction(tableId, rowAction)
}
const persisted = await config.api.rowAction.find(tableId)
const [actionId, actionData] = _.sample(
Object.entries(persisted.actions)
)!
const updatedName = generator.string()
const res = await config.api.rowAction.update(tableId, actionId, {
name: updatedName,
})
expect(res).toEqual({
id: actionId,
tableId,
name: updatedName,
automationId: actionData.automationId,
allowedSources: [tableId],
})
expect(await config.api.rowAction.find(tableId)).toEqual(
expect.objectContaining({
actions: expect.objectContaining({
[actionId]: {
name: updatedName,
id: actionData.id,
tableId: actionData.tableId,
automationId: actionData.automationId,
allowedSources: [tableId],
},
}),
})
)
})
it("trims row action names", async () => {
const rowAction = await createRowAction(tableId, createRowActionRequest())
const res = await config.api.rowAction.update(tableId, rowAction.id, {
name: " action name ",
})
expect(res).toEqual(expect.objectContaining({ name: "action name" }))
expect(await config.api.rowAction.find(tableId)).toEqual(
expect.objectContaining({
actions: expect.objectContaining({
[rowAction.id]: expect.objectContaining({
name: "action name",
}),
}),
})
)
})
it("throws Bad Request when trying to update by a non-existing id", async () => {
await createRowAction(tableId, createRowActionRequest())
await config.api.rowAction.update(
tableId,
generator.guid(),
createRowActionRequest(),
{ status: 400 }
)
})
it("throws Bad Request when trying to update by a via another table id", async () => {
const otherTable = await config.api.table.save(
setup.structures.basicTable()
)
await createRowAction(otherTable._id!, createRowActionRequest())
const action = await createRowAction(tableId, createRowActionRequest())
await config.api.rowAction.update(
otherTable._id!,
action.id,
createRowActionRequest(),
{ status: 400 }
)
})
it("can not use existing row action names (for the same table)", async () => {
const action1 = await createRowAction(tableId, createRowActionRequest())
const action2 = await createRowAction(tableId, createRowActionRequest())
await config.api.rowAction.update(
tableId,
action1.id,
{ name: action2.name },
{
status: 409,
body: {
message: "A row action with the same name already exists.",
},
}
)
})
it("does not throw with name conflicts for the same row action", async () => {
const action1 = await createRowAction(tableId, createRowActionRequest())
await config.api.rowAction.update(tableId, action1.id, {
name: action1.name,
})
})
})
describe("delete", () => { describe("delete", () => {
unauthorisedTests((expectations, testConfig) => unauthorisedTests((expectations, testConfig) =>
config.api.rowAction.delete( config.api.rowAction.delete(

View File

@ -1,4 +1,3 @@
import { sdk } from "@budibase/shared-core"
import { import {
Automation, Automation,
RequiredKeys, RequiredKeys,
@ -99,6 +98,12 @@ export async function get(automationId: string) {
return trimUnexpectedObjectFields(result) return trimUnexpectedObjectFields(result)
} }
export async function find(ids: string[]) {
const db = getDb()
const result = await db.getMultiple<PersistedAutomation>(ids)
return result.map(trimUnexpectedObjectFields)
}
export async function create(automation: Automation) { export async function create(automation: Automation) {
automation = trimUnexpectedObjectFields(automation) automation = trimUnexpectedObjectFields(automation)
const db = getDb() const db = getDb()
@ -289,13 +294,6 @@ function guardInvalidUpdatesAndThrow(
} }
}) })
} }
if (
sdk.automations.isRowAction(automation) &&
automation.name !== oldAutomation.name
) {
throw new Error("Row actions cannot be renamed")
}
} }
function trimUnexpectedObjectFields<T extends Automation>(automation: T): T { function trimUnexpectedObjectFields<T extends Automation>(automation: T): T {

View File

@ -1,5 +1,5 @@
import { sample } from "lodash/fp" import { sample } from "lodash/fp"
import { Automation, AutomationTriggerStepId } from "@budibase/types" import { Automation } from "@budibase/types"
import { generator } from "@budibase/backend-core/tests" import { generator } from "@budibase/backend-core/tests"
import TestConfiguration from "../../../../tests/utilities/TestConfiguration" import TestConfiguration from "../../../../tests/utilities/TestConfiguration"
import automationSdk from "../" import automationSdk from "../"
@ -26,25 +26,6 @@ describe("automation sdk", () => {
}) })
}) })
it("cannot rename row action automations", async () => {
await config.doInContext(config.getAppId(), async () => {
const automation = structures.newAutomation({
trigger: {
...structures.automationTrigger(),
stepId: AutomationTriggerStepId.ROW_ACTION,
},
})
const response = await automationSdk.create(automation)
const newName = generator.guid()
const update = { ...response, name: newName }
await expect(automationSdk.update(update)).rejects.toThrow(
"Row actions cannot be renamed"
)
})
})
it.each([ it.each([
["trigger", (a: Automation) => a.definition.trigger], ["trigger", (a: Automation) => a.definition.trigger],
["step", (a: Automation) => a.definition.steps[0]], ["step", (a: Automation) => a.definition.steps[0]],

View File

@ -2,7 +2,6 @@ import {
Automation, Automation,
AutomationActionStepId, AutomationActionStepId,
AutomationBuilderData, AutomationBuilderData,
TableRowActions,
} from "@budibase/types" } from "@budibase/types"
import { sdk as coreSdk } from "@budibase/shared-core" import { sdk as coreSdk } from "@budibase/shared-core"
import sdk from "../../../sdk" import sdk from "../../../sdk"
@ -26,15 +25,6 @@ export async function getBuilderData(
return tableNameCache[tableId] return tableNameCache[tableId]
} }
const rowActionNameCache: Record<string, TableRowActions | undefined> = {}
async function getRowActionName(tableId: string, rowActionId: string) {
if (!rowActionNameCache[tableId]) {
rowActionNameCache[tableId] = await sdk.rowActions.getAll(tableId)
}
return rowActionNameCache[tableId]?.actions[rowActionId]?.name
}
const result: Record<string, AutomationBuilderData> = {} const result: Record<string, AutomationBuilderData> = {}
for (const automation of automations) { for (const automation of automations) {
const isRowAction = coreSdk.automations.isRowAction(automation) const isRowAction = coreSdk.automations.isRowAction(automation)
@ -49,12 +39,7 @@ export async function getBuilderData(
} }
const tableName = await getTableName(tableId) const tableName = await getTableName(tableId)
const rowActionName = await getRowActionName(tableId, rowActionId) const rowActionName = automation.name
if (!rowActionName) {
throw new Error(`Row action not found: ${rowActionId}`)
}
result[automation._id!] = { result[automation._id!] = {
displayName: rowActionName, displayName: rowActionName,
triggerInfo: { triggerInfo: {

View File

@ -13,16 +13,19 @@ import { definitions as TRIGGER_DEFINITIONS } from "../../automations/triggerInf
import * as triggers from "../../automations/triggers" import * as triggers from "../../automations/triggers"
import sdk from ".." import sdk from ".."
function ensureUniqueAndThrow( async function ensureUniqueAndThrow(
doc: TableRowActions, doc: TableRowActions,
name: string, name: string,
existingRowActionId?: string existingRowActionId?: string
) { ) {
const names = await getNames(doc)
name = name.toLowerCase().trim()
if ( if (
Object.entries(doc.actions).find( Object.entries(names).find(
([id, a]) => ([automationId, automationName]) =>
a.name.toLowerCase() === name.toLowerCase() && automationName.toLowerCase().trim() === name &&
id !== existingRowActionId automationId !== existingRowActionId
) )
) { ) {
throw new HTTPError("A row action with the same name already exists.", 409) throw new HTTPError("A row action with the same name already exists.", 409)
@ -34,18 +37,12 @@ export async function create(tableId: string, rowAction: { name: string }) {
const db = context.getAppDB() const db = context.getAppDB()
const rowActionsId = generateRowActionsID(tableId) const rowActionsId = generateRowActionsID(tableId)
let doc: TableRowActions let doc = await db.tryGet<TableRowActions>(rowActionsId)
try { if (!doc) {
doc = await db.get<TableRowActions>(rowActionsId)
} catch (e: any) {
if (e.status !== 404) {
throw e
}
doc = { _id: rowActionsId, actions: {} } doc = { _id: rowActionsId, actions: {} }
} }
ensureUniqueAndThrow(doc, action.name) await ensureUniqueAndThrow(doc, action.name)
const appId = context.getAppId() const appId = context.getAppId()
if (!appId) { if (!appId) {
@ -74,7 +71,6 @@ export async function create(tableId: string, rowAction: { name: string }) {
}) })
doc.actions[newRowActionId] = { doc.actions[newRowActionId] = {
name: action.name,
automationId: automation._id!, automationId: automation._id!,
permissions: { permissions: {
table: { runAllowed: true }, table: { runAllowed: true },
@ -85,6 +81,7 @@ export async function create(tableId: string, rowAction: { name: string }) {
return { return {
id: newRowActionId, id: newRowActionId,
name: automation.name,
...doc.actions[newRowActionId], ...doc.actions[newRowActionId],
} }
} }
@ -159,20 +156,6 @@ async function updateDoc(
} }
} }
export async function update(
tableId: string,
rowActionId: string,
rowActionData: { name: string }
) {
const newName = rowActionData.name.trim()
return await updateDoc(tableId, rowActionId, actionsDoc => {
ensureUniqueAndThrow(actionsDoc, newName, rowActionId)
actionsDoc.actions[rowActionId].name = newName
return actionsDoc
})
}
async function guardView(tableId: string, viewId: string) { async function guardView(tableId: string, viewId: string) {
let view let view
if (docIds.isViewId(viewId)) { if (docIds.isViewId(viewId)) {
@ -248,13 +231,8 @@ export async function run(
throw new HTTPError("Table not found", 404) throw new HTTPError("Table not found", 404)
} }
const rowActions = await getAll(tableId) const { automationId } = await get(tableId, rowActionId)
const rowAction = rowActions?.actions[rowActionId] const automation = await sdk.automations.get(automationId)
if (!rowAction) {
throw new HTTPError("Row action not found", 404)
}
const automation = await sdk.automations.get(rowAction.automationId)
const row = await sdk.rows.find(tableId, rowId) const row = await sdk.rows.find(tableId, rowId)
await triggers.externalTrigger( await triggers.externalTrigger(
@ -272,3 +250,17 @@ export async function run(
{ getResponses: true } { getResponses: true }
) )
} }
export async function getNames({ actions }: TableRowActions) {
const automations = await sdk.automations.find(
Object.values(actions).map(({ automationId }) => automationId)
)
const automationNames = automations.reduce<Record<string, string>>(
(names, a) => {
names[a._id] = a.name
return names
},
{}
)
return automationNames
}

View File

@ -1,5 +1,6 @@
import { import {
CreateRowActionRequest, CreateRowActionRequest,
RowActionPermissionsResponse,
RowActionResponse, RowActionResponse,
RowActionsResponse, RowActionsResponse,
RowActionTriggerRequest, RowActionTriggerRequest,
@ -40,23 +41,6 @@ export class RowActionAPI extends TestAPI {
) )
} }
update = async (
tableId: string,
rowActionId: string,
rowAction: CreateRowActionRequest,
expectations?: Expectations,
config?: { publicUser?: boolean }
) => {
return await this._put<RowActionResponse>(
`/api/tables/${tableId}/actions/${rowActionId}`,
{
body: rowAction,
expectations,
...config,
}
)
}
delete = async ( delete = async (
tableId: string, tableId: string,
rowActionId: string, rowActionId: string,
@ -78,7 +62,7 @@ export class RowActionAPI extends TestAPI {
expectations?: Expectations, expectations?: Expectations,
config?: { publicUser?: boolean } config?: { publicUser?: boolean }
) => { ) => {
return await this._post<RowActionResponse>( return await this._post<RowActionPermissionsResponse>(
`/api/tables/${tableId}/actions/${rowActionId}/permissions`, `/api/tables/${tableId}/actions/${rowActionId}/permissions`,
{ {
expectations: { expectations: {
@ -96,7 +80,7 @@ export class RowActionAPI extends TestAPI {
expectations?: Expectations, expectations?: Expectations,
config?: { publicUser?: boolean } config?: { publicUser?: boolean }
) => { ) => {
return await this._delete<RowActionResponse>( return await this._delete<RowActionPermissionsResponse>(
`/api/tables/${tableId}/actions/${rowActionId}/permissions`, `/api/tables/${tableId}/actions/${rowActionId}/permissions`,
{ {
expectations: { expectations: {
@ -115,7 +99,7 @@ export class RowActionAPI extends TestAPI {
expectations?: Expectations, expectations?: Expectations,
config?: { publicUser?: boolean } config?: { publicUser?: boolean }
) => { ) => {
return await this._post<RowActionResponse>( return await this._post<RowActionPermissionsResponse>(
`/api/tables/${tableId}/actions/${rowActionId}/permissions/${viewId}`, `/api/tables/${tableId}/actions/${rowActionId}/permissions/${viewId}`,
{ {
expectations: { expectations: {
@ -134,7 +118,7 @@ export class RowActionAPI extends TestAPI {
expectations?: Expectations, expectations?: Expectations,
config?: { publicUser?: boolean } config?: { publicUser?: boolean }
) => { ) => {
return await this._delete<RowActionResponse>( return await this._delete<RowActionPermissionsResponse>(
`/api/tables/${tableId}/actions/${rowActionId}/permissions/${viewId}`, `/api/tables/${tableId}/actions/${rowActionId}/permissions/${viewId}`,
{ {
expectations: { expectations: {

View File

@ -1,14 +1,17 @@
interface RowActionData { interface RowActionData {
name: string name: string
} }
interface RowActionPermissionsData {
allowedSources: string[] | undefined
}
export interface CreateRowActionRequest extends RowActionData {} export interface CreateRowActionRequest extends RowActionData {}
export interface UpdateRowActionRequest extends RowActionData {}
export interface RowActionResponse extends RowActionData { export interface RowActionResponse
extends RowActionData,
RowActionPermissionsData {
id: string id: string
tableId: string tableId: string
automationId: string automationId: string
allowedSources: string[] | undefined
} }
export interface RowActionsResponse { export interface RowActionsResponse {
@ -18,3 +21,6 @@ export interface RowActionsResponse {
export interface RowActionTriggerRequest { export interface RowActionTriggerRequest {
rowId: string rowId: string
} }
export interface RowActionPermissionsResponse
extends RowActionPermissionsData {}

View File

@ -6,7 +6,6 @@ export interface TableRowActions extends Document {
} }
export interface RowActionData { export interface RowActionData {
name: string
automationId: string automationId: string
permissions: RowActionPermissions permissions: RowActionPermissions
} }