2022-10-11 05:55:05 -07:00
|
|
|
import express from 'express';
|
2022-12-06 00:25:39 -08:00
|
|
|
import { v4 as uuid } from 'uuid';
|
2022-11-09 06:25:00 -08:00
|
|
|
import * as Db from '@/Db';
|
|
|
|
import * as ResponseHelper from '@/ResponseHelper';
|
|
|
|
import * as WorkflowHelpers from '@/WorkflowHelpers';
|
|
|
|
import config from '@/config';
|
|
|
|
import { WorkflowEntity } from '@db/entities/WorkflowEntity';
|
|
|
|
import { validateEntity } from '@/GenericHelpers';
|
|
|
|
import type { WorkflowRequest } from '@/requests';
|
|
|
|
import { isSharingEnabled, rightDiff } from '@/UserManagement/UserManagementHelper';
|
2022-10-11 05:55:05 -07:00
|
|
|
import { EEWorkflowsService as EEWorkflows } from './workflows.services.ee';
|
2022-12-21 01:46:26 -08:00
|
|
|
import { ExternalHooks } from '@/ExternalHooks';
|
2022-11-09 06:25:00 -08:00
|
|
|
import { SharedWorkflow } from '@db/entities/SharedWorkflow';
|
2022-10-13 02:55:58 -07:00
|
|
|
import { LoggerProxy } from 'n8n-workflow';
|
2022-11-09 06:25:00 -08:00
|
|
|
import * as TagHelpers from '@/TagHelpers';
|
2022-10-13 02:55:58 -07:00
|
|
|
import { EECredentialsService as EECredentials } from '../credentials/credentials.service.ee';
|
2023-01-27 05:56:56 -08:00
|
|
|
import type { IExecutionPushResponse } from '@/Interfaces';
|
2022-11-11 02:14:45 -08:00
|
|
|
import * as GenericHelpers from '@/GenericHelpers';
|
2023-01-13 09:12:22 -08:00
|
|
|
import { In } from 'typeorm';
|
2023-02-21 10:21:56 -08:00
|
|
|
import { Container } from 'typedi';
|
|
|
|
import { InternalHooks } from '@/InternalHooks';
|
2022-10-11 05:55:05 -07:00
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
export const EEWorkflowController = express.Router();
|
|
|
|
|
|
|
|
EEWorkflowController.use((req, res, next) => {
|
2022-12-21 07:42:07 -08:00
|
|
|
if (!isSharingEnabled()) {
|
2022-10-11 05:55:05 -07:00
|
|
|
// skip ee router and use free one
|
|
|
|
next('router');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// use ee router
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* (EE) PUT /workflows/:id/share
|
|
|
|
*
|
|
|
|
* Grant or remove users' access to a workflow.
|
|
|
|
*/
|
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
EEWorkflowController.put(
|
|
|
|
'/:workflowId/share',
|
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.Share) => {
|
|
|
|
const { workflowId } = req.params;
|
|
|
|
const { shareWithIds } = req.body;
|
|
|
|
|
|
|
|
if (
|
|
|
|
!Array.isArray(shareWithIds) ||
|
|
|
|
!shareWithIds.every((userId) => typeof userId === 'string')
|
|
|
|
) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.BadRequestError('Bad request');
|
2022-11-16 05:13:36 -08:00
|
|
|
}
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
const { ownsWorkflow, workflow } = await EEWorkflows.isOwned(req.user, workflowId);
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
if (!ownsWorkflow || !workflow) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.UnauthorizedError('Forbidden');
|
2022-11-16 05:13:36 -08:00
|
|
|
}
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
let newShareeIds: string[] = [];
|
|
|
|
await Db.transaction(async (trx) => {
|
|
|
|
// remove all sharings that are not supposed to exist anymore
|
|
|
|
await EEWorkflows.pruneSharings(trx, workflowId, [req.user.id, ...shareWithIds]);
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
const sharings = await EEWorkflows.getSharings(trx, workflowId);
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
// extract the new sharings that need to be added
|
|
|
|
newShareeIds = rightDiff(
|
|
|
|
[sharings, (sharing) => sharing.userId],
|
|
|
|
[shareWithIds, (shareeId) => shareeId],
|
|
|
|
);
|
2022-10-11 05:55:05 -07:00
|
|
|
|
2022-11-16 05:13:36 -08:00
|
|
|
if (newShareeIds.length) {
|
|
|
|
await EEWorkflows.share(trx, workflow, newShareeIds);
|
|
|
|
}
|
|
|
|
});
|
2022-12-21 07:42:07 -08:00
|
|
|
|
2023-02-21 10:21:56 -08:00
|
|
|
void Container.get(InternalHooks).onWorkflowSharingUpdate(
|
2022-12-21 07:42:07 -08:00
|
|
|
workflowId,
|
|
|
|
req.user.id,
|
|
|
|
shareWithIds,
|
|
|
|
);
|
2022-11-16 05:13:36 -08:00
|
|
|
}),
|
|
|
|
);
|
2022-10-11 07:40:39 -07:00
|
|
|
|
|
|
|
EEWorkflowController.get(
|
2022-10-13 02:55:58 -07:00
|
|
|
'/:id(\\d+)',
|
2022-10-11 07:40:39 -07:00
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.Get) => {
|
|
|
|
const { id: workflowId } = req.params;
|
|
|
|
|
2022-12-22 01:16:27 -08:00
|
|
|
const relations = ['shared', 'shared.user', 'shared.role'];
|
|
|
|
if (!config.getEnv('workflowTagsDisabled')) {
|
|
|
|
relations.push('tags');
|
|
|
|
}
|
|
|
|
|
2023-01-02 08:42:32 -08:00
|
|
|
const workflow = await EEWorkflows.get({ id: workflowId }, { relations });
|
2022-10-11 07:40:39 -07:00
|
|
|
|
|
|
|
if (!workflow) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.NotFoundError(`Workflow with ID "${workflowId}" does not exist`);
|
2022-10-11 07:40:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const userSharing = workflow.shared?.find((shared) => shared.user.id === req.user.id);
|
|
|
|
|
|
|
|
if (!userSharing && req.user.globalRole.name !== 'owner') {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.UnauthorizedError(
|
2022-12-21 07:42:07 -08:00
|
|
|
'You do not have permission to access this workflow. Ask the owner to share it with you',
|
2022-11-22 04:05:51 -08:00
|
|
|
);
|
2022-10-11 07:40:39 -07:00
|
|
|
}
|
|
|
|
|
2022-11-25 05:20:28 -08:00
|
|
|
EEWorkflows.addOwnerAndSharings(workflow);
|
|
|
|
await EEWorkflows.addCredentialsToWorkflow(workflow, req.user);
|
2023-01-02 08:42:32 -08:00
|
|
|
return workflow;
|
2022-10-11 07:40:39 -07:00
|
|
|
}),
|
|
|
|
);
|
2022-10-13 02:55:58 -07:00
|
|
|
|
|
|
|
EEWorkflowController.post(
|
|
|
|
'/',
|
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.Create) => {
|
|
|
|
delete req.body.id; // delete if sent
|
|
|
|
|
|
|
|
const newWorkflow = new WorkflowEntity();
|
|
|
|
|
|
|
|
Object.assign(newWorkflow, req.body);
|
|
|
|
|
2022-12-06 00:25:39 -08:00
|
|
|
newWorkflow.versionId = uuid();
|
|
|
|
|
2022-10-13 02:55:58 -07:00
|
|
|
await validateEntity(newWorkflow);
|
|
|
|
|
2023-02-21 10:21:56 -08:00
|
|
|
await Container.get(ExternalHooks).run('workflow.create', [newWorkflow]);
|
2022-10-13 02:55:58 -07:00
|
|
|
|
|
|
|
const { tags: tagIds } = req.body;
|
|
|
|
|
|
|
|
if (tagIds?.length && !config.getEnv('workflowTagsDisabled')) {
|
2023-01-13 09:12:22 -08:00
|
|
|
newWorkflow.tags = await Db.collections.Tag.find({
|
2022-10-13 02:55:58 -07:00
|
|
|
select: ['id', 'name'],
|
2023-01-13 09:12:22 -08:00
|
|
|
where: {
|
|
|
|
id: In(tagIds),
|
|
|
|
},
|
2022-10-13 02:55:58 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
await WorkflowHelpers.replaceInvalidCredentials(newWorkflow);
|
|
|
|
|
|
|
|
WorkflowHelpers.addNodeIds(newWorkflow);
|
|
|
|
|
|
|
|
// This is a new workflow, so we simply check if the user has access to
|
|
|
|
// all used workflows
|
|
|
|
|
|
|
|
const allCredentials = await EECredentials.getAll(req.user);
|
|
|
|
|
|
|
|
try {
|
|
|
|
EEWorkflows.validateCredentialPermissionsToUser(newWorkflow, allCredentials);
|
|
|
|
} catch (error) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.BadRequestError(
|
2022-11-22 04:05:51 -08:00
|
|
|
'The workflow you are trying to save contains credentials that are not shared with you',
|
2022-10-13 02:55:58 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
let savedWorkflow: undefined | WorkflowEntity;
|
|
|
|
|
|
|
|
await Db.transaction(async (transactionManager) => {
|
|
|
|
savedWorkflow = await transactionManager.save<WorkflowEntity>(newWorkflow);
|
|
|
|
|
2023-01-13 09:12:22 -08:00
|
|
|
const role = await Db.collections.Role.findOneByOrFail({
|
2022-10-13 02:55:58 -07:00
|
|
|
name: 'owner',
|
|
|
|
scope: 'workflow',
|
|
|
|
});
|
|
|
|
|
|
|
|
const newSharedWorkflow = new SharedWorkflow();
|
|
|
|
|
|
|
|
Object.assign(newSharedWorkflow, {
|
|
|
|
role,
|
|
|
|
user: req.user,
|
|
|
|
workflow: savedWorkflow,
|
|
|
|
});
|
|
|
|
|
|
|
|
await transactionManager.save<SharedWorkflow>(newSharedWorkflow);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!savedWorkflow) {
|
|
|
|
LoggerProxy.error('Failed to create workflow', { userId: req.user.id });
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.InternalServerError(
|
2022-11-22 04:05:51 -08:00
|
|
|
'An error occurred while saving your workflow. Please try again.',
|
|
|
|
);
|
2022-10-13 02:55:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tagIds && !config.getEnv('workflowTagsDisabled') && savedWorkflow.tags) {
|
|
|
|
savedWorkflow.tags = TagHelpers.sortByRequestOrder(savedWorkflow.tags, {
|
|
|
|
requestOrder: tagIds,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-02-21 10:21:56 -08:00
|
|
|
await Container.get(ExternalHooks).run('workflow.afterCreate', [savedWorkflow]);
|
|
|
|
void Container.get(InternalHooks).onWorkflowCreated(req.user, newWorkflow, false);
|
2022-10-13 02:55:58 -07:00
|
|
|
|
2023-01-02 08:42:32 -08:00
|
|
|
return savedWorkflow;
|
2022-10-13 02:55:58 -07:00
|
|
|
}),
|
|
|
|
);
|
2022-10-26 06:49:43 -07:00
|
|
|
|
2022-11-08 08:52:42 -08:00
|
|
|
/**
|
|
|
|
* (EE) GET /workflows
|
|
|
|
*/
|
|
|
|
EEWorkflowController.get(
|
|
|
|
'/',
|
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.GetAll) => {
|
2023-02-16 01:36:24 -08:00
|
|
|
const [workflows, workflowOwnerRole] = await Promise.all([
|
|
|
|
EEWorkflows.getMany(req.user, req.query.filter),
|
|
|
|
Db.collections.Role.findOneOrFail({
|
|
|
|
select: ['id'],
|
|
|
|
where: { name: 'owner', scope: 'workflow' },
|
|
|
|
}),
|
|
|
|
]);
|
2022-12-23 04:58:34 -08:00
|
|
|
|
|
|
|
return workflows.map((workflow) => {
|
2023-02-16 01:36:24 -08:00
|
|
|
EEWorkflows.addOwnerId(workflow, workflowOwnerRole);
|
2023-01-02 08:42:32 -08:00
|
|
|
return workflow;
|
2022-12-23 04:58:34 -08:00
|
|
|
});
|
2022-11-08 08:52:42 -08:00
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
2022-10-26 06:49:43 -07:00
|
|
|
EEWorkflowController.patch(
|
|
|
|
'/:id(\\d+)',
|
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.Update) => {
|
|
|
|
const { id: workflowId } = req.params;
|
2022-12-06 00:25:39 -08:00
|
|
|
const forceSave = req.query.forceSave === 'true';
|
2022-10-26 06:49:43 -07:00
|
|
|
|
|
|
|
const updateData = new WorkflowEntity();
|
|
|
|
const { tags, ...rest } = req.body;
|
|
|
|
Object.assign(updateData, rest);
|
|
|
|
|
2022-11-11 02:14:45 -08:00
|
|
|
const safeWorkflow = await EEWorkflows.preventTampering(updateData, workflowId, req.user);
|
|
|
|
|
2022-11-18 04:07:39 -08:00
|
|
|
const updatedWorkflow = await EEWorkflows.update(
|
2022-10-26 06:49:43 -07:00
|
|
|
req.user,
|
2022-11-11 02:14:45 -08:00
|
|
|
safeWorkflow,
|
2022-10-26 06:49:43 -07:00
|
|
|
workflowId,
|
|
|
|
tags,
|
2022-12-06 00:25:39 -08:00
|
|
|
forceSave,
|
2022-10-26 06:49:43 -07:00
|
|
|
);
|
|
|
|
|
2023-01-02 08:42:32 -08:00
|
|
|
return updatedWorkflow;
|
2022-10-26 06:49:43 -07:00
|
|
|
}),
|
|
|
|
);
|
2022-11-11 02:14:45 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* (EE) POST /workflows/run
|
|
|
|
*/
|
|
|
|
EEWorkflowController.post(
|
|
|
|
'/run',
|
|
|
|
ResponseHelper.send(async (req: WorkflowRequest.ManualRun): Promise<IExecutionPushResponse> => {
|
|
|
|
const workflow = new WorkflowEntity();
|
|
|
|
Object.assign(workflow, req.body.workflowData);
|
|
|
|
|
2022-11-21 23:37:52 -08:00
|
|
|
if (workflow.id !== undefined) {
|
2023-01-02 08:42:32 -08:00
|
|
|
const safeWorkflow = await EEWorkflows.preventTampering(workflow, workflow.id, req.user);
|
2022-11-21 23:37:52 -08:00
|
|
|
req.body.workflowData.nodes = safeWorkflow.nodes;
|
|
|
|
}
|
2022-11-11 02:14:45 -08:00
|
|
|
|
2022-11-18 04:07:39 -08:00
|
|
|
return EEWorkflows.runManually(req.body, req.user, GenericHelpers.getSessionId(req));
|
2022-11-11 02:14:45 -08:00
|
|
|
}),
|
|
|
|
);
|