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:
Ali BARIN
2025-03-10 16:40:12 +01:00
committed by GitHub
7 changed files with 115 additions and 16 deletions

View File

@@ -1,11 +1,11 @@
import { renderObject } from '../../../../helpers/renderer.js';
export default async (request, response) => {
const flow = await request.currentUser.$relatedQuery('flows').insertAndFetch({
name: 'Name your flow',
});
const { templateId } = request.query;
await flow.createInitialSteps();
const flow = templateId
? await request.currentUser.createFlowFromTemplate(templateId)
: await request.currentUser.createEmptyFlow();
renderObject(response, flow, { status: 201 });
};

View File

@@ -4,6 +4,7 @@ import request from 'supertest';
import app from '../../../../app.js';
import createAuthTokenByUserId from '../../../../helpers/create-auth-token-by-user-id.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 { createPermission } from '../../../../../test/factories/permission.js';
@@ -17,7 +18,7 @@ describe('POST /api/v1/flows', () => {
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({
action: 'create',
subject: 'Flow',
@@ -38,4 +39,25 @@ describe('POST /api/v1/flows', () => {
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);
});
});

View File

@@ -1,12 +1,8 @@
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) {
const flow = await importFlow(
request.currentUser,
flowParams(request),
response
);
const flow = await Flow.import(request.currentUser, flowParams(request));
return renderObject(response, flow, { status: 201 });
}

View File

@@ -1,14 +1,12 @@
import Crypto from 'crypto';
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 || [];
// Validation: the first step must be a trigger
if (!steps.length || steps[0].type !== 'trigger') {
return renderObjectionError(response, {
statusCode: 422,
throw new ValidationError({
type: 'ValidationError',
data: {
steps: [{ message: 'The first step must be a trigger!' }],

View File

@@ -9,6 +9,7 @@ import globalVariable from '../helpers/global-variable.js';
import logger from '../helpers/logger.js';
import Telemetry from '../helpers/telemetry/index.js';
import exportFlow from '../helpers/export-flow.js';
import importFlow from '../helpers/import-flow.js';
import flowQueue from '../queues/flow.js';
import {
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) {
const referenceFlow = flows[0];

View File

@@ -22,6 +22,7 @@ import Step from './step.js';
import Subscription from './subscription.ee.js';
import Folder from './folder.js';
import UsageData from './usage-data.ee.js';
import Template from './template.ee.js';
import Billing from '../helpers/billing/index.ee.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) {
await super.$beforeInsert(queryContext);

View File

@@ -34,7 +34,9 @@ import { createExecution } from '../../test/factories/execution.js';
import { createSubscription } from '../../test/factories/subscription.js';
import { createUsageData } from '../../test/factories/usage-data.js';
import { createFolder } from '../../test/factories/folder.js';
import { createTemplate } from '../../test/factories/template.js';
import Billing from '../helpers/billing/index.ee.js';
import Template from './template.ee.js';
describe('User model', () => {
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', () => {
it('should call super.$beforeInsert', async () => {
const superBeforeInsertSpy = vi