2023-03-16 07:34:13 -07:00
|
|
|
import { Container } from 'typedi';
|
2024-01-17 01:16:13 -08:00
|
|
|
import { v4 as uuid } from 'uuid';
|
2023-01-27 05:56:56 -08:00
|
|
|
import type {
|
2019-08-08 11:38:25 -07:00
|
|
|
IDataObject,
|
2021-08-29 11:58:11 -07:00
|
|
|
INode,
|
2021-10-13 15:21:00 -07:00
|
|
|
INodeCredentialsDetails,
|
2021-08-29 11:58:11 -07:00
|
|
|
IRun,
|
2019-12-19 14:07:55 -08:00
|
|
|
ITaskData,
|
2023-01-27 05:56:56 -08:00
|
|
|
NodeApiError,
|
|
|
|
WorkflowExecuteMode,
|
2023-08-22 06:26:33 -07:00
|
|
|
WorkflowOperationError,
|
2021-08-29 11:58:11 -07:00
|
|
|
Workflow,
|
2024-01-17 01:16:13 -08:00
|
|
|
NodeOperationError,
|
2021-05-01 20:43:01 -07:00
|
|
|
} from 'n8n-workflow';
|
2024-01-17 01:16:13 -08:00
|
|
|
|
|
|
|
import type { IWorkflowExecutionDataProcess } from '@/Interfaces';
|
2023-01-27 05:56:56 -08:00
|
|
|
import type { WorkflowEntity } from '@db/entities/WorkflowEntity';
|
2023-11-10 06:04:26 -08:00
|
|
|
import { CredentialsRepository } from '@db/repositories/credentials.repository';
|
2024-01-17 01:16:13 -08:00
|
|
|
import { VariablesService } from '@/environments/variables/variables.service.ee';
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2023-08-22 06:26:33 -07:00
|
|
|
export function generateFailedExecutionFromError(
|
|
|
|
mode: WorkflowExecuteMode,
|
|
|
|
error: NodeApiError | NodeOperationError | WorkflowOperationError,
|
|
|
|
node: INode,
|
|
|
|
): IRun {
|
|
|
|
return {
|
|
|
|
data: {
|
|
|
|
startData: {
|
|
|
|
destinationNode: node.name,
|
|
|
|
runNodeFilter: [node.name],
|
|
|
|
},
|
|
|
|
resultData: {
|
|
|
|
error,
|
|
|
|
runData: {
|
|
|
|
[node.name]: [
|
|
|
|
{
|
|
|
|
startTime: 0,
|
|
|
|
executionTime: 0,
|
|
|
|
error,
|
|
|
|
source: [],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
lastNodeExecuted: node.name,
|
|
|
|
},
|
|
|
|
executionData: {
|
|
|
|
contextData: {},
|
2023-10-02 08:33:43 -07:00
|
|
|
metadata: {},
|
2023-08-22 06:26:33 -07:00
|
|
|
nodeExecutionStack: [
|
|
|
|
{
|
|
|
|
node,
|
|
|
|
data: {},
|
|
|
|
source: null,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
waitingExecution: {},
|
|
|
|
waitingExecutionSource: {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
finished: false,
|
|
|
|
mode,
|
|
|
|
startedAt: new Date(),
|
|
|
|
stoppedAt: new Date(),
|
|
|
|
status: 'failed',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-12-19 14:07:55 -08:00
|
|
|
/**
|
|
|
|
* Returns the data of the last executed node
|
|
|
|
*/
|
|
|
|
export function getDataLastExecutedNodeData(inputData: IRun): ITaskData | undefined {
|
2022-07-20 08:50:39 -07:00
|
|
|
const { runData, pinData = {} } = inputData.data.resultData;
|
2019-12-19 14:07:55 -08:00
|
|
|
const { lastNodeExecuted } = inputData.data.resultData;
|
|
|
|
|
|
|
|
if (lastNodeExecuted === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (runData[lastNodeExecuted] === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2022-07-20 08:50:39 -07:00
|
|
|
const lastNodeRunData = runData[lastNodeExecuted][runData[lastNodeExecuted].length - 1];
|
|
|
|
|
|
|
|
let lastNodePinData = pinData[lastNodeExecuted];
|
|
|
|
|
2023-01-23 03:22:05 -08:00
|
|
|
if (lastNodePinData && inputData.mode === 'manual') {
|
2022-07-20 08:50:39 -07:00
|
|
|
if (!Array.isArray(lastNodePinData)) lastNodePinData = [lastNodePinData];
|
|
|
|
|
|
|
|
const itemsPerRun = lastNodePinData.map((item, index) => {
|
|
|
|
return { json: item, pairedItem: { item: index } };
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
startTime: 0,
|
|
|
|
executionTime: 0,
|
|
|
|
data: { main: [itemsPerRun] },
|
|
|
|
source: lastNodeRunData.source,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return lastNodeRunData;
|
2019-12-19 14:07:55 -08:00
|
|
|
}
|
|
|
|
|
2022-08-03 04:06:53 -07:00
|
|
|
/**
|
|
|
|
* Set node ids if not already set
|
|
|
|
*/
|
|
|
|
export function addNodeIds(workflow: WorkflowEntity) {
|
|
|
|
const { nodes } = workflow;
|
|
|
|
if (!nodes) return;
|
|
|
|
|
|
|
|
nodes.forEach((node) => {
|
|
|
|
if (!node.id) {
|
|
|
|
node.id = uuid();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-10-13 15:21:00 -07:00
|
|
|
// Checking if credentials of old format are in use and run a DB check if they might exist uniquely
|
|
|
|
export async function replaceInvalidCredentials(workflow: WorkflowEntity): Promise<WorkflowEntity> {
|
|
|
|
const { nodes } = workflow;
|
|
|
|
if (!nodes) return workflow;
|
|
|
|
|
|
|
|
// caching
|
|
|
|
const credentialsByName: Record<string, Record<string, INodeCredentialsDetails>> = {};
|
|
|
|
const credentialsById: Record<string, Record<string, INodeCredentialsDetails>> = {};
|
|
|
|
|
|
|
|
// for loop to run DB fetches sequential and use cache to keep pressure off DB
|
|
|
|
// trade-off: longer response time for less DB queries
|
2023-07-31 02:00:48 -07:00
|
|
|
|
2021-10-13 15:21:00 -07:00
|
|
|
for (const node of nodes) {
|
|
|
|
if (!node.credentials || node.disabled) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// extract credentials types
|
|
|
|
const allNodeCredentials = Object.entries(node.credentials);
|
|
|
|
for (const [nodeCredentialType, nodeCredentials] of allNodeCredentials) {
|
|
|
|
// Check if Node applies old credentials style
|
|
|
|
if (typeof nodeCredentials === 'string' || nodeCredentials.id === null) {
|
|
|
|
const name = typeof nodeCredentials === 'string' ? nodeCredentials : nodeCredentials.name;
|
|
|
|
// init cache for type
|
|
|
|
if (!credentialsByName[nodeCredentialType]) {
|
|
|
|
credentialsByName[nodeCredentialType] = {};
|
|
|
|
}
|
|
|
|
if (credentialsByName[nodeCredentialType][name] === undefined) {
|
2023-11-10 06:04:26 -08:00
|
|
|
const credentials = await Container.get(CredentialsRepository).findBy({
|
2021-10-13 15:21:00 -07:00
|
|
|
name,
|
|
|
|
type: nodeCredentialType,
|
|
|
|
});
|
|
|
|
// if credential name-type combination is unique, use it
|
|
|
|
if (credentials?.length === 1) {
|
|
|
|
credentialsByName[nodeCredentialType][name] = {
|
2023-01-02 08:42:32 -08:00
|
|
|
id: credentials[0].id,
|
2021-10-13 15:21:00 -07:00
|
|
|
name: credentials[0].name,
|
|
|
|
};
|
|
|
|
node.credentials[nodeCredentialType] = credentialsByName[nodeCredentialType][name];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// nothing found - add invalid credentials to cache to prevent further DB checks
|
|
|
|
credentialsByName[nodeCredentialType][name] = {
|
|
|
|
id: null,
|
|
|
|
name,
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
// get credentials from cache
|
|
|
|
node.credentials[nodeCredentialType] = credentialsByName[nodeCredentialType][name];
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Node has credentials with an ID
|
|
|
|
|
|
|
|
// init cache for type
|
|
|
|
if (!credentialsById[nodeCredentialType]) {
|
|
|
|
credentialsById[nodeCredentialType] = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if credentials for ID-type are not yet cached
|
|
|
|
if (credentialsById[nodeCredentialType][nodeCredentials.id] === undefined) {
|
|
|
|
// check first if ID-type combination exists
|
2023-11-10 06:04:26 -08:00
|
|
|
const credentials = await Container.get(CredentialsRepository).findOneBy({
|
2021-10-13 15:21:00 -07:00
|
|
|
id: nodeCredentials.id,
|
|
|
|
type: nodeCredentialType,
|
|
|
|
});
|
|
|
|
if (credentials) {
|
|
|
|
credentialsById[nodeCredentialType][nodeCredentials.id] = {
|
2023-01-02 08:42:32 -08:00
|
|
|
id: credentials.id,
|
2021-10-13 15:21:00 -07:00
|
|
|
name: credentials.name,
|
|
|
|
};
|
|
|
|
node.credentials[nodeCredentialType] =
|
|
|
|
credentialsById[nodeCredentialType][nodeCredentials.id];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// no credentials found for ID, check if some exist for name
|
2023-11-10 06:04:26 -08:00
|
|
|
const credsByName = await Container.get(CredentialsRepository).findBy({
|
2021-10-13 15:21:00 -07:00
|
|
|
name: nodeCredentials.name,
|
|
|
|
type: nodeCredentialType,
|
|
|
|
});
|
|
|
|
// if credential name-type combination is unique, take it
|
|
|
|
if (credsByName?.length === 1) {
|
|
|
|
// add found credential to cache
|
|
|
|
credentialsById[nodeCredentialType][credsByName[0].id] = {
|
2023-01-02 08:42:32 -08:00
|
|
|
id: credsByName[0].id,
|
2021-10-13 15:21:00 -07:00
|
|
|
name: credsByName[0].name,
|
|
|
|
};
|
|
|
|
node.credentials[nodeCredentialType] =
|
|
|
|
credentialsById[nodeCredentialType][credsByName[0].id];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// nothing found - add invalid credentials to cache to prevent further DB checks
|
|
|
|
credentialsById[nodeCredentialType][nodeCredentials.id] = nodeCredentials;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get credentials from cache
|
|
|
|
node.credentials[nodeCredentialType] =
|
|
|
|
credentialsById[nodeCredentialType][nodeCredentials.id];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* eslint-enable no-await-in-loop */
|
|
|
|
return workflow;
|
|
|
|
}
|
|
|
|
|
2023-08-10 05:06:16 -07:00
|
|
|
export function getExecutionStartNode(data: IWorkflowExecutionDataProcess, workflow: Workflow) {
|
|
|
|
let startNode;
|
|
|
|
if (
|
|
|
|
data.startNodes?.length === 1 &&
|
2024-02-23 02:43:08 -08:00
|
|
|
Object.keys(data.pinData ?? {}).includes(data.startNodes[0].name)
|
2023-08-10 05:06:16 -07:00
|
|
|
) {
|
2024-02-23 02:43:08 -08:00
|
|
|
startNode = workflow.getNode(data.startNodes[0].name) ?? undefined;
|
2023-08-10 05:06:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return startNode;
|
|
|
|
}
|
|
|
|
|
2023-04-18 03:41:55 -07:00
|
|
|
export async function getVariables(): Promise<IDataObject> {
|
2023-08-02 05:51:09 -07:00
|
|
|
const variables = await Container.get(VariablesService).getAllCached();
|
2023-04-18 03:41:55 -07:00
|
|
|
return Object.freeze(
|
2023-08-02 05:51:09 -07:00
|
|
|
variables.reduce((prev, curr) => {
|
2023-04-18 03:41:55 -07:00
|
|
|
prev[curr.key] = curr.value;
|
|
|
|
return prev;
|
|
|
|
}, {} as IDataObject),
|
|
|
|
);
|
|
|
|
}
|