mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
Add --output flag to save JSON formated results
This commit is contained in:
parent
fbeb56ab69
commit
c5e1741763
|
@ -29,6 +29,7 @@ import {
|
||||||
diff,
|
diff,
|
||||||
diffString
|
diffString
|
||||||
} from 'json-diff';
|
} from 'json-diff';
|
||||||
|
import { ObjectID } from 'typeorm';
|
||||||
|
|
||||||
export class ExecuteAll extends Command {
|
export class ExecuteAll extends Command {
|
||||||
static description = '\nExecutes all workflows once';
|
static description = '\nExecutes all workflows once';
|
||||||
|
@ -47,6 +48,9 @@ export class ExecuteAll extends Command {
|
||||||
json: flags.boolean({
|
json: flags.boolean({
|
||||||
description: 'Toggles on displaying results in JSON format.',
|
description: 'Toggles on displaying results in JSON format.',
|
||||||
}),
|
}),
|
||||||
|
output: flags.string({
|
||||||
|
description: 'Enable execution saving, You must inform an existing folder to save execution via this param',
|
||||||
|
}),
|
||||||
snapshot: flags.string({
|
snapshot: flags.string({
|
||||||
description: 'Enables snapshot saving. You must inform an existing folder to save snapshots via this param.',
|
description: 'Enables snapshot saving. You must inform an existing folder to save snapshots via this param.',
|
||||||
}),
|
}),
|
||||||
|
@ -85,6 +89,21 @@ export class ExecuteAll extends Command {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (flags.output !== undefined) {
|
||||||
|
if (fs.existsSync(flags.output)) {
|
||||||
|
if (!fs.lstatSync(flags.output).isDirectory()) {
|
||||||
|
GenericHelpers.logOutput(`The paramenter --output must be an existing directory`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
GenericHelpers.logOutput(`The paramenter --output must be an existing directory`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flags.output !== undefined && json!==true) {
|
||||||
|
GenericHelpers.logOutput(`You must inform an json format --json when using --output`);
|
||||||
|
}
|
||||||
// Start directly with the init of the database to improve startup time
|
// Start directly with the init of the database to improve startup time
|
||||||
const startDbInitPromise = Db.init();
|
const startDbInitPromise = Db.init();
|
||||||
|
|
||||||
|
@ -120,6 +139,17 @@ export class ExecuteAll extends Command {
|
||||||
const credentialTypes = CredentialTypes();
|
const credentialTypes = CredentialTypes();
|
||||||
await credentialTypes.init(loadNodesAndCredentials.credentialTypes);
|
await credentialTypes.init(loadNodesAndCredentials.credentialTypes);
|
||||||
|
|
||||||
|
const result:IResult = {
|
||||||
|
workflowsNumber:allWorkflows.length,
|
||||||
|
summary:{
|
||||||
|
failedExecutions:0,
|
||||||
|
succeeedExecution:0,
|
||||||
|
exceptions:0,
|
||||||
|
},
|
||||||
|
nodesCovered:{},
|
||||||
|
executions:[],
|
||||||
|
};
|
||||||
|
|
||||||
// Check if the workflow contains the required "Start" node
|
// Check if the workflow contains the required "Start" node
|
||||||
// "requiredNodeTypes" are also defined in editor-ui/views/NodeView.vue
|
// "requiredNodeTypes" are also defined in editor-ui/views/NodeView.vue
|
||||||
const requiredNodeTypes = ['n8n-nodes-base.start'];
|
const requiredNodeTypes = ['n8n-nodes-base.start'];
|
||||||
|
@ -130,7 +160,7 @@ export class ExecuteAll extends Command {
|
||||||
this.log(`Starting execution of workflow ID ${workflowData.id}.`);
|
this.log(`Starting execution of workflow ID ${workflowData.id}.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const executionResult = {
|
const executionResult:IExecutionResult = {
|
||||||
workflowId: workflowData.id,
|
workflowId: workflowData.id,
|
||||||
executionTime: 0,
|
executionTime: 0,
|
||||||
finished: false,
|
finished: false,
|
||||||
|
@ -156,7 +186,8 @@ export class ExecuteAll extends Command {
|
||||||
// should be executed and with which data to start.
|
// should be executed and with which data to start.
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = 'Workflow cannot be started as it does not contain a "Start" node.';
|
executionResult.error = 'Workflow cannot be started as it does not contain a "Start" node.';
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.failedExecutions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Workflow ID ${workflowData.id} cannot be started as it does not contain a "Start" node.`);
|
GenericHelpers.logOutput(`Workflow ID ${workflowData.id} cannot be started as it does not contain a "Start" node.`);
|
||||||
}
|
}
|
||||||
|
@ -182,20 +213,24 @@ export class ExecuteAll extends Command {
|
||||||
if (data === undefined) {
|
if (data === undefined) {
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = 'Workflow did not return any data.';
|
executionResult.error = 'Workflow did not return any data.';
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.failedExecutions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Workflow ${workflowData.id} did not return any data.`);
|
GenericHelpers.logOutput(`Workflow ${workflowData.id} did not return any data.`);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
workflowData.nodes.forEach(node => {
|
||||||
|
result.nodesCovered[node.type] = (result.nodesCovered[node.type] || 0) +1;
|
||||||
|
});
|
||||||
executionResult.executionTime = (Date.parse(data.stoppedAt as unknown as string) - Date.parse(data.startedAt as unknown as string))/1000;
|
executionResult.executionTime = (Date.parse(data.stoppedAt as unknown as string) - Date.parse(data.startedAt as unknown as string))/1000;
|
||||||
executionResult.finished = (data?.finished !== undefined) as boolean;
|
executionResult.finished = (data?.finished !== undefined) as boolean;
|
||||||
|
|
||||||
if (data.data.resultData.error) {
|
if (data.data.resultData.error) {
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = data.data.resultData.error.message;
|
executionResult.error = data.data.resultData.error.message;
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.failedExecutions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Workflow ${workflowData.id} failed.`);
|
GenericHelpers.logOutput(`Workflow ${workflowData.id} failed.`);
|
||||||
}
|
}
|
||||||
|
@ -209,7 +244,8 @@ export class ExecuteAll extends Command {
|
||||||
const serializedData = JSON.stringify(data, null, 2);
|
const serializedData = JSON.stringify(data, null, 2);
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
if (flags.compare === undefined){
|
if (flags.compare === undefined){
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.succeeedExecution++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Workflow ${workflowData.id} succeeded.`);
|
GenericHelpers.logOutput(`Workflow ${workflowData.id} succeeded.`);
|
||||||
|
@ -228,7 +264,8 @@ export class ExecuteAll extends Command {
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = `Workflow may contain breaking changes`;
|
executionResult.error = `Workflow may contain breaking changes`;
|
||||||
executionResult.changes = changes;
|
executionResult.changes = changes;
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.failedExecutions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
console.log(`Workflow ID ${workflowData.id} may contain breaking changes: `, changes);
|
console.log(`Workflow ID ${workflowData.id} may contain breaking changes: `, changes);
|
||||||
}
|
}
|
||||||
|
@ -238,13 +275,15 @@ export class ExecuteAll extends Command {
|
||||||
}
|
}
|
||||||
}else{
|
}else{
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.succeeedExecution++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = 'Snapshot for not found.';
|
executionResult.error = 'Snapshot for not found.';
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.failedExecutions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Snapshot for ${workflowData.id} not found.`);
|
GenericHelpers.logOutput(`Snapshot for ${workflowData.id} not found.`);
|
||||||
}
|
}
|
||||||
|
@ -260,7 +299,8 @@ export class ExecuteAll extends Command {
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (json === true) {
|
if (json === true) {
|
||||||
executionResult.error = 'Workflow failed to execute.';
|
executionResult.error = 'Workflow failed to execute.';
|
||||||
GenericHelpers.logOutput(JSON.stringify(executionResult, null, 2));
|
result.summary.exceptions++;
|
||||||
|
result.executions.push(executionResult);
|
||||||
}else{
|
}else{
|
||||||
GenericHelpers.logOutput(`Workflow ${workflowData.id} failed to execute.`);
|
GenericHelpers.logOutput(`Workflow ${workflowData.id} failed to execute.`);
|
||||||
}
|
}
|
||||||
|
@ -271,6 +311,42 @@ export class ExecuteAll extends Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
if (json === true){
|
||||||
|
if(flags.output !== undefined){
|
||||||
|
const fileName = (flags.output.endsWith(sep) ? flags.output : flags.output + sep) + `workflows-executions-${Date.now()}.json`;
|
||||||
|
fs.writeFileSync(fileName,JSON.stringify(result, null, 2));
|
||||||
|
}else{
|
||||||
|
GenericHelpers.logOutput(JSON.stringify(result, null, 2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(result.summary.succeeedExecution !== result.workflowsNumber){
|
||||||
|
this.exit(1);
|
||||||
|
}
|
||||||
this.exit(0);
|
this.exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface IResult {
|
||||||
|
workflowsNumber:number;
|
||||||
|
summary:{
|
||||||
|
failedExecutions:number,
|
||||||
|
succeeedExecution:number,
|
||||||
|
exceptions:number,
|
||||||
|
};
|
||||||
|
nodesCovered:{
|
||||||
|
[key:string]:number
|
||||||
|
};
|
||||||
|
executions:IExecutionResult[];
|
||||||
|
}
|
||||||
|
interface IExecutionResult{
|
||||||
|
workflowId: string | number | ObjectID;
|
||||||
|
executionTime: number;
|
||||||
|
finished: boolean;
|
||||||
|
nodes: Array<{
|
||||||
|
name: string,
|
||||||
|
typeVersion: number,
|
||||||
|
type: string,
|
||||||
|
}>;
|
||||||
|
error: string;
|
||||||
|
changes: string;
|
||||||
|
}
|
Loading…
Reference in a new issue