mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-24 20:24:05 -08:00
feat(Markdown Node): Add new node to covert between Markdown <> HTML (#1728)
* ✨ Markdown Node * Tweaked wording * ⬆️ Bump showdown to latest version * ⚡ Small improvement * 👕 Fix linting issue * ⚡ Small improvements * 🔨 added options, added continue on fail, some clean up * ⚡ removed test code * ⚡ added missing semicolumn * 🔨 wip * 🔨 replaced library for converting html to markdown, added options * ⚡ lock file fix * 🔨 clean up Co-authored-by: sirdavidoff <1670123+sirdavidoff@users.noreply.github.com> Co-authored-by: Michael Kret <michael.k@radency.com>
This commit is contained in:
parent
2485c5fd88
commit
5d1ddb0e9b
608
packages/nodes-base/nodes/Markdown/Markdown.node.ts
Normal file
608
packages/nodes-base/nodes/Markdown/Markdown.node.ts
Normal file
|
@ -0,0 +1,608 @@
|
|||
import { IExecuteFunctions } from 'n8n-core';
|
||||
|
||||
import {
|
||||
IDataObject,
|
||||
INodeExecutionData,
|
||||
INodeType,
|
||||
INodeTypeDescription,
|
||||
JsonObject,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import { Converter } from 'showdown';
|
||||
|
||||
import { NodeHtmlMarkdown } from 'node-html-markdown';
|
||||
|
||||
import { isEmpty, set } from 'lodash';
|
||||
|
||||
export class Markdown implements INodeType {
|
||||
description: INodeTypeDescription = {
|
||||
displayName: 'Markdown',
|
||||
name: 'markdown',
|
||||
icon: 'file:markdown.svg',
|
||||
group: ['output'],
|
||||
version: 1,
|
||||
subtitle: '={{$parameter["mode"]==="markdownToHtml" ? "Markdown to HTML" : "HTML to Markdown"}}',
|
||||
description: 'Convert data between Markdown and HTML',
|
||||
defaults: {
|
||||
name: 'Markdown',
|
||||
color: '#000000',
|
||||
},
|
||||
inputs: ['main'],
|
||||
outputs: ['main'],
|
||||
credentials: [],
|
||||
properties: [
|
||||
{
|
||||
displayName: 'Mode',
|
||||
name: 'mode',
|
||||
type: 'options',
|
||||
options: [
|
||||
{
|
||||
name: 'Markdown to HTML',
|
||||
value: 'markdownToHtml',
|
||||
description: 'Convert data from Markdown to HTML',
|
||||
},
|
||||
{
|
||||
name: 'HTML to Markdown',
|
||||
value: 'htmlToMarkdown',
|
||||
description: 'Convert data from HTML to Markdown',
|
||||
},
|
||||
],
|
||||
default: 'htmlToMarkdown',
|
||||
},
|
||||
{
|
||||
displayName: 'HTML',
|
||||
name: 'html',
|
||||
type: 'string',
|
||||
displayOptions: {
|
||||
show: {
|
||||
mode: ['htmlToMarkdown'],
|
||||
},
|
||||
},
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'The HTML to be converted to markdown',
|
||||
},
|
||||
{
|
||||
displayName: 'Markdown',
|
||||
name: 'markdown',
|
||||
type: 'string',
|
||||
typeOptions: {
|
||||
alwaysOpenEditWindow: true,
|
||||
},
|
||||
displayOptions: {
|
||||
show: {
|
||||
mode: ['markdownToHtml'],
|
||||
},
|
||||
},
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'The Markdown to be converted to html',
|
||||
},
|
||||
{
|
||||
displayName: 'Destination Key',
|
||||
name: 'destinationKey',
|
||||
type: 'string',
|
||||
displayOptions: {
|
||||
show: {
|
||||
mode: ['markdownToHtml', 'htmlToMarkdown'],
|
||||
},
|
||||
},
|
||||
default: 'data',
|
||||
required: true,
|
||||
placeholder: '',
|
||||
description:
|
||||
'The field to put the output in. Specify nested fields using dots, e.g."level1.level2.newKey".',
|
||||
},
|
||||
|
||||
//============= HTML to Markdown Options ===============
|
||||
{
|
||||
displayName: 'Options',
|
||||
name: 'options',
|
||||
type: 'collection',
|
||||
placeholder: 'Add Option',
|
||||
default: {},
|
||||
displayOptions: {
|
||||
show: {
|
||||
mode: ['htmlToMarkdown'],
|
||||
},
|
||||
},
|
||||
options: [
|
||||
{
|
||||
displayName: 'Bullet Marker',
|
||||
name: 'bulletMarker',
|
||||
type: 'string',
|
||||
default: '*',
|
||||
description: 'Specify bullet marker, default *',
|
||||
},
|
||||
{
|
||||
displayName: 'Code Block Fence',
|
||||
name: 'codeFence',
|
||||
type: 'string',
|
||||
default: '```',
|
||||
description: 'Specify code block fence, default ```',
|
||||
},
|
||||
{
|
||||
displayName: 'Emphasis Delimiter',
|
||||
name: 'emDelimiter',
|
||||
type: 'string',
|
||||
default: '_',
|
||||
description: 'Specify emphasis delimiter, default _',
|
||||
},
|
||||
{
|
||||
displayName: 'Global Escape Pattern',
|
||||
name: 'globalEscape',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
description:
|
||||
'Setting this will override the default escape settings, you might want to use textReplace option instead',
|
||||
options: [
|
||||
{
|
||||
name: 'value',
|
||||
displayName: 'Value',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Pattern',
|
||||
name: 'pattern',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'RegEx for pattern',
|
||||
},
|
||||
{
|
||||
displayName: 'Replacement',
|
||||
name: 'replacement',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'String replacement',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Ignored Elements',
|
||||
name: 'ignore',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description:
|
||||
'Supplied elements will be ignored (ignores inner text does not parse children)',
|
||||
placeholder: 'e.g. h1, p ...',
|
||||
hint: 'Comma separated elements',
|
||||
},
|
||||
{
|
||||
displayName: 'Keep Images With Data',
|
||||
name: 'keepDataImages',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to keep images with data: URI (Note: These can be up to 1MB each), e.g. <img src="......0o/">.',
|
||||
},
|
||||
{
|
||||
displayName: 'Line Start Escape Pattern',
|
||||
name: 'lineStartEscape',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
description:
|
||||
'Setting this will override the default escape settings, you might want to use textReplace option instead',
|
||||
options: [
|
||||
{
|
||||
name: 'value',
|
||||
displayName: 'Value',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Pattern',
|
||||
name: 'pattern',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'RegEx for pattern',
|
||||
},
|
||||
{
|
||||
displayName: 'Replacement',
|
||||
name: 'replacement',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'String replacement',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Max Consecutive New Lines',
|
||||
name: 'maxConsecutiveNewlines',
|
||||
type: 'number',
|
||||
default: 3,
|
||||
description: 'Specify max consecutive new lines allowed',
|
||||
},
|
||||
{
|
||||
displayName: 'Place URLs At The Bottom',
|
||||
name: 'useLinkReferenceDefinitions',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to Place URLS at the bottom and format links using link reference definitions',
|
||||
},
|
||||
{
|
||||
displayName: 'Strong Delimiter',
|
||||
name: 'strongDelimiter',
|
||||
type: 'string',
|
||||
default: '**',
|
||||
description: 'Specify strong delimiter, default **',
|
||||
},
|
||||
{
|
||||
displayName: 'Style For Code Block',
|
||||
name: 'codeBlockStyle',
|
||||
type: 'options',
|
||||
default: 'fence',
|
||||
description: 'Specify style for code block, default "fence"',
|
||||
options: [
|
||||
{
|
||||
name: 'Fence',
|
||||
value: 'fence',
|
||||
},
|
||||
{
|
||||
name: 'Indented',
|
||||
value: 'indented',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Text Replacement Pattern',
|
||||
name: 'textReplace',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: true,
|
||||
},
|
||||
default: [],
|
||||
description:
|
||||
'User-defined text replacement pattern (Replaces matching text retrieved from nodes)',
|
||||
options: [
|
||||
{
|
||||
name: 'values',
|
||||
displayName: 'Values',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Pattern',
|
||||
name: 'pattern',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'RegEx for pattern',
|
||||
},
|
||||
{
|
||||
displayName: 'Replacement',
|
||||
name: 'replacement',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'String replacement',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Treat As Blocks',
|
||||
name: 'blockElements',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description:
|
||||
'Supplied elements will be treated as blocks (surrounded with blank lines)',
|
||||
placeholder: 'e.g. p, div, ...',
|
||||
hint: 'Comma separated elements',
|
||||
},
|
||||
],
|
||||
},
|
||||
//============= Markdown to HTML Options ===============
|
||||
{
|
||||
displayName: 'Options',
|
||||
name: 'options',
|
||||
type: 'collection',
|
||||
placeholder: 'Add Option',
|
||||
default: {},
|
||||
displayOptions: {
|
||||
show: {
|
||||
mode: ['markdownToHtml'],
|
||||
},
|
||||
},
|
||||
options: [
|
||||
{
|
||||
displayName: 'Add Blank To Links',
|
||||
name: 'openLinksInNewWindow',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to open all links in new windows (by adding the attribute target="_blank" to <a> tags)',
|
||||
},
|
||||
{
|
||||
displayName: 'Automatic Linking To URLs',
|
||||
name: 'simplifiedAutoLink',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to enable automatic linking to urls',
|
||||
},
|
||||
{
|
||||
displayName: 'Backslash Escapes HTML Tags',
|
||||
name: 'backslashEscapesHTMLTags',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to support for HTML Tag escaping ex: <div>foo</div',
|
||||
},
|
||||
{
|
||||
displayName: 'Complete HTML Document',
|
||||
name: 'completeHTMLDocument',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to output a complete html document, including <html>, <head> and <body> tags instead of an HTML fragment',
|
||||
},
|
||||
{
|
||||
displayName: 'Customized Header ID',
|
||||
name: 'customizedHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to use text in curly braces as header id',
|
||||
},
|
||||
{
|
||||
displayName: 'Emoji Support',
|
||||
name: 'emoji',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to enable emoji support. Ex: this is a :smile: emoji For more info on available emojis, see https://github.com/showdownjs/showdown/wiki/Emojis.',
|
||||
},
|
||||
{
|
||||
displayName: 'Encode Emails',
|
||||
name: 'encodeEmails',
|
||||
type: 'boolean',
|
||||
default: true,
|
||||
description:
|
||||
'Whether to enable e-mail addresses encoding through the use of Character Entities, transforming ASCII e-mail addresses into its equivalent decimal entities',
|
||||
},
|
||||
{
|
||||
displayName: 'Exclude Trailing Punctuation From URLs',
|
||||
name: 'excludeTrailingPunctuationFromURLs',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to exclude trailing punctuation from autolinking urls. Punctuation excluded: . ! ? ( ). Only applies if simplifiedAutoLink option is set to true.',
|
||||
},
|
||||
{
|
||||
displayName: 'GitHub Code Blocks',
|
||||
name: 'ghCodeBlocks',
|
||||
type: 'boolean',
|
||||
default: true,
|
||||
description: 'Whether to enable support for GFM code block style',
|
||||
},
|
||||
{
|
||||
displayName: 'GitHub Compatible Header IDs',
|
||||
name: 'ghCompatibleHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to generate header ids compatible with github style (spaces are replaced with dashes and a bunch of non alphanumeric chars are removed)',
|
||||
},
|
||||
{
|
||||
displayName: 'GitHub Mention Link',
|
||||
name: 'ghMentionsLink',
|
||||
type: 'string',
|
||||
default: 'https://github.com/{u}',
|
||||
description:
|
||||
'Whether to change the link generated by @mentions. Showdown will replace {u} with the username. Only applies if ghMentions option is enabled.',
|
||||
},
|
||||
{
|
||||
displayName: 'GitHub Mentions',
|
||||
name: 'ghMentions',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to enable github @mentions, which link to the username mentioned',
|
||||
},
|
||||
{
|
||||
displayName: 'GitHub Task Lists',
|
||||
name: 'tasklists',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to enable support for GFM tasklists',
|
||||
},
|
||||
{
|
||||
displayName: 'Header Level Start',
|
||||
name: 'headerLevelStart',
|
||||
type: 'number',
|
||||
default: 1,
|
||||
description: 'Whether to set the header starting level',
|
||||
},
|
||||
{
|
||||
displayName: 'Mandatory Space Before Header',
|
||||
name: 'requireSpaceBeforeHeadingText',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to make adding a space between # and the header text mandatory',
|
||||
},
|
||||
{
|
||||
displayName: 'Middle Word Asterisks',
|
||||
name: 'literalMidWordAsterisks',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to stop showdown from interpreting asterisks in the middle of words as <em> and <strong> and instead treat them as literal asterisks',
|
||||
},
|
||||
{
|
||||
displayName: 'Middle Word Underscores',
|
||||
name: 'literalMidWordUnderscores',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to stop showdown from interpreting underscores in the middle of words as <em> and <strong> and instead treat them as literal underscores',
|
||||
},
|
||||
{
|
||||
displayName: 'No Header ID',
|
||||
name: 'noHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to disable the automatic generation of header ids',
|
||||
},
|
||||
{
|
||||
displayName: 'Parse Image Dimensions',
|
||||
name: 'parseImgDimensions',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to enable support for setting image dimensions from within markdown syntax',
|
||||
},
|
||||
{
|
||||
displayName: 'Prefix Header ID',
|
||||
name: 'prefixHeaderId',
|
||||
type: 'string',
|
||||
default: 'section',
|
||||
description: 'Add a prefix to the generated header ids',
|
||||
},
|
||||
{
|
||||
displayName: 'Raw Header ID',
|
||||
name: 'rawHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to remove only spaces, \' and " from generated header ids (including prefixes), replacing them with dashes (-)',
|
||||
},
|
||||
{
|
||||
displayName: 'Raw Prefix Header ID',
|
||||
name: 'rawPrefixHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to prevent showdown from modifying the prefix',
|
||||
},
|
||||
{
|
||||
displayName: 'Simple Line Breaks',
|
||||
name: 'simpleLineBreaks',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to parse line breaks as <br>, like GitHub does, without needing 2 spaces at the end of the line',
|
||||
},
|
||||
{
|
||||
displayName: 'Smart Indentation Fix',
|
||||
name: 'smartIndentationFix',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to try to smartly fix indentation problems related to es6 template strings in the midst of indented code',
|
||||
},
|
||||
{
|
||||
displayName: 'Spaces Indented Sublists',
|
||||
name: 'disableForced4SpacesIndentedSublists',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description:
|
||||
'Whether to disable the requirement of indenting sublists by 4 spaces for them to be nested, effectively reverting to the old behavior where 2 or 3 spaces were enough',
|
||||
},
|
||||
{
|
||||
displayName: 'Split Adjacent Blockquotes',
|
||||
name: 'splitAdjacentBlockquotes',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to split adjacent blockquote blocks',
|
||||
},
|
||||
{
|
||||
displayName: 'Strikethrough',
|
||||
name: 'strikethrough',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to enable support for strikethrough syntax',
|
||||
},
|
||||
{
|
||||
displayName: 'Tables Header ID',
|
||||
name: 'tablesHeaderId',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to add an ID property to table headers tags',
|
||||
},
|
||||
{
|
||||
displayName: 'Tables Support',
|
||||
name: 'tables',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to enable support for tables syntax',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||
const items = this.getInputData();
|
||||
const returnData: IDataObject[] = [];
|
||||
const mode = this.getNodeParameter('mode', 0) as string;
|
||||
|
||||
const { length } = items;
|
||||
for (let i = 0; i < length; i++) {
|
||||
try {
|
||||
if (mode === 'htmlToMarkdown') {
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
const destinationKey = this.getNodeParameter('destinationKey', i) as string;
|
||||
|
||||
const textReplaceOption = this.getNodeParameter('options.textReplace.values', i, []) as IDataObject[];
|
||||
options.textReplace = !isEmpty(textReplaceOption)
|
||||
? textReplaceOption.map((entry) => [entry.pattern, entry.replacement])
|
||||
: undefined;
|
||||
|
||||
const lineStartEscapeOption = this.getNodeParameter('options.lineStartEscape.value', i, {}) as IDataObject;
|
||||
options.lineStartEscape = !isEmpty(lineStartEscapeOption)
|
||||
? [lineStartEscapeOption.pattern, lineStartEscapeOption.replacement]
|
||||
: undefined;
|
||||
|
||||
const globalEscapeOption = this.getNodeParameter('options.globalEscape.value', i, {}) as IDataObject;
|
||||
options.globalEscape = !isEmpty(globalEscapeOption)
|
||||
? [globalEscapeOption.pattern, globalEscapeOption.replacement]
|
||||
: undefined;
|
||||
|
||||
options.ignore = options.ignore
|
||||
? (options.ignore as string).split(',').map(element => element.trim()) : undefined;
|
||||
options.blockElements = options.blockElements
|
||||
? (options.blockElements as string).split(',').map(element => element.trim()) : undefined;
|
||||
|
||||
const markdownOptions = {} as IDataObject;
|
||||
|
||||
Object.keys(options).forEach((option) => {
|
||||
if (options[option]) {
|
||||
markdownOptions[option] = options[option];
|
||||
}
|
||||
});
|
||||
|
||||
const html = this.getNodeParameter('html', i) as string;
|
||||
|
||||
const markdownFromHTML = NodeHtmlMarkdown.translate(html, markdownOptions);
|
||||
|
||||
const newItem = JSON.parse(JSON.stringify(items[i].json));
|
||||
set(newItem, destinationKey, markdownFromHTML);
|
||||
returnData.push(newItem);
|
||||
}
|
||||
|
||||
if (mode === 'markdownToHtml') {
|
||||
const markdown = this.getNodeParameter('markdown', i) as string;
|
||||
const destinationKey = this.getNodeParameter('destinationKey', i) as string;
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
|
||||
const converter = new Converter();
|
||||
|
||||
Object.keys(options).forEach((key) => converter.setOption(key, options[key]));
|
||||
const htmlFromMarkdown = converter.makeHtml(markdown);
|
||||
|
||||
const newItem = JSON.parse(JSON.stringify(items[i].json));
|
||||
set(newItem, destinationKey, htmlFromMarkdown);
|
||||
|
||||
returnData.push(newItem);
|
||||
}
|
||||
} catch (error) {
|
||||
if (this.continueOnFail()) {
|
||||
returnData.push({ error: (error as JsonObject).message });
|
||||
continue;
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
return [this.helpers.returnJsonArray(returnData)];
|
||||
}
|
||||
}
|
6
packages/nodes-base/nodes/Markdown/markdown.svg
Normal file
6
packages/nodes-base/nodes/Markdown/markdown.svg
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no" ?>
|
||||
<svg width="256px" height="158px" viewBox="0 0 256 158" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" preserveAspectRatio="xMidYMid">
|
||||
<g>
|
||||
<path d="M238.371257,157.892216 L18.3952096,157.892216 C8.43113772,157.892216 0,149.461078 0,139.497006 L0,18.3952096 C0,8.43113772 8.43113772,0 18.3952096,0 L237.60479,0 C247.568862,0 256,8.43113772 256,18.3952096 L256,139.497006 C256,149.461078 248.335329,157.892216 238.371257,157.892216 L238.371257,157.892216 Z M18.3952096,12.2634731 C15.3293413,12.2634731 12.2634731,15.3293413 12.2634731,18.3952096 L12.2634731,139.497006 C12.2634731,143.329341 15.3293413,145.628743 18.3952096,145.628743 L237.60479,145.628743 C241.437126,145.628743 243.736527,142.562874 243.736527,139.497006 L243.736527,18.3952096 C243.736527,14.5628743 240.670659,12.2634731 237.60479,12.2634731 C238.371257,12.2634731 18.3952096,12.2634731 18.3952096,12.2634731 L18.3952096,12.2634731 Z M36.7904192,121.101796 L36.7904192,36.7904192 L61.3173653,36.7904192 L85.8443114,67.4491018 L110.371257,36.7904192 L134.898204,36.7904192 L134.898204,121.101796 L110.371257,121.101796 L110.371257,72.8143713 L85.8443114,103.473054 L61.3173653,72.8143713 L61.3173653,121.101796 L36.7904192,121.101796 L36.7904192,121.101796 Z M190.850299,121.101796 L154.05988,80.4790419 L178.586826,80.4790419 L178.586826,36.7904192 L203.113772,36.7904192 L203.113772,79.7125749 L227.640719,79.7125749 L190.850299,121.101796 L190.850299,121.101796 Z" fill="#000000"></path>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 1.6 KiB |
|
@ -515,6 +515,7 @@
|
|||
"dist/nodes/Mailjet/Mailjet.node.js",
|
||||
"dist/nodes/Mailjet/MailjetTrigger.node.js",
|
||||
"dist/nodes/Mandrill/Mandrill.node.js",
|
||||
"dist/nodes/Markdown/Markdown.node.js",
|
||||
"dist/nodes/Marketstack/Marketstack.node.js",
|
||||
"dist/nodes/Matrix/Matrix.node.js",
|
||||
"dist/nodes/Mattermost/Mattermost.node.js",
|
||||
|
@ -707,6 +708,7 @@
|
|||
"@types/nodemailer": "^6.4.0",
|
||||
"@types/redis": "^2.8.11",
|
||||
"@types/request-promise-native": "~1.0.15",
|
||||
"@types/showdown": "^1.9.4",
|
||||
"@types/ssh2-sftp-client": "^5.1.0",
|
||||
"@types/tmp": "^0.2.0",
|
||||
"@types/uuid": "^8.3.2",
|
||||
|
@ -756,6 +758,7 @@
|
|||
"mssql": "^6.2.0",
|
||||
"mysql2": "~2.3.0",
|
||||
"n8n-core": "~0.113.0",
|
||||
"node-html-markdown": "^1.1.3",
|
||||
"node-ssh": "^12.0.0",
|
||||
"nodemailer": "^6.5.0",
|
||||
"pdf-parse": "^1.1.1",
|
||||
|
@ -766,6 +769,7 @@
|
|||
"request": "^2.88.2",
|
||||
"rhea": "^1.0.11",
|
||||
"rss-parser": "^3.7.0",
|
||||
"showdown": "^2.0.3",
|
||||
"simple-git": "^3.5.0",
|
||||
"snowflake-sdk": "^1.5.3",
|
||||
"ssh2-sftp-client": "^7.0.0",
|
||||
|
|
Loading…
Reference in a new issue