mirror of
https://github.com/n8n-io/n8n.git
synced 2025-02-02 07:01:30 -08:00
* ⚡ Introduce a framework for expression extension * 💡 Add some inline comments * ⚡ Introduce hash alias for encrypt * ⚡ Introduce a manual granular level approach to shadowing/overrideing extensions * 🔥 Cleanup comments * ⚡ Introduce a basic method of extension for native functions * ⚡ Add length to StringExtension * ⚡ Add number type to extension return types * ⚡ Temporarily introduce DateTime with extension * ⚡ Cleanup comments * ⚡ Organize imports * ♻️ Fix up some typings * ⚡ Fix typings * ♻️ Remove unnecessary resolve of expression * ⚡ Extensions Improvement * ♻️ Refactor EXPRESSION_EXTENSION_METHODS * ♻️ Refactor EXPRESSION_EXTENSION_METHODS * ♻️ Update extraArgs types * ♻️ Fix tests * ♻️ Fix bind type issue * ♻️ Fixing duration type issue * ♻️ Refactor to allow overrides on native methods * ♻️ Temporarily remove Date Extensions to pass tests * feat(dt-functions): introduce date expression extensions (#4045) * 🎉 Add Date Extensions into the mix * ✨ Introduce additional date extension methods * ✅ Add Date Expression Extension tests * 🔧 Add ability to debug tests * ♻️ Refactor extension for native types * 🔥 Move sayHi method to String Extension class * ♻️ Update scope when binding member methods * ✅ Add String Extension tests * feat(dt-functions): introduce array expression extensions (#4044) * ✨ Introduce Array Extensions * ✅ Add Array Expression tests * feat(dt-functions): introduce number expression extensions (#4046) * 🎉 Introduce Number Extensions * ⚡ Support more shared extensions * ⚡ Improve handling of name collision * ✅ Update tests * Fixed up tests * 🔥 Remove remove markdown * :recylce: Replace remove-markdown dependencies with implementation * ♻️ Replace remove-markdown dependencies with implementation * ✅ Update tests * ♻️ Fix scoping and cleanup * ♻️ Update comments and errors * ♻️ Fix linting errors * ➖ Remove unused dependencies * fix: expression extension not working with multiple extensions * refactor: change extension transform to be more efficient * test: update most test to work with new extend function * fix: update and fix type error in config * refactor: replace babel with recast * feat: add hashing functions to string extension * fix: removed export * test: add extension parser and transform tests * fix: vite tests breaking * refactor: remove commented out code * fix: parse dates passed from $json in extend function * refactor: review feedback changes for date extensions * refactor: review feedback changes for number extensions * fix: date extension beginningOf test * fix: broken build from merge * fix: another merge issue * refactor: address review feedback (remove ignores) * feat: new extension functions and tests * feat: non-dot notation functions * test: most of the other tests * fix: toSentenceCase for node versions below 16.6 * feat: add $if and $not expression extensions * Fix test to work on every timezone * lint: fix remaining lint issues Co-authored-by: Csaba Tuncsik <csaba@n8n.io> Co-authored-by: Omar Ajoue <krynble@gmail.com>
100 lines
2.2 KiB
TypeScript
100 lines
2.2 KiB
TypeScript
export interface ExpressionText {
|
|
type: 'text';
|
|
text: string;
|
|
}
|
|
|
|
export interface ExpressionCode {
|
|
type: 'code';
|
|
text: string;
|
|
// tmpl has different behaviours if the last expression
|
|
// doesn't close itself.
|
|
hasClosingBrackets: boolean;
|
|
}
|
|
|
|
export type ExpressionChunk = ExpressionCode | ExpressionText;
|
|
|
|
const OPEN_BRACKET = /(?<escape>\\|)(?<brackets>\{\{)/;
|
|
const CLOSE_BRACKET = /(?<escape>\\|)(?<brackets>\}\})/;
|
|
|
|
export const escapeCode = (text: string): string => {
|
|
return text.replace('\\}}', '}}');
|
|
};
|
|
|
|
export const splitExpression = (expression: string): ExpressionChunk[] => {
|
|
const chunks: ExpressionChunk[] = [];
|
|
let searchingFor: 'open' | 'close' = 'open';
|
|
let activeRegex = OPEN_BRACKET;
|
|
|
|
let buffer = '';
|
|
|
|
let index = 0;
|
|
|
|
while (index < expression.length) {
|
|
const expr = expression.slice(index);
|
|
const res = activeRegex.exec(expr);
|
|
// No more brackets. If it's a closing bracket
|
|
// this is sort of valid so we accept it but mark
|
|
// that it has no closing bracket.
|
|
if (!res?.groups) {
|
|
buffer += expr;
|
|
if (searchingFor === 'open') {
|
|
chunks.push({
|
|
type: 'text',
|
|
text: buffer,
|
|
});
|
|
} else {
|
|
chunks.push({
|
|
type: 'code',
|
|
text: escapeCode(buffer),
|
|
hasClosingBrackets: false,
|
|
});
|
|
}
|
|
break;
|
|
}
|
|
if (res.groups.escape) {
|
|
buffer += expr.slice(0, res.index + 3);
|
|
index += res.index + 3;
|
|
} else {
|
|
buffer += expr.slice(0, res.index);
|
|
|
|
if (searchingFor === 'open') {
|
|
chunks.push({
|
|
type: 'text',
|
|
text: buffer,
|
|
});
|
|
searchingFor = 'close';
|
|
activeRegex = CLOSE_BRACKET;
|
|
} else {
|
|
chunks.push({
|
|
type: 'code',
|
|
text: escapeCode(buffer),
|
|
hasClosingBrackets: true,
|
|
});
|
|
searchingFor = 'open';
|
|
activeRegex = OPEN_BRACKET;
|
|
}
|
|
|
|
index += res.index + 2;
|
|
buffer = '';
|
|
}
|
|
}
|
|
|
|
return chunks;
|
|
};
|
|
|
|
// Expressions only have closing brackets escaped
|
|
const escapeTmplExpression = (part: string) => {
|
|
return part.replace('}}', '\\}}');
|
|
};
|
|
|
|
export const joinExpression = (parts: ExpressionChunk[]): string => {
|
|
return parts
|
|
.map((chunk) => {
|
|
if (chunk.type === 'code') {
|
|
return `{{${escapeTmplExpression(chunk.text)}${chunk.hasClosingBrackets ? '}}' : ''}`;
|
|
}
|
|
return chunk.text;
|
|
})
|
|
.join('');
|
|
};
|