2022-11-08 08:52:42 -08:00
|
|
|
import { validate as jsonSchemaValidate } from 'jsonschema';
|
2022-11-11 02:14:45 -08:00
|
|
|
import { INode, IPinData, JsonObject, jsonParse, LoggerProxy, Workflow } from 'n8n-workflow';
|
|
|
|
import { FindManyOptions, FindOneOptions, In, ObjectLiteral } from 'typeorm';
|
2022-11-21 06:51:23 -08:00
|
|
|
import pick from 'lodash.pick';
|
2022-12-06 00:25:39 -08:00
|
|
|
import { v4 as uuid } from 'uuid';
|
2022-11-09 06:25:00 -08:00
|
|
|
import * as ActiveWorkflowRunner from '@/ActiveWorkflowRunner';
|
|
|
|
import * as Db from '@/Db';
|
|
|
|
import { InternalHooksManager } from '@/InternalHooksManager';
|
|
|
|
import * as ResponseHelper from '@/ResponseHelper';
|
|
|
|
import * as WorkflowHelpers from '@/WorkflowHelpers';
|
|
|
|
import config from '@/config';
|
|
|
|
import { SharedWorkflow } from '@db/entities/SharedWorkflow';
|
|
|
|
import { User } from '@db/entities/User';
|
|
|
|
import { WorkflowEntity } from '@db/entities/WorkflowEntity';
|
|
|
|
import { validateEntity } from '@/GenericHelpers';
|
2022-12-21 01:46:26 -08:00
|
|
|
import { ExternalHooks } from '@/ExternalHooks';
|
2022-11-09 06:25:00 -08:00
|
|
|
import * as TagHelpers from '@/TagHelpers';
|
2022-11-11 02:14:45 -08:00
|
|
|
import { WorkflowRequest } from '@/requests';
|
|
|
|
import { IWorkflowDb, IWorkflowExecutionDataProcess } from '@/Interfaces';
|
|
|
|
import { NodeTypes } from '@/NodeTypes';
|
|
|
|
import { WorkflowRunner } from '@/WorkflowRunner';
|
|
|
|
import * as WorkflowExecuteAdditionalData from '@/WorkflowExecuteAdditionalData';
|
|
|
|
import * as TestWebhooks from '@/TestWebhooks';
|
2022-11-09 06:25:00 -08:00
|
|
|
import { getSharedWorkflowIds } from '@/WorkflowHelpers';
|
2022-11-18 04:07:39 -08:00
|
|
|
import { whereClause } from '@/UserManagement/UserManagementHelper';
|
2022-11-08 08:52:42 -08:00
|
|
|
|
|
|
|
export interface IGetWorkflowsQueryFilter {
|
|
|
|
id?: number | string;
|
|
|
|
name?: string;
|
|
|
|
active?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const schemaGetWorkflowsQueryFilter = {
|
|
|
|
$id: '/IGetWorkflowsQueryFilter',
|
|
|
|
type: 'object',
|
|
|
|
properties: {
|
|
|
|
id: { anyOf: [{ type: 'integer' }, { type: 'string' }] },
|
|
|
|
name: { type: 'string' },
|
|
|
|
active: { type: 'boolean' },
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const allowedWorkflowsQueryFilterFields = Object.keys(schemaGetWorkflowsQueryFilter.properties);
|
2022-10-11 05:55:05 -07:00
|
|
|
|
|
|
|
export class WorkflowsService {
|
|
|
|
static async getSharing(
|
|
|
|
user: User,
|
|
|
|
workflowId: number | string,
|
|
|
|
relations: string[] = ['workflow'],
|
|
|
|
{ allowGlobalOwner } = { allowGlobalOwner: true },
|
|
|
|
): Promise<SharedWorkflow | undefined> {
|
|
|
|
const options: FindOneOptions<SharedWorkflow> & { where: ObjectLiteral } = {
|
|
|
|
where: {
|
|
|
|
workflow: { id: workflowId },
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
// Omit user from where if the requesting user is the global
|
|
|
|
// owner. This allows the global owner to view and delete
|
|
|
|
// workflows they don't own.
|
|
|
|
if (!allowGlobalOwner || user.globalRole.name !== 'owner') {
|
|
|
|
options.where.user = { id: user.id };
|
|
|
|
}
|
|
|
|
|
|
|
|
if (relations?.length) {
|
|
|
|
options.relations = relations;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Db.collections.SharedWorkflow.findOne(options);
|
|
|
|
}
|
2022-10-11 07:40:39 -07:00
|
|
|
|
2022-11-10 05:03:14 -08:00
|
|
|
/**
|
|
|
|
* Find the pinned trigger to execute the workflow from, if any.
|
2022-12-05 01:09:31 -08:00
|
|
|
*
|
|
|
|
* - In a full execution, select the _first_ pinned trigger.
|
|
|
|
* - In a partial execution,
|
|
|
|
* - select the _first_ pinned trigger that leads to the executed node,
|
|
|
|
* - else select the executed pinned trigger.
|
2022-11-10 05:03:14 -08:00
|
|
|
*/
|
|
|
|
static findPinnedTrigger(workflow: IWorkflowDb, startNodes?: string[], pinData?: IPinData) {
|
|
|
|
if (!pinData || !startNodes) return null;
|
|
|
|
|
|
|
|
const isTrigger = (nodeTypeName: string) =>
|
|
|
|
['trigger', 'webhook'].some((suffix) => nodeTypeName.toLowerCase().includes(suffix));
|
|
|
|
|
|
|
|
const pinnedTriggers = workflow.nodes.filter(
|
|
|
|
(node) => !node.disabled && pinData[node.name] && isTrigger(node.type),
|
|
|
|
);
|
|
|
|
|
|
|
|
if (pinnedTriggers.length === 0) return null;
|
|
|
|
|
|
|
|
if (startNodes?.length === 0) return pinnedTriggers[0]; // full execution
|
|
|
|
|
|
|
|
const [startNodeName] = startNodes;
|
|
|
|
|
2022-12-05 01:09:31 -08:00
|
|
|
const parentNames = new Workflow({
|
|
|
|
nodes: workflow.nodes,
|
|
|
|
connections: workflow.connections,
|
|
|
|
active: workflow.active,
|
|
|
|
nodeTypes: NodeTypes(),
|
|
|
|
}).getParentNodes(startNodeName);
|
|
|
|
|
|
|
|
let checkNodeName = '';
|
|
|
|
|
|
|
|
if (parentNames.length === 0) {
|
|
|
|
checkNodeName = startNodeName;
|
|
|
|
} else {
|
|
|
|
checkNodeName = parentNames.find((pn) => pn === pinnedTriggers[0].name) as string;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pinnedTriggers.find((pt) => pt.name === checkNodeName) ?? null; // partial execution
|
2022-11-10 05:03:14 -08:00
|
|
|
}
|
|
|
|
|
2022-10-11 07:40:39 -07:00
|
|
|
static async get(workflow: Partial<WorkflowEntity>, options?: { relations: string[] }) {
|
|
|
|
return Db.collections.Workflow.findOne(workflow, options);
|
|
|
|
}
|
2022-10-26 06:49:43 -07:00
|
|
|
|
2022-11-18 04:07:39 -08:00
|
|
|
// Warning: this function is overriden by EE to disregard role list.
|
2022-12-19 08:53:36 -08:00
|
|
|
static async getWorkflowIdsForUser(user: User, roles?: string[]): Promise<string[]> {
|
2022-11-18 04:07:39 -08:00
|
|
|
return getSharedWorkflowIds(user, roles);
|
|
|
|
}
|
|
|
|
|
2022-11-08 08:52:42 -08:00
|
|
|
static async getMany(user: User, rawFilter: string) {
|
2022-11-18 04:07:39 -08:00
|
|
|
const sharedWorkflowIds = await this.getWorkflowIdsForUser(user, ['owner']);
|
2022-11-08 08:52:42 -08:00
|
|
|
if (sharedWorkflowIds.length === 0) {
|
|
|
|
// return early since without shared workflows there can be no hits
|
|
|
|
// (note: getSharedWorkflowIds() returns _all_ workflow ids for global owners)
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
let filter: IGetWorkflowsQueryFilter | undefined = undefined;
|
|
|
|
if (rawFilter) {
|
|
|
|
try {
|
|
|
|
const filterJson: JsonObject = jsonParse(rawFilter);
|
|
|
|
if (filterJson) {
|
|
|
|
Object.keys(filterJson).map((key) => {
|
|
|
|
if (!allowedWorkflowsQueryFilterFields.includes(key)) delete filterJson[key];
|
|
|
|
});
|
|
|
|
if (jsonSchemaValidate(filterJson, schemaGetWorkflowsQueryFilter).valid) {
|
|
|
|
filter = filterJson as IGetWorkflowsQueryFilter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
LoggerProxy.error('Failed to parse filter', {
|
|
|
|
userId: user.id,
|
|
|
|
filter,
|
|
|
|
});
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.InternalServerError(
|
2022-11-08 08:52:42 -08:00
|
|
|
`Parameter "filter" contained invalid JSON string.`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// safeguard against querying ids not shared with the user
|
2022-12-19 08:53:36 -08:00
|
|
|
const workflowId = filter?.id?.toString();
|
|
|
|
if (workflowId !== undefined && !sharedWorkflowIds.includes(workflowId)) {
|
|
|
|
LoggerProxy.verbose(`User ${user.id} attempted to query non-shared workflow ${workflowId}`);
|
|
|
|
return [];
|
2022-11-08 08:52:42 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
const fields: Array<keyof WorkflowEntity> = ['id', 'name', 'active', 'createdAt', 'updatedAt'];
|
2022-11-25 05:20:28 -08:00
|
|
|
const relations: string[] = [];
|
2022-11-08 08:52:42 -08:00
|
|
|
|
2022-11-25 05:20:28 -08:00
|
|
|
if (!config.getEnv('workflowTagsDisabled')) {
|
|
|
|
relations.push('tags');
|
|
|
|
}
|
2022-11-08 08:52:42 -08:00
|
|
|
|
2022-11-25 05:20:28 -08:00
|
|
|
const isSharingEnabled = config.getEnv('enterprise.features.sharing');
|
|
|
|
if (isSharingEnabled) {
|
|
|
|
relations.push('shared', 'shared.user', 'shared.role');
|
2022-11-08 08:52:42 -08:00
|
|
|
}
|
|
|
|
|
2022-11-25 05:20:28 -08:00
|
|
|
const query: FindManyOptions<WorkflowEntity> = {
|
2022-12-06 00:25:39 -08:00
|
|
|
select: isSharingEnabled ? [...fields, 'nodes', 'versionId'] : fields,
|
2022-11-25 05:20:28 -08:00
|
|
|
relations,
|
|
|
|
where: {
|
|
|
|
id: In(sharedWorkflowIds),
|
|
|
|
...filter,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const workflows = await Db.collections.Workflow.find(query);
|
2022-11-08 08:52:42 -08:00
|
|
|
|
|
|
|
return workflows.map((workflow) => {
|
|
|
|
const { id, ...rest } = workflow;
|
|
|
|
|
|
|
|
return {
|
|
|
|
id: id.toString(),
|
|
|
|
...rest,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-11-11 02:14:45 -08:00
|
|
|
static async update(
|
2022-10-26 06:49:43 -07:00
|
|
|
user: User,
|
|
|
|
workflow: WorkflowEntity,
|
|
|
|
workflowId: string,
|
|
|
|
tags?: string[],
|
2022-10-31 02:35:24 -07:00
|
|
|
forceSave?: boolean,
|
2022-11-18 04:07:39 -08:00
|
|
|
roles?: string[],
|
2022-10-26 06:49:43 -07:00
|
|
|
): Promise<WorkflowEntity> {
|
|
|
|
const shared = await Db.collections.SharedWorkflow.findOne({
|
2022-11-18 04:07:39 -08:00
|
|
|
relations: ['workflow', 'role'],
|
2022-10-26 06:49:43 -07:00
|
|
|
where: whereClause({
|
|
|
|
user,
|
|
|
|
entityType: 'workflow',
|
|
|
|
entityId: workflowId,
|
2022-11-18 04:07:39 -08:00
|
|
|
roles,
|
2022-10-26 06:49:43 -07:00
|
|
|
}),
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!shared) {
|
|
|
|
LoggerProxy.info('User attempted to update a workflow without permissions', {
|
|
|
|
workflowId,
|
|
|
|
userId: user.id,
|
|
|
|
});
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.NotFoundError(
|
2022-11-22 04:05:51 -08:00
|
|
|
'You do not have permission to update this workflow. Ask the owner to share it with you.',
|
2022-10-26 06:49:43 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-12-06 00:25:39 -08:00
|
|
|
if (
|
|
|
|
!forceSave &&
|
|
|
|
workflow.versionId !== '' &&
|
|
|
|
workflow.versionId !== shared.workflow.versionId
|
|
|
|
) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.BadRequestError(
|
2022-11-28 12:05:19 -08:00
|
|
|
'Your most recent changes may be lost, because someone else just updated this workflow. Open this workflow in a new tab to see those new updates.',
|
2022-12-06 00:25:39 -08:00
|
|
|
100,
|
2022-11-14 06:38:19 -08:00
|
|
|
);
|
|
|
|
}
|
2022-10-31 02:35:24 -07:00
|
|
|
|
2022-12-06 00:25:39 -08:00
|
|
|
// Update the workflow's version
|
|
|
|
workflow.versionId = uuid();
|
|
|
|
|
|
|
|
LoggerProxy.verbose(
|
|
|
|
`Updating versionId for workflow ${workflowId} for user ${user.id} after saving`,
|
|
|
|
{
|
|
|
|
previousVersionId: shared.workflow.versionId,
|
|
|
|
newVersionId: workflow.versionId,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2022-10-26 06:49:43 -07:00
|
|
|
// check credentials for old format
|
|
|
|
await WorkflowHelpers.replaceInvalidCredentials(workflow);
|
|
|
|
|
|
|
|
WorkflowHelpers.addNodeIds(workflow);
|
|
|
|
|
2022-12-21 01:46:26 -08:00
|
|
|
await ExternalHooks().run('workflow.update', [workflow]);
|
2022-10-26 06:49:43 -07:00
|
|
|
|
|
|
|
if (shared.workflow.active) {
|
|
|
|
// When workflow gets saved always remove it as the triggers could have been
|
|
|
|
// changed and so the changes would not take effect
|
|
|
|
await ActiveWorkflowRunner.getInstance().remove(workflowId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (workflow.settings) {
|
|
|
|
if (workflow.settings.timezone === 'DEFAULT') {
|
|
|
|
// Do not save the default timezone
|
|
|
|
delete workflow.settings.timezone;
|
|
|
|
}
|
|
|
|
if (workflow.settings.saveDataErrorExecution === 'DEFAULT') {
|
|
|
|
// Do not save when default got set
|
|
|
|
delete workflow.settings.saveDataErrorExecution;
|
|
|
|
}
|
|
|
|
if (workflow.settings.saveDataSuccessExecution === 'DEFAULT') {
|
|
|
|
// Do not save when default got set
|
|
|
|
delete workflow.settings.saveDataSuccessExecution;
|
|
|
|
}
|
|
|
|
if (workflow.settings.saveManualExecutions === 'DEFAULT') {
|
|
|
|
// Do not save when default got set
|
|
|
|
delete workflow.settings.saveManualExecutions;
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
parseInt(workflow.settings.executionTimeout as string, 10) ===
|
|
|
|
config.get('executions.timeout')
|
|
|
|
) {
|
|
|
|
// Do not save when default got set
|
|
|
|
delete workflow.settings.executionTimeout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (workflow.name) {
|
|
|
|
workflow.updatedAt = new Date(); // required due to atomic update
|
|
|
|
await validateEntity(workflow);
|
|
|
|
}
|
|
|
|
|
2022-11-21 06:51:23 -08:00
|
|
|
await Db.collections.Workflow.update(
|
|
|
|
workflowId,
|
|
|
|
pick(workflow, [
|
|
|
|
'name',
|
|
|
|
'active',
|
|
|
|
'nodes',
|
|
|
|
'connections',
|
|
|
|
'settings',
|
|
|
|
'staticData',
|
|
|
|
'pinData',
|
2022-12-06 00:25:39 -08:00
|
|
|
'versionId',
|
2022-11-21 06:51:23 -08:00
|
|
|
]),
|
|
|
|
);
|
2022-10-26 06:49:43 -07:00
|
|
|
|
|
|
|
if (tags && !config.getEnv('workflowTagsDisabled')) {
|
|
|
|
const tablePrefix = config.getEnv('database.tablePrefix');
|
|
|
|
await TagHelpers.removeRelations(workflowId, tablePrefix);
|
|
|
|
|
|
|
|
if (tags.length) {
|
|
|
|
await TagHelpers.createRelations(workflowId, tags, tablePrefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const options: FindManyOptions<WorkflowEntity> = {
|
|
|
|
relations: ['tags'],
|
|
|
|
};
|
|
|
|
|
|
|
|
if (config.getEnv('workflowTagsDisabled')) {
|
|
|
|
delete options.relations;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We sadly get nothing back from "update". Neither if it updated a record
|
|
|
|
// nor the new value. So query now the hopefully updated entry.
|
|
|
|
const updatedWorkflow = await Db.collections.Workflow.findOne(workflowId, options);
|
|
|
|
|
|
|
|
if (updatedWorkflow === undefined) {
|
2022-11-22 05:00:36 -08:00
|
|
|
throw new ResponseHelper.BadRequestError(
|
2022-10-26 06:49:43 -07:00
|
|
|
`Workflow with ID "${workflowId}" could not be found to be updated.`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (updatedWorkflow.tags?.length && tags?.length) {
|
|
|
|
updatedWorkflow.tags = TagHelpers.sortByRequestOrder(updatedWorkflow.tags, {
|
|
|
|
requestOrder: tags,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-12-21 01:46:26 -08:00
|
|
|
await ExternalHooks().run('workflow.afterUpdate', [updatedWorkflow]);
|
2022-10-26 06:49:43 -07:00
|
|
|
void InternalHooksManager.getInstance().onWorkflowSaved(user.id, updatedWorkflow, false);
|
|
|
|
|
|
|
|
if (updatedWorkflow.active) {
|
|
|
|
// When the workflow is supposed to be active add it again
|
|
|
|
try {
|
2022-12-21 01:46:26 -08:00
|
|
|
await ExternalHooks().run('workflow.activate', [updatedWorkflow]);
|
2022-10-26 06:49:43 -07:00
|
|
|
await ActiveWorkflowRunner.getInstance().add(
|
|
|
|
workflowId,
|
|
|
|
shared.workflow.active ? 'update' : 'activate',
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
// If workflow could not be activated set it again to inactive
|
2022-11-21 06:51:23 -08:00
|
|
|
await Db.collections.Workflow.update(workflowId, { active: false });
|
2022-10-26 06:49:43 -07:00
|
|
|
|
|
|
|
// Also set it in the returned data
|
|
|
|
updatedWorkflow.active = false;
|
|
|
|
|
|
|
|
// Now return the original error for UI to display
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return updatedWorkflow;
|
|
|
|
}
|
2022-11-11 02:14:45 -08:00
|
|
|
|
|
|
|
static async runManually(
|
|
|
|
{
|
|
|
|
workflowData,
|
|
|
|
runData,
|
|
|
|
pinData,
|
|
|
|
startNodes,
|
|
|
|
destinationNode,
|
|
|
|
}: WorkflowRequest.ManualRunPayload,
|
|
|
|
user: User,
|
|
|
|
sessionId?: string,
|
|
|
|
) {
|
|
|
|
const EXECUTION_MODE = 'manual';
|
|
|
|
const ACTIVATION_MODE = 'manual';
|
|
|
|
|
|
|
|
const pinnedTrigger = WorkflowsService.findPinnedTrigger(workflowData, startNodes, pinData);
|
|
|
|
|
|
|
|
// If webhooks nodes exist and are active we have to wait for till we receive a call
|
|
|
|
if (
|
|
|
|
pinnedTrigger === null &&
|
|
|
|
(runData === undefined ||
|
|
|
|
startNodes === undefined ||
|
|
|
|
startNodes.length === 0 ||
|
|
|
|
destinationNode === undefined)
|
|
|
|
) {
|
|
|
|
const workflow = new Workflow({
|
|
|
|
id: workflowData.id?.toString(),
|
|
|
|
name: workflowData.name,
|
|
|
|
nodes: workflowData.nodes,
|
|
|
|
connections: workflowData.connections,
|
|
|
|
active: false,
|
|
|
|
nodeTypes: NodeTypes(),
|
|
|
|
staticData: undefined,
|
|
|
|
settings: workflowData.settings,
|
|
|
|
});
|
|
|
|
|
|
|
|
const additionalData = await WorkflowExecuteAdditionalData.getBase(user.id);
|
|
|
|
|
|
|
|
const needsWebhook = await TestWebhooks.getInstance().needsWebhookData(
|
|
|
|
workflowData,
|
|
|
|
workflow,
|
|
|
|
additionalData,
|
|
|
|
EXECUTION_MODE,
|
|
|
|
ACTIVATION_MODE,
|
|
|
|
sessionId,
|
|
|
|
destinationNode,
|
|
|
|
);
|
|
|
|
if (needsWebhook) {
|
|
|
|
return {
|
|
|
|
waitingForWebhook: true,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// For manual testing always set to not active
|
|
|
|
workflowData.active = false;
|
|
|
|
|
|
|
|
// Start the workflow
|
|
|
|
const data: IWorkflowExecutionDataProcess = {
|
|
|
|
destinationNode,
|
|
|
|
executionMode: EXECUTION_MODE,
|
|
|
|
runData,
|
|
|
|
pinData,
|
|
|
|
sessionId,
|
|
|
|
startNodes,
|
|
|
|
workflowData,
|
|
|
|
userId: user.id,
|
|
|
|
};
|
|
|
|
|
|
|
|
const hasRunData = (node: INode) => runData !== undefined && !!runData[node.name];
|
|
|
|
|
|
|
|
if (pinnedTrigger && !hasRunData(pinnedTrigger)) {
|
|
|
|
data.startNodes = [pinnedTrigger.name];
|
|
|
|
}
|
|
|
|
|
|
|
|
const workflowRunner = new WorkflowRunner();
|
|
|
|
const executionId = await workflowRunner.run(data);
|
|
|
|
|
|
|
|
return {
|
|
|
|
executionId,
|
|
|
|
};
|
|
|
|
}
|
2022-10-11 05:55:05 -07:00
|
|
|
}
|