Naturally allows to write Flowed specs using natural language instead of JSON.
Define a flow named testFlow that has the following tasks:
A task named firstTask
that requires the following inputs: someInput, someOtherInput
that provides the following outputs: someOutput
using a resolver named testResolver
with the following mapped inputs:
param p1 mapped from someInput
param p2 mapped from someOtherInput
param p3 transformed with "{{someInput.property}}";
param p4 transformed with {
"a1": "{{someInput.property}}",
"a2": "{{someOtherInput.propertyB}}"
};
param p5 with value 123
param p6 with value 456.789
param p7 with value "sample string"
param p8 with a null value
param p9 with a true value
param p10 with a false value
param p11 with value ["a", "b", "c"];
param p12 with value {
"a1": "123",
"a2": true
};
with the following mapped outputs:
param taskResult mapped to someOutput
Once parsed will return the following javascript object:
{
code: 'testFlow',
tasks: {
firstTask: {
requires: [ 'someInput', 'someOtherInput' ],
provides: [ 'someOutput' ],
resolver: {
name: 'testResolver',
params: {
p1: 'someInput',
p2: 'someOtherInput',
p3: { transform: '{{someInput.property}}' },
p4: {
transform: {
a1: '{{someInput.property}}',
a2: '{{someOtherInput.propertyB}}'
}
},
p5: { value: 123 },
p6: { value: 456.789 },
p7: { value: 'sample string' },
p8: { value: null },
p9: { value: true },
p10: { value: false },
p11: { value: [ 'a', 'b', 'c' ] },
p12: { value: { a1: '123', a2: true } }
},
results: { taskResult: 'someOutput' }
}
}
}
}
naturally
follows the following grammar:
Let's see each section in detail.
The flow statement is a string that matches the following RegEx:
/Define a flow named ([._a-zA-Z][.\-_a-zA-Z0-9]+) that has the following tasks:/i
The flow statement is a string that matches the following RegEx:
/A task named ([._a-zA-Z][.\-_a-zA-Z0-9]+)/i
The task requires statement is a string that matches the following RegEx:
/that requires the following inputs: (([._a-zA-Z][.\-_a-zA-Z]+,\s?)*[._a-zA-Z][.\-_a-zA-Z]+)/i
The task provides statement is a string that matches the following RegEx:
/that provides the following outputs: (([._a-zA-Z][.\-_a-zA-Z]+,\s?)*[._a-zA-Z][.\-_a-zA-Z]+)/i
The task resolver statement is a string that matches the following RegEx:
/using a resolver named ([._a-zA-Z][.:\-_a-zA-Z0-9]+)/i
Resolver params are basically two types of statements:
One statement that matches
/with the following mapped inputs:/i
followed by one or more of the statements:
A string that matches the following RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) mapped from ([._a-zA-Z][.\-_a-zA-Z0-9]+)/i
String or number values statements are strings that matches the following RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) with value ((?<string>"(.*)")|(?<number>[\+\-]?\d*\.?\d+(?:[Ee][\+\-]?\d+)?))/i
Object or array values are a special construct that allows to define a JSON object (object or array) as param value. It's a string that matches the following RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) with value ((?<object>\{(.+)\});|(?<array>\[(.*)\]);)/si
Note the s
modifier, that allows to write the object in multiple lines.
Also note that both object and array values should end with };
or ];
respectively.
Null value is a special value and thus it have its own statement: a string that matches the following RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) with a (null|true|false) value/i
Transformations values are strings that matches then follwoing RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) transformed with ((?<object>\{(.+)\});|(?<array>\[(.*)\]);|(?<string>"(.+)");)/si
Note the s
modifier in the second RegEx, that allows to write the object in multiple lines.
Note that the expression support three types of transformations: object, array and string, each one should end with ;
or ];
or ";
respectively.
Resolver results are two types of statements.
One statement that matches the following RegEx:
/with the following mapped outputs:/i
followed by one or more statements matching the folloging RegEx:
/param ([._a-zA-Z][.\-_a-zA-Z0-9]+) mapped to ([._a-zA-Z][.\-_a-zA-Z0-9]+)/i
NOTE: All regex have the i
modifier in order to be case insensitive.
const { NaturallyParser } = require('naturally');
const naturallyFlowedExpression = `
Define a flow named testFlow that has the following tasks:
A task named firstTask
that requires the following inputs: someInput, someOtherInput
that provides the following outputs: someOutput
using a resolver named testResolver
with the following mapped inputs:
param p1 mapped from someInput
param p2 mapped from someOtherInput
param p3 transformed with "{{someInput.property}}";
param p4 transformed with {
"a1": "{{someInput.property}}",
"a2": "{{someOtherInput.propertyB}}"
};
param p5 with value 123
param p6 with value 456.789
param p7 with value "sample string"
param p8 with a null value
param p9 with a true value
param p10 with a false value
param p11 with value ["a", "b", "c"];
param p12 with value {
"a1": "123",
"a2": true
};
with the following mapped outputs:
param taskResult mapped to someOutput
`;
const parser = new NaturallyParser();
const jsonFlow = parser.parse(naturallyFlowedExpression);
console.log(jsonFlow);
npm install naturally