Merge pull request #2387 from automatisch/create-flow-from-template
feat: Implement initial version of create flow from template logic
This commit is contained in:
@@ -1,11 +1,11 @@
|
|||||||
import { renderObject } from '../../../../helpers/renderer.js';
|
import { renderObject } from '../../../../helpers/renderer.js';
|
||||||
|
|
||||||
export default async (request, response) => {
|
export default async (request, response) => {
|
||||||
const flow = await request.currentUser.$relatedQuery('flows').insertAndFetch({
|
const { templateId } = request.query;
|
||||||
name: 'Name your flow',
|
|
||||||
});
|
|
||||||
|
|
||||||
await flow.createInitialSteps();
|
const flow = templateId
|
||||||
|
? await request.currentUser.createFlowFromTemplate(templateId)
|
||||||
|
: await request.currentUser.createEmptyFlow();
|
||||||
|
|
||||||
renderObject(response, flow, { status: 201 });
|
renderObject(response, flow, { status: 201 });
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import request from 'supertest';
|
|||||||
import app from '../../../../app.js';
|
import app from '../../../../app.js';
|
||||||
import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js';
|
import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.js';
|
||||||
import { createUser } from '../../../../../test/factories/user.js';
|
import { createUser } from '../../../../../test/factories/user.js';
|
||||||
|
import { createTemplate } from '../../../../../test/factories/template.js';
|
||||||
import createFlowMock from '../../../../../test/mocks/rest/api/v1/flows/create-flow.js';
|
import createFlowMock from '../../../../../test/mocks/rest/api/v1/flows/create-flow.js';
|
||||||
import { createPermission } from '../../../../../test/factories/permission.js';
|
import { createPermission } from '../../../../../test/factories/permission.js';
|
||||||
|
|
||||||
@@ -17,7 +18,7 @@ describe('POST /api/v1/flows', () => {
|
|||||||
token = await createAuthTokenByUserId(currentUser.id);
|
token = await createAuthTokenByUserId(currentUser.id);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return created flow', async () => {
|
it('should create an empty flow when no templateId is provided', async () => {
|
||||||
await createPermission({
|
await createPermission({
|
||||||
action: 'create',
|
action: 'create',
|
||||||
subject: 'Flow',
|
subject: 'Flow',
|
||||||
@@ -38,4 +39,25 @@ describe('POST /api/v1/flows', () => {
|
|||||||
|
|
||||||
expect(response.body).toMatchObject(expectedPayload);
|
expect(response.body).toMatchObject(expectedPayload);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should create a flow from template when templateId is provided', async () => {
|
||||||
|
await createPermission({
|
||||||
|
action: 'create',
|
||||||
|
subject: 'Flow',
|
||||||
|
roleId: currentUserRole.id,
|
||||||
|
conditions: ['isCreator'],
|
||||||
|
});
|
||||||
|
|
||||||
|
const template = await createTemplate({
|
||||||
|
name: 'Sample template',
|
||||||
|
});
|
||||||
|
|
||||||
|
const response = await request(app)
|
||||||
|
.post('/api/v1/flows')
|
||||||
|
.query({ templateId: template.id })
|
||||||
|
.set('Authorization', token)
|
||||||
|
.expect(201);
|
||||||
|
|
||||||
|
expect(response.body.data.name).toBe(template.flowData.name);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import { renderObject } from '../../../../helpers/renderer.js';
|
import { renderObject } from '../../../../helpers/renderer.js';
|
||||||
import importFlow from '../../../../helpers/import-flow.js';
|
import Flow from '../../../../models/flow.js';
|
||||||
|
|
||||||
export default async function importFlowController(request, response) {
|
export default async function importFlowController(request, response) {
|
||||||
const flow = await importFlow(
|
const flow = await Flow.import(request.currentUser, flowParams(request));
|
||||||
request.currentUser,
|
|
||||||
flowParams(request),
|
|
||||||
response
|
|
||||||
);
|
|
||||||
|
|
||||||
return renderObject(response, flow, { status: 201 });
|
return renderObject(response, flow, { status: 201 });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,12 @@
|
|||||||
import Crypto from 'crypto';
|
import Crypto from 'crypto';
|
||||||
import Step from '../models/step.js';
|
import Step from '../models/step.js';
|
||||||
import { renderObjectionError } from './renderer.js';
|
import { ValidationError } from 'objection';
|
||||||
|
|
||||||
const importFlow = async (user, flowData, response) => {
|
const importFlow = async (user, flowData) => {
|
||||||
const steps = flowData.steps || [];
|
const steps = flowData.steps || [];
|
||||||
|
|
||||||
// Validation: the first step must be a trigger
|
|
||||||
if (!steps.length || steps[0].type !== 'trigger') {
|
if (!steps.length || steps[0].type !== 'trigger') {
|
||||||
return renderObjectionError(response, {
|
throw new ValidationError({
|
||||||
statusCode: 422,
|
|
||||||
type: 'ValidationError',
|
type: 'ValidationError',
|
||||||
data: {
|
data: {
|
||||||
steps: [{ message: 'The first step must be a trigger!' }],
|
steps: [{ message: 'The first step must be a trigger!' }],
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import globalVariable from '../helpers/global-variable.js';
|
|||||||
import logger from '../helpers/logger.js';
|
import logger from '../helpers/logger.js';
|
||||||
import Telemetry from '../helpers/telemetry/index.js';
|
import Telemetry from '../helpers/telemetry/index.js';
|
||||||
import exportFlow from '../helpers/export-flow.js';
|
import exportFlow from '../helpers/export-flow.js';
|
||||||
|
import importFlow from '../helpers/import-flow.js';
|
||||||
import flowQueue from '../queues/flow.js';
|
import flowQueue from '../queues/flow.js';
|
||||||
import {
|
import {
|
||||||
REMOVE_AFTER_30_DAYS_OR_150_JOBS,
|
REMOVE_AFTER_30_DAYS_OR_150_JOBS,
|
||||||
@@ -99,6 +100,10 @@ class Flow extends Base {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
static async import(user, flowData) {
|
||||||
|
return importFlow(user, flowData);
|
||||||
|
}
|
||||||
|
|
||||||
static async populateStatusProperty(flows) {
|
static async populateStatusProperty(flows) {
|
||||||
const referenceFlow = flows[0];
|
const referenceFlow = flows[0];
|
||||||
|
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ import Step from './step.js';
|
|||||||
import Subscription from './subscription.ee.js';
|
import Subscription from './subscription.ee.js';
|
||||||
import Folder from './folder.js';
|
import Folder from './folder.js';
|
||||||
import UsageData from './usage-data.ee.js';
|
import UsageData from './usage-data.ee.js';
|
||||||
|
import Template from './template.ee.js';
|
||||||
import Billing from '../helpers/billing/index.ee.js';
|
import Billing from '../helpers/billing/index.ee.js';
|
||||||
import NotAuthorizedError from '../errors/not-authorized.js';
|
import NotAuthorizedError from '../errors/not-authorized.js';
|
||||||
|
|
||||||
@@ -675,6 +676,26 @@ class User extends Base {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async createEmptyFlow() {
|
||||||
|
const flow = await this.$relatedQuery('flows').insertAndFetch({
|
||||||
|
name: 'Name your flow',
|
||||||
|
});
|
||||||
|
|
||||||
|
await flow.createInitialSteps();
|
||||||
|
|
||||||
|
return flow;
|
||||||
|
}
|
||||||
|
|
||||||
|
async createFlowFromTemplate(templateId) {
|
||||||
|
const template = await Template.query()
|
||||||
|
.findById(templateId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
const flow = await Flow.import(this, template.flowData);
|
||||||
|
|
||||||
|
return flow;
|
||||||
|
}
|
||||||
|
|
||||||
async $beforeInsert(queryContext) {
|
async $beforeInsert(queryContext) {
|
||||||
await super.$beforeInsert(queryContext);
|
await super.$beforeInsert(queryContext);
|
||||||
|
|
||||||
|
|||||||
@@ -34,7 +34,9 @@ import { createExecution } from '../../test/factories/execution.js';
|
|||||||
import { createSubscription } from '../../test/factories/subscription.js';
|
import { createSubscription } from '../../test/factories/subscription.js';
|
||||||
import { createUsageData } from '../../test/factories/usage-data.js';
|
import { createUsageData } from '../../test/factories/usage-data.js';
|
||||||
import { createFolder } from '../../test/factories/folder.js';
|
import { createFolder } from '../../test/factories/folder.js';
|
||||||
|
import { createTemplate } from '../../test/factories/template.js';
|
||||||
import Billing from '../helpers/billing/index.ee.js';
|
import Billing from '../helpers/billing/index.ee.js';
|
||||||
|
import Template from './template.ee.js';
|
||||||
|
|
||||||
describe('User model', () => {
|
describe('User model', () => {
|
||||||
it('tableName should return correct name', () => {
|
it('tableName should return correct name', () => {
|
||||||
@@ -1507,6 +1509,61 @@ describe('User model', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('createEmptyFlow', () => {
|
||||||
|
it('should create a flow with default name', async () => {
|
||||||
|
const user = await createUser();
|
||||||
|
const flow = await user.createEmptyFlow();
|
||||||
|
|
||||||
|
expect(flow.name).toBe('Name your flow');
|
||||||
|
expect(flow.userId).toBe(user.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should call createInitialSteps on the created flow', async () => {
|
||||||
|
const user = await createUser();
|
||||||
|
const createInitialStepsSpy = vi.spyOn(
|
||||||
|
Flow.prototype,
|
||||||
|
'createInitialSteps'
|
||||||
|
);
|
||||||
|
|
||||||
|
await user.createEmptyFlow();
|
||||||
|
|
||||||
|
expect(createInitialStepsSpy).toHaveBeenCalledOnce();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('createFlowFromTemplate', () => {
|
||||||
|
let user, template;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
user = await createUser();
|
||||||
|
template = await createTemplate();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should throw an error if template is not found', async () => {
|
||||||
|
const nonExistentTemplateId = Crypto.randomUUID();
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
user.createFlowFromTemplate(nonExistentTemplateId)
|
||||||
|
).rejects.toThrow('NotFoundError');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should call Flow.import with the correct parameters', async () => {
|
||||||
|
vi.spyOn(Template.query(), 'findById').mockImplementation(() => ({
|
||||||
|
throwIfNotFound: () => template,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const importSpy = vi.spyOn(Flow, 'import').mockResolvedValue({
|
||||||
|
id: Crypto.randomUUID(),
|
||||||
|
name: template.flowData.name,
|
||||||
|
steps: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.createFlowFromTemplate(template.id);
|
||||||
|
|
||||||
|
expect(importSpy).toHaveBeenCalledWith(user, template.flowData);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('$beforeInsert', () => {
|
describe('$beforeInsert', () => {
|
||||||
it('should call super.$beforeInsert', async () => {
|
it('should call super.$beforeInsert', async () => {
|
||||||
const superBeforeInsertSpy = vi
|
const superBeforeInsertSpy = vi
|
||||||
|
|||||||
Reference in New Issue
Block a user