mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-03 17:07:29 -08:00
124 lines
3.5 KiB
TypeScript
124 lines
3.5 KiB
TypeScript
import { NodeVM, makeResolverFromLegacyOptions, type Resolver } from '@n8n/vm2';
|
|
import type { IExecuteFunctions, INodeExecutionData } from 'n8n-workflow';
|
|
|
|
import { ValidationError } from './ValidationError';
|
|
import { ExecutionError } from './ExecutionError';
|
|
import type { SandboxContext } from './Sandbox';
|
|
import { Sandbox } from './Sandbox';
|
|
import {
|
|
mapItemNotDefinedErrorIfNeededForRunForEach,
|
|
mapItemsNotDefinedErrorIfNeededForRunForAll,
|
|
validateNoDisallowedMethodsInRunForEach,
|
|
} from './JsCodeValidator';
|
|
|
|
const { NODE_FUNCTION_ALLOW_BUILTIN: builtIn, NODE_FUNCTION_ALLOW_EXTERNAL: external } =
|
|
process.env;
|
|
|
|
export const vmResolver = makeResolverFromLegacyOptions({
|
|
external: external
|
|
? {
|
|
modules: external.split(','),
|
|
transitive: false,
|
|
}
|
|
: false,
|
|
builtin: builtIn?.split(',') ?? [],
|
|
});
|
|
|
|
export class JavaScriptSandbox extends Sandbox {
|
|
private readonly vm: NodeVM;
|
|
|
|
constructor(
|
|
context: SandboxContext,
|
|
private jsCode: string,
|
|
helpers: IExecuteFunctions['helpers'],
|
|
options?: { resolver?: Resolver },
|
|
) {
|
|
super(
|
|
{
|
|
object: {
|
|
singular: 'object',
|
|
plural: 'objects',
|
|
},
|
|
},
|
|
helpers,
|
|
);
|
|
this.vm = new NodeVM({
|
|
console: 'redirect',
|
|
sandbox: context,
|
|
require: options?.resolver ?? vmResolver,
|
|
wasm: false,
|
|
});
|
|
|
|
this.vm.on('console.log', (...args: unknown[]) => this.emit('output', ...args));
|
|
}
|
|
|
|
async runCode<T = unknown>(): Promise<T> {
|
|
const script = `module.exports = async function() {${this.jsCode}\n}()`;
|
|
try {
|
|
const executionResult = (await this.vm.run(script, __dirname)) as T;
|
|
return executionResult;
|
|
} catch (error) {
|
|
throw new ExecutionError(error);
|
|
}
|
|
}
|
|
|
|
async runCodeAllItems(options?: {
|
|
multiOutput?: boolean;
|
|
}): Promise<INodeExecutionData[] | INodeExecutionData[][]> {
|
|
const script = `module.exports = async function() {${this.jsCode}\n}()`;
|
|
|
|
let executionResult: INodeExecutionData | INodeExecutionData[] | INodeExecutionData[][];
|
|
|
|
try {
|
|
executionResult = await this.vm.run(script, __dirname);
|
|
} catch (error) {
|
|
// anticipate user expecting `items` to pre-exist as in Function Item node
|
|
mapItemsNotDefinedErrorIfNeededForRunForAll(this.jsCode, error);
|
|
|
|
throw new ExecutionError(error);
|
|
}
|
|
|
|
if (executionResult === null) return [];
|
|
|
|
if (options?.multiOutput === true) {
|
|
// Check if executionResult is an array of arrays
|
|
if (!Array.isArray(executionResult) || executionResult.some((item) => !Array.isArray(item))) {
|
|
throw new ValidationError({
|
|
message: "The code doesn't return an array of arrays",
|
|
description:
|
|
'Please return an array of arrays. One array for the different outputs and one for the different items that get returned.',
|
|
});
|
|
}
|
|
|
|
return executionResult.map((data) => {
|
|
return this.validateRunCodeAllItems(data);
|
|
});
|
|
}
|
|
|
|
return this.validateRunCodeAllItems(
|
|
executionResult as INodeExecutionData | INodeExecutionData[],
|
|
);
|
|
}
|
|
|
|
async runCodeEachItem(itemIndex: number): Promise<INodeExecutionData | undefined> {
|
|
const script = `module.exports = async function() {${this.jsCode}\n}()`;
|
|
|
|
validateNoDisallowedMethodsInRunForEach(this.jsCode, itemIndex);
|
|
|
|
let executionResult: INodeExecutionData;
|
|
|
|
try {
|
|
executionResult = await this.vm.run(script, __dirname);
|
|
} catch (error) {
|
|
// anticipate user expecting `item` to pre-exist as in Function Item node
|
|
mapItemNotDefinedErrorIfNeededForRunForEach(this.jsCode, error);
|
|
|
|
throw new ExecutionError(error, itemIndex);
|
|
}
|
|
|
|
if (executionResult === null) return undefined;
|
|
|
|
return this.validateRunCodeEachItem(executionResult, itemIndex);
|
|
}
|
|
}
|