mirror of
				https://github.com/n8n-io/n8n-nodes-starter.git
				synced 2025-10-30 14:52:27 -05:00 
			
		
		
		
	Outgrow Trigger V1
This commit is contained in:
		
					parent
					
						
							
								e863c55657
							
						
					
				
			
			
				commit
				
					
						24f7ae4a19
					
				
			
		
					 14 changed files with 564 additions and 585 deletions
				
			
		
							
								
								
									
										79
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										79
									
								
								README.md
									
										
									
									
									
								
							|  | @ -1,48 +1,57 @@ | |||
|  | ||||
| # Outgrow Trigger Node for n8n | ||||
| 
 | ||||
| # n8n-nodes-starter | ||||
| [](https://n8n.io/integrations) | ||||
| 
 | ||||
| This repo contains example nodes to help you get started building your own custom integrations for [n8n](https://n8n.io). It includes the node linter and other dependencies. | ||||
| Trigger node that fetches leads from Outgrow calculators at configurable intervals. | ||||
| 
 | ||||
| To make your custom node available to the community, you must create it as an npm package, and [submit it to the npm registry](https://docs.npmjs.com/packages-and-modules/contributing-packages-to-the-registry). | ||||
|  *(Optional: Add screenshot later)* | ||||
| 
 | ||||
| If you would like your node to be available on n8n cloud you can also [submit your node for verification](https://docs.n8n.io/integrations/creating-nodes/deploy/submit-community-nodes/). | ||||
| ## Features | ||||
| 
 | ||||
| - 🕒 **Polling API**: Checks for new leads at customizable intervals (default: 5 minutes) | ||||
| - 📋 **Calculator Selection**: Dynamic dropdown of available calculators | ||||
| - 🔔 **Smart Notifications**: Clear "no leads" messages in the UI | ||||
| - ⚙️ **Manual Trigger**: Instant API calls for testing | ||||
| 
 | ||||
| ## Prerequisites | ||||
| 
 | ||||
| You need the following installed on your development machine: | ||||
| - Outgrow account with API access | ||||
| - API key from [Outgrow Settings](https://app.outgrow.co/settings/api) | ||||
| 
 | ||||
| * [git](https://git-scm.com/downloads) | ||||
| * Node.js and npm. Minimum version Node 20. You can find instructions on how to install both using nvm (Node Version Manager) for Linux, Mac, and WSL [here](https://github.com/nvm-sh/nvm). For Windows users, refer to Microsoft's guide to [Install NodeJS on Windows](https://docs.microsoft.com/en-us/windows/dev-environment/javascript/nodejs-on-windows). | ||||
| * Install n8n with: | ||||
|   ``` | ||||
|   npm install n8n -g | ||||
|   ``` | ||||
| * Recommended: follow n8n's guide to [set up your development environment](https://docs.n8n.io/integrations/creating-nodes/build/node-development-environment/). | ||||
| ## Installation | ||||
| 
 | ||||
| ## Using this starter | ||||
| 1. Add the node to your n8n instance | ||||
| 2. Configure Outgrow API credentials: | ||||
|    - Navigate to **Credentials > Add New > Outgrow API** | ||||
|    - Enter your API key | ||||
| 
 | ||||
| These are the basic steps for working with the starter. For detailed guidance on creating and publishing nodes, refer to the [documentation](https://docs.n8n.io/integrations/creating-nodes/). | ||||
| ## Node Configuration | ||||
| 
 | ||||
| 1. [Generate a new repository](https://github.com/n8n-io/n8n-nodes-starter/generate) from this template repository. | ||||
| 2. Clone your new repo: | ||||
|    ``` | ||||
|    git clone https://github.com/<your organization>/<your-repo-name>.git | ||||
|    ``` | ||||
| 3. Run `npm i` to install dependencies. | ||||
| 4. Open the project in your editor. | ||||
| 5. Browse the examples in `/nodes` and `/credentials`. Modify the examples, or replace them with your own nodes. | ||||
| 6. Update the `package.json` to match your details. | ||||
| 7. Run `npm run lint` to check for errors or `npm run lintfix` to automatically fix errors when possible. | ||||
| 8. Test your node locally. Refer to [Run your node locally](https://docs.n8n.io/integrations/creating-nodes/test/run-node-locally/) for guidance. | ||||
| 9. Replace this README with documentation for your node. Use the [README_TEMPLATE](README_TEMPLATE.md) to get started. | ||||
| 10. Update the LICENSE file to use your details. | ||||
| 11. [Publish](https://docs.npmjs.com/packages-and-modules/contributing-packages-to-the-registry) your package to npm. | ||||
| ### Parameters | ||||
| | Field               | Required | Description                                                                 | | ||||
| |---------------------|----------|-----------------------------------------------------------------------------| | ||||
| | **Calculator**      | Yes      | Select which Outgrow calculator to monitor                                  | | ||||
| | **Polling Interval**| No       | Frequency (in minutes) to check for new leads (default: 5)                  | | ||||
| 
 | ||||
| ## More information | ||||
| ## Usage Examples | ||||
| 
 | ||||
| Refer to our [documentation on creating nodes](https://docs.n8n.io/integrations/creating-nodes/) for detailed information on building your own nodes. | ||||
| 
 | ||||
| ## License | ||||
| 
 | ||||
| [MIT](https://github.com/n8n-io/n8n-nodes-starter/blob/master/LICENSE.md) | ||||
| ### Basic Automation | ||||
| ```json | ||||
| { | ||||
|   "workflow": { | ||||
|     "nodes": [ | ||||
|       { | ||||
|         "parameters": { | ||||
|           "calcId": "{{CALCULATOR_ID}}", | ||||
|           "pollingInterval": 10 | ||||
|         }, | ||||
|         "name": "Outgrow Trigger", | ||||
|         "type": "outgrowTrigger", | ||||
|         "typeVersion": 1 | ||||
|       }, | ||||
|       { | ||||
|         // Connect to other nodes (Email, Slack, etc.) | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| } | ||||
|  | @ -1,59 +0,0 @@ | |||
| import { | ||||
| 	IAuthenticateGeneric, | ||||
| 	ICredentialTestRequest, | ||||
| 	ICredentialType, | ||||
| 	INodeProperties, | ||||
| } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class ExampleCredentialsApi implements ICredentialType { | ||||
| 	name = 'exampleCredentialsApi'; | ||||
| 	displayName = 'Example Credentials API'; | ||||
| 
 | ||||
| 	documentationUrl = 'https://your-docs-url'; | ||||
| 
 | ||||
| 	properties: INodeProperties[] = [ | ||||
| 		// The credentials to get from user and save encrypted.
 | ||||
| 		// Properties can be defined exactly in the same way
 | ||||
| 		// as node properties.
 | ||||
| 		{ | ||||
| 			displayName: 'User Name', | ||||
| 			name: 'username', | ||||
| 			type: 'string', | ||||
| 			default: '', | ||||
| 		}, | ||||
| 		{ | ||||
| 			displayName: 'Password', | ||||
| 			name: 'password', | ||||
| 			type: 'string', | ||||
| 			typeOptions: { | ||||
| 				password: true, | ||||
| 			}, | ||||
| 			default: '', | ||||
| 		}, | ||||
| 	]; | ||||
| 
 | ||||
| 	// This credential is currently not used by any node directly
 | ||||
| 	// but the HTTP Request node can use it to make requests.
 | ||||
| 	// The credential is also testable due to the `test` property below
 | ||||
| 	authenticate: IAuthenticateGeneric = { | ||||
| 		type: 'generic', | ||||
| 		properties: { | ||||
| 			auth: { | ||||
| 				username: '={{ $credentials.username }}', | ||||
| 				password: '={{ $credentials.password }}', | ||||
| 			}, | ||||
| 			qs: { | ||||
| 				// Send this as part of the query string
 | ||||
| 				n8n: 'rocks', | ||||
| 			}, | ||||
| 		}, | ||||
| 	}; | ||||
| 
 | ||||
| 	// The block below tells how this credential can be tested
 | ||||
| 	test: ICredentialTestRequest = { | ||||
| 		request: { | ||||
| 			baseURL: 'https://example.com/', | ||||
| 			url: '', | ||||
| 		}, | ||||
| 	}; | ||||
| } | ||||
|  | @ -1,50 +0,0 @@ | |||
| import { | ||||
| 	IAuthenticateGeneric, | ||||
| 	ICredentialTestRequest, | ||||
| 	ICredentialType, | ||||
| 	INodeProperties, | ||||
| } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class HttpBinApi implements ICredentialType { | ||||
| 	name = 'httpbinApi'; | ||||
| 	displayName = 'HttpBin API'; | ||||
| 	documentationUrl = 'https://your-docs-url'; | ||||
| 	properties: INodeProperties[] = [ | ||||
| 		{ | ||||
| 			displayName: 'Token', | ||||
| 			name: 'token', | ||||
| 			type: 'string', | ||||
| 			default: '', | ||||
| 			typeOptions: { | ||||
| 				password: true, | ||||
| 			} | ||||
| 		}, | ||||
| 		{ | ||||
| 			displayName: 'Domain', | ||||
| 			name: 'domain', | ||||
| 			type: 'string', | ||||
| 			default: 'https://httpbin.org', | ||||
| 		}, | ||||
| 	]; | ||||
| 
 | ||||
| 	// This allows the credential to be used by other parts of n8n
 | ||||
| 	// stating how this credential is injected as part of the request
 | ||||
| 	// An example is the Http Request node that can make generic calls
 | ||||
| 	// reusing this credential
 | ||||
| 	authenticate: IAuthenticateGeneric = { | ||||
| 		type: 'generic', | ||||
| 		properties: { | ||||
| 			headers: { | ||||
| 				Authorization: '={{"Bearer " + $credentials.token}}', | ||||
| 			}, | ||||
| 		}, | ||||
| 	}; | ||||
| 
 | ||||
| 	// The block below tells how this credential can be tested
 | ||||
| 	test: ICredentialTestRequest = { | ||||
| 		request: { | ||||
| 			baseURL: '={{$credentials?.domain}}', | ||||
| 			url: '/bearer', | ||||
| 		}, | ||||
| 	}; | ||||
| } | ||||
							
								
								
									
										22
									
								
								credentials/outgrowApi.credentials.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								credentials/outgrowApi.credentials.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | |||
| import { | ||||
| 	ICredentialType, | ||||
| 	INodeProperties, | ||||
| } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class outgrowApi implements ICredentialType { | ||||
| 	name = 'outgrowApi'; // 👈 Must match what you use in the node
 | ||||
| 	displayName = 'Outgrow API'; | ||||
| 	documentationUrl = 'https://docs.n8n.io/integrations/creating-nodes/build/credentials/'; | ||||
| 	properties: INodeProperties[] = [ | ||||
| 		{ | ||||
| 			displayName: 'API Key', | ||||
| 			name: 'apiKey', | ||||
| 			type: 'string', | ||||
| 			default: '', | ||||
| 			required: true, | ||||
| 			description: 'Your Outgrow API key', | ||||
| 		}, | ||||
| 	]; | ||||
| 
 | ||||
| 	// No built-in auth here since we inject key manually in node code
 | ||||
| } | ||||
|  | @ -1,77 +0,0 @@ | |||
| import type { | ||||
| 	IExecuteFunctions, | ||||
| 	INodeExecutionData, | ||||
| 	INodeType, | ||||
| 	INodeTypeDescription, | ||||
| } from 'n8n-workflow'; | ||||
| import { NodeConnectionType, NodeOperationError } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class ExampleNode implements INodeType { | ||||
| 	description: INodeTypeDescription = { | ||||
| 		displayName: 'Example Node', | ||||
| 		name: 'exampleNode', | ||||
| 		group: ['transform'], | ||||
| 		version: 1, | ||||
| 		description: 'Basic Example Node', | ||||
| 		defaults: { | ||||
| 			name: 'Example Node', | ||||
| 		}, | ||||
| 		inputs: [NodeConnectionType.Main], | ||||
| 		outputs: [NodeConnectionType.Main], | ||||
| 		usableAsTool: true, | ||||
| 		properties: [ | ||||
| 			// Node properties which the user gets displayed and
 | ||||
| 			// can change on the node.
 | ||||
| 			{ | ||||
| 				displayName: 'My String', | ||||
| 				name: 'myString', | ||||
| 				type: 'string', | ||||
| 				default: '', | ||||
| 				placeholder: 'Placeholder value', | ||||
| 				description: 'The description text', | ||||
| 			}, | ||||
| 		], | ||||
| 	}; | ||||
| 
 | ||||
| 	// The function below is responsible for actually doing whatever this node
 | ||||
| 	// is supposed to do. In this case, we're just appending the `myString` property
 | ||||
| 	// with whatever the user has entered.
 | ||||
| 	// You can make async calls and use `await`.
 | ||||
| 	async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> { | ||||
| 		const items = this.getInputData(); | ||||
| 
 | ||||
| 		let item: INodeExecutionData; | ||||
| 		let myString: string; | ||||
| 
 | ||||
| 		// Iterates over all input items and add the key "myString" with the
 | ||||
| 		// value the parameter "myString" resolves to.
 | ||||
| 		// (This could be a different value for each item in case it contains an expression)
 | ||||
| 		for (let itemIndex = 0; itemIndex < items.length; itemIndex++) { | ||||
| 			try { | ||||
| 				myString = this.getNodeParameter('myString', itemIndex, '') as string; | ||||
| 				item = items[itemIndex]; | ||||
| 
 | ||||
| 				item.json.myString = myString; | ||||
| 			} catch (error) { | ||||
| 				// This node should never fail but we want to showcase how
 | ||||
| 				// to handle errors.
 | ||||
| 				if (this.continueOnFail()) { | ||||
| 					items.push({ json: this.getInputData(itemIndex)[0].json, error, pairedItem: itemIndex }); | ||||
| 				} else { | ||||
| 					// Adding `itemIndex` allows other workflows to handle this error
 | ||||
| 					if (error.context) { | ||||
| 						// If the error thrown already contains the context property,
 | ||||
| 						// only append the itemIndex
 | ||||
| 						error.context.itemIndex = itemIndex; | ||||
| 						throw error; | ||||
| 					} | ||||
| 					throw new NodeOperationError(this.getNode(), error, { | ||||
| 						itemIndex, | ||||
| 					}); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		return [items]; | ||||
| 	} | ||||
| } | ||||
|  | @ -1,18 +0,0 @@ | |||
| { | ||||
| 	"node": "n8n-nodes-base.httpbin", | ||||
| 	"nodeVersion": "1.0", | ||||
| 	"codexVersion": "1.0", | ||||
| 	"categories": ["Development", "Developer Tools"], | ||||
| 	"resources": { | ||||
| 		"credentialDocumentation": [ | ||||
| 			{ | ||||
| 				"url": "http://httpbin.org/#/Auth/get_bearer" | ||||
| 			} | ||||
| 		], | ||||
| 		"primaryDocumentation": [ | ||||
| 			{ | ||||
| 				"url": "http://httpbin.org/" | ||||
| 			} | ||||
| 		] | ||||
| 	} | ||||
| } | ||||
|  | @ -1,63 +0,0 @@ | |||
| import { INodeType, INodeTypeDescription, NodeConnectionType } from 'n8n-workflow'; | ||||
| import { httpVerbFields, httpVerbOperations } from './HttpVerbDescription'; | ||||
| 
 | ||||
| export class HttpBin implements INodeType { | ||||
| 	description: INodeTypeDescription = { | ||||
| 		displayName: 'HttpBin', | ||||
| 		name: 'httpBin', | ||||
| 		icon: { light: 'file:httpbin.svg', dark: 'file:httpbin.svg' }, | ||||
| 		group: ['transform'], | ||||
| 		version: 1, | ||||
| 		subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}', | ||||
| 		description: 'Interact with HttpBin API', | ||||
| 		defaults: { | ||||
| 			name: 'HttpBin', | ||||
| 		}, | ||||
| 		inputs: [NodeConnectionType.Main], | ||||
| 		outputs: [NodeConnectionType.Main], | ||||
| 		usableAsTool: true, | ||||
| 		credentials: [ | ||||
| 			{ | ||||
| 				name: 'httpbinApi', | ||||
| 				required: false, | ||||
| 			}, | ||||
| 		], | ||||
| 		requestDefaults: { | ||||
| 			baseURL: 'https://httpbin.org', | ||||
| 			url: '', | ||||
| 			headers: { | ||||
| 				Accept: 'application/json', | ||||
| 				'Content-Type': 'application/json', | ||||
| 			}, | ||||
| 		}, | ||||
| 		/** | ||||
| 		 * In the properties array we have two mandatory options objects required | ||||
| 		 * | ||||
| 		 * [Resource & Operation] | ||||
| 		 * | ||||
| 		 * https://docs.n8n.io/integrations/creating-nodes/code/create-first-node/#resources-and-operations
 | ||||
| 		 * | ||||
| 		 * In our example, the operations are separated into their own file (HTTPVerbDescription.ts) | ||||
| 		 * to keep this class easy to read. | ||||
| 		 * | ||||
| 		 */ | ||||
| 		properties: [ | ||||
| 			{ | ||||
| 				displayName: 'Resource', | ||||
| 				name: 'resource', | ||||
| 				type: 'options', | ||||
| 				noDataExpression: true, | ||||
| 				options: [ | ||||
| 					{ | ||||
| 						name: 'HTTP Verb', | ||||
| 						value: 'httpVerb', | ||||
| 					}, | ||||
| 				], | ||||
| 				default: 'httpVerb', | ||||
| 			}, | ||||
| 
 | ||||
| 			...httpVerbOperations, | ||||
| 			...httpVerbFields, | ||||
| 		], | ||||
| 	}; | ||||
| } | ||||
|  | @ -1,250 +0,0 @@ | |||
| import { INodeProperties } from 'n8n-workflow'; | ||||
| 
 | ||||
| // When the resource `httpVerb` is selected, this `operation` parameter will be shown.
 | ||||
| export const httpVerbOperations: INodeProperties[] = [ | ||||
| 	{ | ||||
| 		displayName: 'Operation', | ||||
| 		name: 'operation', | ||||
| 		type: 'options', | ||||
| 		noDataExpression: true, | ||||
| 
 | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'GET', | ||||
| 				value: 'get', | ||||
| 				description: 'Perform a GET request', | ||||
| 				action: 'Perform a GET request', | ||||
| 				routing: { | ||||
| 					request: { | ||||
| 						method: 'GET', | ||||
| 						url: '/get', | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			{ | ||||
| 				name: 'DELETE', | ||||
| 				value: 'delete', | ||||
| 				description: 'Perform a DELETE request', | ||||
| 				action: 'Perform a DELETE request', | ||||
| 				routing: { | ||||
| 					request: { | ||||
| 						method: 'DELETE', | ||||
| 						url: '/delete', | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		], | ||||
| 		default: 'get', | ||||
| 	}, | ||||
| ]; | ||||
| 
 | ||||
| // Here we define what to show when the `get` operation is selected.
 | ||||
| // We do that by adding `operation: ["get"]` to `displayOptions.show`
 | ||||
| const getOperation: INodeProperties[] = [ | ||||
| 	{ | ||||
| 		displayName: 'Type of Data', | ||||
| 		name: 'typeofData', | ||||
| 		default: 'queryParameter', | ||||
| 		description: 'Select type of data to send [Query Parameters]', | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 				operation: ['get'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		type: 'options', | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'Query', | ||||
| 				value: 'queryParameter', | ||||
| 			}, | ||||
| 		], | ||||
| 		required: true, | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Query Parameters', | ||||
| 		name: 'arguments', | ||||
| 		default: {}, | ||||
| 		description: "The request's query parameters", | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 				operation: ['get'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'keyvalue', | ||||
| 				displayName: 'Key:Value', | ||||
| 				values: [ | ||||
| 					{ | ||||
| 						displayName: 'Key', | ||||
| 						name: 'key', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						required: true, | ||||
| 						description: 'Key of query parameter', | ||||
| 					}, | ||||
| 					{ | ||||
| 						displayName: 'Value', | ||||
| 						name: 'value', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						routing: { | ||||
| 							send: { | ||||
| 								property: '={{$parent.key}}', | ||||
| 								type: 'query', | ||||
| 							}, | ||||
| 						}, | ||||
| 						required: true, | ||||
| 						description: 'Value of query parameter', | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 		], | ||||
| 		type: 'fixedCollection', | ||||
| 		typeOptions: { | ||||
| 			multipleValues: true, | ||||
| 		}, | ||||
| 	}, | ||||
| ]; | ||||
| 
 | ||||
| // Here we define what to show when the DELETE Operation is selected.
 | ||||
| // We do that by adding `operation: ["delete"]` to `displayOptions.show`
 | ||||
| const deleteOperation: INodeProperties[] = [ | ||||
| 	{ | ||||
| 		displayName: 'Type of Data', | ||||
| 		name: 'typeofData', | ||||
| 		default: 'queryParameter', | ||||
| 		description: 'Select type of data to send [Query Parameter Arguments, JSON-Body]', | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 				operation: ['delete'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'Query', | ||||
| 				value: 'queryParameter', | ||||
| 			}, | ||||
| 			{ | ||||
| 				name: 'JSON', | ||||
| 				value: 'jsonData', | ||||
| 			}, | ||||
| 		], | ||||
| 		required: true, | ||||
| 		type: 'options', | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Query Parameters', | ||||
| 		name: 'arguments', | ||||
| 		default: {}, | ||||
| 		description: "The request's query parameters", | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 				operation: ['delete'], | ||||
| 				typeofData: ['queryParameter'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'keyvalue', | ||||
| 				displayName: 'Key:Value', | ||||
| 				values: [ | ||||
| 					{ | ||||
| 						displayName: 'Key', | ||||
| 						name: 'key', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						required: true, | ||||
| 						description: 'Key of query parameter', | ||||
| 					}, | ||||
| 					{ | ||||
| 						displayName: 'Value', | ||||
| 						name: 'value', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						routing: { | ||||
| 							send: { | ||||
| 								property: '={{$parent.key}}', | ||||
| 								type: 'query', | ||||
| 							}, | ||||
| 						}, | ||||
| 						required: true, | ||||
| 						description: 'Value of query parameter', | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 		], | ||||
| 		type: 'fixedCollection', | ||||
| 		typeOptions: { | ||||
| 			multipleValues: true, | ||||
| 		}, | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'JSON Object', | ||||
| 		name: 'arguments', | ||||
| 		default: {}, | ||||
| 		description: "The request's JSON properties", | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: ['httpVerb'], | ||||
| 				operation: ['delete'], | ||||
| 				typeofData: ['jsonData'], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				name: 'keyvalue', | ||||
| 				displayName: 'Key:Value', | ||||
| 				values: [ | ||||
| 					{ | ||||
| 						displayName: 'Key', | ||||
| 						name: 'key', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						required: true, | ||||
| 						description: 'Key of JSON property', | ||||
| 					}, | ||||
| 					{ | ||||
| 						displayName: 'Value', | ||||
| 						name: 'value', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						routing: { | ||||
| 							send: { | ||||
| 								property: '={{$parent.key}}', | ||||
| 								type: 'body', | ||||
| 							}, | ||||
| 						}, | ||||
| 						required: true, | ||||
| 						description: 'Value of JSON property', | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 		], | ||||
| 		type: 'fixedCollection', | ||||
| 		typeOptions: { | ||||
| 			multipleValues: true, | ||||
| 		}, | ||||
| 	}, | ||||
| ]; | ||||
| 
 | ||||
| export const httpVerbFields: INodeProperties[] = [ | ||||
| 	/* -------------------------------------------------------------------------- */ | ||||
| 	/*                                httpVerb:get                                */ | ||||
| 	/* -------------------------------------------------------------------------- */ | ||||
| 	...getOperation, | ||||
| 
 | ||||
| 	/* -------------------------------------------------------------------------- */ | ||||
| 	/*                              httpVerb:delete                               */ | ||||
| 	/* -------------------------------------------------------------------------- */ | ||||
| 	...deleteOperation, | ||||
| ]; | ||||
|  | @ -1,18 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" width="32px" height="32px" viewBox="0 0 32 32" enable-background="new 0 0 32 32" xml:space="preserve">  <image id="image0" width="32" height="32" x="0" y="0" | ||||
|     href=" | ||||
| AAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAAAmJLR0QA/4ePzL8AAAAHdElN | ||||
| RQfmBg4UAC/TqOZZAAACA0lEQVRIx5XVv09TURwF8M+jFHDSyRkGFhPAEfyRdDHi5uriXyDoYgKT | ||||
| MJDWzUT/Ahf/AiOEpajEgCESmpiYmDCxGowDTYE+h76+vte+15Zzk753b7733HNO772PbEw7ECba | ||||
| genswtEcgl0/PHARV72066YrIDSZ6k8KBym4741r0XsB284TdUX8chn1zrzwJUmw4KFXPqjFE0Y0 | ||||
| u5YKEhpmfLZuy7f2wLKGI8WhDRYdaVhurdTCidmU5P44N+skaaGQH1IfFFrOYMotT932zNgQExve | ||||
| OfTeT8dtBceO3TFlOyopY7UPxV+/fWyn3Y0xrFhJjZWFXhs12pKdRO9ObGSuyB8Xbd9JjMjDc6HQ | ||||
| IcrKqAiVe8vyCEJPrGBWxZYqqtZt9RbmHabAvAAVdVUlJTvWshbMt0AYn40OmlchSKOePTyYIMQn | ||||
| rb8yI8TsDCrRs4od7Jv3KOoPGWKboBqp2LN3FQvdO7EPshSsRSTXrSop2cSiiUGkG/bj2JqaQiHW | ||||
| 4nv50mFcu28j30KQarAnEPhuzvwwGYQ975vx7+JwGXTjTIAzoYlhCArR5d0KkfauqJAVY6+FG5hD | ||||
| OS6veqyCuSiTAQT/jKmlQtyxIBCoZV28HQvN6LuQvJFC4xjvibfYOZUdUXd9taTWJbOubiIVXmjG | ||||
| W/fs9qpZcpr6pOe1U0udSf8BR7ef4yxyOskAAAAldEVYdGRhdGU6Y3JlYXRlADIwMjItMDYtMTRU | ||||
| MTc6MDA6NDcrMDM6MDBfo1sRAAAAJXRFWHRkYXRlOm1vZGlmeQAyMDIyLTA2LTE0VDE3OjAwOjQ3 | ||||
| KzAzOjAwLv7jrQAAAABJRU5ErkJggg==" /> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 1.5 KiB | 
							
								
								
									
										20
									
								
								nodes/outgrow/outgrow.node.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								nodes/outgrow/outgrow.node.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,20 @@ | |||
| { | ||||
| 	"node": "n8n-nodes-base.outgrow", | ||||
| 	"nodeVersion": "1.0", | ||||
| 	"codexVersion": "1.0", | ||||
| 	"categories": [ | ||||
| 		"Miscellaneous" | ||||
| 	], | ||||
| 	"resources": { | ||||
| 		"credentialDocumentation": [ | ||||
| 			{ | ||||
| 				"url": "" | ||||
| 			} | ||||
| 		], | ||||
| 		"primaryDocumentation": [ | ||||
| 			{ | ||||
| 				"url": "" | ||||
| 			} | ||||
| 		] | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										145
									
								
								nodes/outgrow/outgrow.node.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								nodes/outgrow/outgrow.node.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,145 @@ | |||
| import { | ||||
| 	ITriggerFunctions, | ||||
| 	INodeType, | ||||
| 	INodeTypeDescription, | ||||
| 	ILoadOptionsFunctions, | ||||
| 	INodePropertyOptions, | ||||
| 	ITriggerResponse, | ||||
| 	NodeApiError, | ||||
| 	NodeConnectionType, | ||||
| } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class outgrow implements INodeType { | ||||
| 	description: INodeTypeDescription = { | ||||
| 		displayName: 'Outgrow Trigger', | ||||
| 		name: 'outgrowTrigger', | ||||
| 		icon: 'file:outgrow.svg', | ||||
| 		group: ['trigger'], | ||||
| 		version: 1, | ||||
| 		subtitle: '={{$parameter["calcId"]}}', | ||||
| 		description: 'Fetch leads from Outgrow calculators at regular intervals', | ||||
| 		defaults: { | ||||
| 			name: 'Outgrow Trigger', | ||||
| 		}, | ||||
| 		inputs: [], | ||||
| 		outputs: [NodeConnectionType.Main], | ||||
| 		credentials: [ | ||||
| 			{ | ||||
| 				name: 'outgrowApi', | ||||
| 				required: true, | ||||
| 			}, | ||||
| 		], | ||||
| 		properties: [ | ||||
| 			{ | ||||
| 				displayName: 'Calculator', | ||||
| 				name: 'calcId', | ||||
| 				type: 'options', | ||||
| 				required: true, | ||||
| 				typeOptions: { | ||||
| 					loadOptionsMethod: 'getCalculators', | ||||
| 				}, | ||||
| 				default: '', | ||||
| 				description: 'Which Outgrow calculator to fetch leads from', | ||||
| 			}, | ||||
| 			{ | ||||
| 				displayName: 'Polling Interval (Minutes)', | ||||
| 				name: 'pollingInterval', | ||||
| 				type: 'number', | ||||
| 				required: false, | ||||
| 				default: 5, | ||||
| 				description: 'How often (in minutes) to check for new leads', | ||||
| 			}, | ||||
| 		], | ||||
| 	}; | ||||
| 
 | ||||
| 	methods = { | ||||
| 		loadOptions: { | ||||
| 			async getCalculators(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { | ||||
| 				const credentials = await this.getCredentials('outgrowApi'); | ||||
| 				if (!credentials?.apiKey) { | ||||
| 					throw new NodeApiError(this.getNode(), { | ||||
| 						message: 'API Key is missing or invalid', | ||||
| 					}); | ||||
| 				} | ||||
| 
 | ||||
| 				const url = `https://api-calc.outgrow.co/api/v1/get_cal/${credentials.apiKey}`; | ||||
| 				try { | ||||
| 					const response = await this.helpers.request({ method: 'GET', url }); | ||||
| 					const calculators = JSON.parse(response); | ||||
| 
 | ||||
| 					return calculators.map((calc: { id: string; calculator: string }) => ({ | ||||
| 						name: calc.calculator, | ||||
| 						value: calc.id, | ||||
| 					})); | ||||
| 				} catch (error) { | ||||
| 					throw new NodeApiError(this.getNode(), error, { | ||||
| 						message: 'Failed to load calculators', | ||||
| 					}); | ||||
| 				} | ||||
| 			}, | ||||
| 		}, | ||||
| 	}; | ||||
| 
 | ||||
| 	async trigger(this: ITriggerFunctions): Promise<ITriggerResponse> { | ||||
| 		const credentials = await this.getCredentials('outgrowApi'); | ||||
| 		if (!credentials?.apiKey) { | ||||
| 			throw new NodeApiError(this.getNode(), { | ||||
| 				message: 'API Key is missing or invalid', | ||||
| 			}); | ||||
| 		} | ||||
| 
 | ||||
| 		const calcId = this.getNodeParameter('calcId', 0) as string; | ||||
| 		if (!calcId) { | ||||
| 			throw new NodeApiError(this.getNode(), { | ||||
| 				message: 'No calculator selected', | ||||
| 			}); | ||||
| 		} | ||||
| 
 | ||||
| 		const pollingInterval = (this.getNodeParameter('pollingInterval', 0) as number) * 60 * 1000; | ||||
| 		const url = `https://api-calc.outgrow.co/api/v1/get_leads/${credentials.apiKey}/${calcId}`; | ||||
| 
 | ||||
| 		const poll = async () => { | ||||
| 			try { | ||||
| 				const responseData = await this.helpers.request({ | ||||
| 					method: 'GET', | ||||
| 					url, | ||||
| 					json: true, | ||||
| 				}); | ||||
| 
 | ||||
| 				if (responseData?.length > 0) { | ||||
| 					this.emit([this.helpers.returnJsonArray(responseData)]); | ||||
| 				} else { | ||||
| 					// UI-friendly "no leads" response
 | ||||
| 					this.emit([this.helpers.returnJsonArray([{ | ||||
| 						status: 'no_data', | ||||
| 						message: 'No new leads in the last 24 hours', | ||||
| 						calculatorId: calcId, | ||||
| 						timestamp: new Date().toISOString(), | ||||
| 					}])]); | ||||
| 				} | ||||
| 			} catch (error) { | ||||
| 				throw new NodeApiError(this.getNode(), error, { | ||||
| 					message: 'Outgrow API Error', | ||||
| 				}); | ||||
| 			} | ||||
| 		}; | ||||
| 
 | ||||
| 		const mode = this.getMode(); | ||||
| 		if (mode === 'manual') { | ||||
| 			return { | ||||
| 				manualTriggerFunction: async () => await poll(), | ||||
| 			}; | ||||
| 		} | ||||
| 
 | ||||
| 		if (mode === 'trigger') { | ||||
| 			const interval = setInterval(poll, pollingInterval); | ||||
| 			await poll(); // Initial call
 | ||||
| 
 | ||||
| 			return { | ||||
| 				closeFunction: async () => clearInterval(interval), | ||||
| 			}; | ||||
| 		} | ||||
| 
 | ||||
| 		return {}; | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										3
									
								
								nodes/outgrow/outgrow.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								nodes/outgrow/outgrow.svg
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| After Width: | Height: | Size: 29 KiB | 
							
								
								
									
										339
									
								
								package.json
									
										
									
									
									
								
							
							
						
						
									
										339
									
								
								package.json
									
										
									
									
									
								
							|  | @ -1,19 +1,16 @@ | |||
| { | ||||
|   "name": "n8n-nodes-<...>", | ||||
|   "name": "n8n-nodes-outgrow", | ||||
|   "version": "0.1.0", | ||||
|   "description": "", | ||||
|   "description": "", | ||||
|   "keywords": [ | ||||
|     "n8n-community-node-package" | ||||
|   ], | ||||
|   "license": "MIT", | ||||
|   "homepage": "", | ||||
|   "author": { | ||||
|     "name": "", | ||||
|     "email": "" | ||||
|   }, | ||||
|   "homepage": "https://github.com/outgrow/n8n-nodes-outgrow#readme", | ||||
|   "author": "", | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/<...>/n8n-nodes-<...>.git" | ||||
|     "url": "git+https://github.com/outgrow/n8n-nodes-outgrow.git" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": ">=20.15" | ||||
|  | @ -33,12 +30,10 @@ | |||
|   "n8n": { | ||||
|     "n8nNodesApiVersion": 1, | ||||
|     "credentials": [ | ||||
|       "dist/credentials/ExampleCredentialsApi.credentials.js", | ||||
|       "dist/credentials/HttpBinApi.credentials.js" | ||||
|       "dist/credentials/outgrowApi.credentials.js" | ||||
|     ], | ||||
|     "nodes": [ | ||||
|       "dist/nodes/ExampleNode/ExampleNode.node.js", | ||||
|       "dist/nodes/HttpBin/HttpBin.node.js" | ||||
|       "dist/nodes/outgrow/outgrow.node.js" | ||||
|     ] | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|  | @ -51,5 +46,325 @@ | |||
|   }, | ||||
|   "peerDependencies": { | ||||
|     "n8n-workflow": "*" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "acorn": "^8.14.1", | ||||
|     "acorn-jsx": "^5.3.2", | ||||
|     "ajv": "^6.12.6", | ||||
|     "ansi-regex": "^5.0.1", | ||||
|     "ansi-styles": "^4.3.0", | ||||
|     "anymatch": "^3.1.3", | ||||
|     "argparse": "^2.0.1", | ||||
|     "array-each": "^1.0.1", | ||||
|     "array-slice": "^1.1.0", | ||||
|     "array-union": "^2.1.0", | ||||
|     "assert": "^2.1.0", | ||||
|     "ast-types": "^0.15.2", | ||||
|     "async-done": "^2.0.0", | ||||
|     "async-settle": "^2.0.0", | ||||
|     "asynckit": "^0.4.0", | ||||
|     "available-typed-arrays": "^1.0.7", | ||||
|     "axios": "^1.8.2", | ||||
|     "b4a": "^1.6.7", | ||||
|     "bach": "^2.0.1", | ||||
|     "balanced-match": "^1.0.2", | ||||
|     "bare-events": "^2.5.4", | ||||
|     "base64-js": "^1.5.1", | ||||
|     "binary-extensions": "^2.3.0", | ||||
|     "bl": "^5.1.0", | ||||
|     "brace-expansion": "^2.0.1", | ||||
|     "braces": "^3.0.3", | ||||
|     "buffer": "^6.0.3", | ||||
|     "call-bind": "^1.0.8", | ||||
|     "call-bind-apply-helpers": "^1.0.2", | ||||
|     "call-bound": "^1.0.4", | ||||
|     "callsites": "^3.1.0", | ||||
|     "camel-case": "^4.1.2", | ||||
|     "chalk": "^4.1.2", | ||||
|     "charenc": "^0.0.2", | ||||
|     "chokidar": "^3.6.0", | ||||
|     "cliui": "^7.0.4", | ||||
|     "clone": "^2.1.2", | ||||
|     "color-convert": "^2.0.1", | ||||
|     "color-name": "^1.1.4", | ||||
|     "combined-stream": "^1.0.8", | ||||
|     "concat-map": "^0.0.1", | ||||
|     "convert-source-map": "^2.0.0", | ||||
|     "copy-props": "^4.0.0", | ||||
|     "cross-spawn": "^7.0.6", | ||||
|     "crypt": "^0.0.2", | ||||
|     "debug": "^4.4.1", | ||||
|     "deep-equal": "^2.2.0", | ||||
|     "deep-is": "^0.1.4", | ||||
|     "define-data-property": "^1.1.4", | ||||
|     "define-properties": "^1.2.1", | ||||
|     "delayed-stream": "^1.0.0", | ||||
|     "detect-file": "^1.0.0", | ||||
|     "dir-glob": "^3.0.1", | ||||
|     "doctrine": "^3.0.0", | ||||
|     "dunder-proto": "^1.0.1", | ||||
|     "each-props": "^3.0.0", | ||||
|     "emoji-regex": "^8.0.0", | ||||
|     "end-of-stream": "^1.4.4", | ||||
|     "es-define-property": "^1.0.1", | ||||
|     "es-errors": "^1.3.0", | ||||
|     "es-get-iterator": "^1.1.3", | ||||
|     "es-object-atoms": "^1.1.1", | ||||
|     "escalade": "^3.2.0", | ||||
|     "escape-string-regexp": "^4.0.0", | ||||
|     "eslint-config-riot": "^1.0.0", | ||||
|     "eslint-plugin-local": "^1.0.0", | ||||
|     "eslint-scope": "^7.2.2", | ||||
|     "eslint-visitor-keys": "^4.2.0", | ||||
|     "espree": "^9.6.1", | ||||
|     "esprima": "^4.0.1", | ||||
|     "esprima-next": "^5.8.4", | ||||
|     "esquery": "^1.6.0", | ||||
|     "esrecurse": "^4.3.0", | ||||
|     "estraverse": "^5.3.0", | ||||
|     "esutils": "^2.0.3", | ||||
|     "expand-tilde": "^2.0.2", | ||||
|     "extend": "^3.0.2", | ||||
|     "fast-deep-equal": "^3.1.3", | ||||
|     "fast-fifo": "^1.3.2", | ||||
|     "fast-glob": "^3.3.3", | ||||
|     "fast-json-stable-stringify": "^2.1.0", | ||||
|     "fast-levenshtein": "^2.0.6", | ||||
|     "fastest-levenshtein": "^1.0.16", | ||||
|     "fastq": "^1.19.1", | ||||
|     "file-entry-cache": "^6.0.1", | ||||
|     "fill-range": "^7.1.1", | ||||
|     "find-up": "^5.0.0", | ||||
|     "findup-sync": "^5.0.0", | ||||
|     "fined": "^2.0.0", | ||||
|     "flagged-respawn": "^2.0.0", | ||||
|     "flat-cache": "^3.2.0", | ||||
|     "flatted": "^3.3.3", | ||||
|     "follow-redirects": "^1.15.9", | ||||
|     "for-each": "^0.3.5", | ||||
|     "for-in": "^1.0.2", | ||||
|     "for-own": "^1.0.0", | ||||
|     "form-data": "^4.0.0", | ||||
|     "fs-mkdirp-stream": "^2.0.1", | ||||
|     "fs.realpath": "^1.0.0", | ||||
|     "function-bind": "^1.1.2", | ||||
|     "functions-have-names": "^1.2.3", | ||||
|     "get-caller-file": "^2.0.5", | ||||
|     "get-intrinsic": "^1.3.0", | ||||
|     "get-proto": "^1.0.1", | ||||
|     "glob": "^7.2.3", | ||||
|     "glob-parent": "^6.0.2", | ||||
|     "glob-stream": "^8.0.3", | ||||
|     "glob-watcher": "^6.0.0", | ||||
|     "global-modules": "^1.0.0", | ||||
|     "global-prefix": "^1.0.2", | ||||
|     "globals": "^13.24.0", | ||||
|     "globby": "^11.1.0", | ||||
|     "glogg": "^2.2.0", | ||||
|     "gopd": "^1.2.0", | ||||
|     "graceful-fs": "^4.2.11", | ||||
|     "graphemer": "^1.4.0", | ||||
|     "gulp-cli": "^3.1.0", | ||||
|     "gulplog": "^2.2.0", | ||||
|     "has-bigints": "^1.1.0", | ||||
|     "has-flag": "^4.0.0", | ||||
|     "has-property-descriptors": "^1.0.2", | ||||
|     "has-symbols": "^1.1.0", | ||||
|     "has-tostringtag": "^1.0.2", | ||||
|     "hasown": "^2.0.2", | ||||
|     "homedir-polyfill": "^1.0.3", | ||||
|     "iconv-lite": "^0.6.3", | ||||
|     "ieee754": "^1.2.1", | ||||
|     "ignore": "^5.3.2", | ||||
|     "import-fresh": "^3.3.1", | ||||
|     "imurmurhash": "^0.1.4", | ||||
|     "indefinite": "^2.5.1", | ||||
|     "inflight": "^1.0.6", | ||||
|     "inherits": "^2.0.4", | ||||
|     "ini": "^1.3.8", | ||||
|     "internal-slot": "^1.1.0", | ||||
|     "interpret": "^3.1.1", | ||||
|     "is-absolute": "^1.0.0", | ||||
|     "is-arguments": "^1.2.0", | ||||
|     "is-array-buffer": "^3.0.5", | ||||
|     "is-bigint": "^1.1.0", | ||||
|     "is-binary-path": "^2.1.0", | ||||
|     "is-boolean-object": "^1.2.2", | ||||
|     "is-buffer": "^1.1.6", | ||||
|     "is-callable": "^1.2.7", | ||||
|     "is-core-module": "^2.16.1", | ||||
|     "is-date-object": "^1.1.0", | ||||
|     "is-extglob": "^2.1.1", | ||||
|     "is-fullwidth-code-point": "^3.0.0", | ||||
|     "is-generator-function": "^1.1.0", | ||||
|     "is-glob": "^4.0.3", | ||||
|     "is-map": "^2.0.3", | ||||
|     "is-nan": "^1.3.2", | ||||
|     "is-negated-glob": "^1.0.0", | ||||
|     "is-number": "^7.0.0", | ||||
|     "is-number-object": "^1.1.1", | ||||
|     "is-path-inside": "^3.0.3", | ||||
|     "is-plain-object": "^5.0.0", | ||||
|     "is-regex": "^1.2.1", | ||||
|     "is-relative": "^1.0.0", | ||||
|     "is-set": "^2.0.3", | ||||
|     "is-shared-array-buffer": "^1.0.4", | ||||
|     "is-string": "^1.1.1", | ||||
|     "is-symbol": "^1.1.1", | ||||
|     "is-typed-array": "^1.1.15", | ||||
|     "is-unc-path": "^1.0.0", | ||||
|     "is-valid-glob": "^1.0.0", | ||||
|     "is-weakmap": "^2.0.2", | ||||
|     "is-weakset": "^2.0.4", | ||||
|     "is-windows": "^1.0.2", | ||||
|     "isarray": "^2.0.5", | ||||
|     "isexe": "^2.0.0", | ||||
|     "isobject": "^3.0.1", | ||||
|     "jmespath": "^0.16.0", | ||||
|     "js-base64": "^3.7.2", | ||||
|     "js-yaml": "^4.1.0", | ||||
|     "json-buffer": "^3.0.1", | ||||
|     "json-schema-traverse": "^0.4.1", | ||||
|     "json-stable-stringify-without-jsonify": "^1.0.1", | ||||
|     "jssha": "^3.3.1", | ||||
|     "keyv": "^4.5.4", | ||||
|     "last-run": "^2.0.0", | ||||
|     "lead": "^4.0.0", | ||||
|     "levn": "^0.4.1", | ||||
|     "liftoff": "^5.0.1", | ||||
|     "locate-path": "^6.0.0", | ||||
|     "lodash": "^4.17.21", | ||||
|     "lodash.merge": "^4.6.2", | ||||
|     "lower-case": "^2.0.2", | ||||
|     "luxon": "^3.4.4", | ||||
|     "map-cache": "^0.2.2", | ||||
|     "math-intrinsics": "^1.1.0", | ||||
|     "md5": "^2.3.0", | ||||
|     "merge2": "^1.4.1", | ||||
|     "micromatch": "^4.0.8", | ||||
|     "mime-db": "^1.52.0", | ||||
|     "mime-types": "^2.1.35", | ||||
|     "minimatch": "^9.0.5", | ||||
|     "ms": "^2.1.3", | ||||
|     "mute-stdout": "^2.0.0", | ||||
|     "n8n-workflow": "^1.82.0", | ||||
|     "natural-compare": "^1.4.0", | ||||
|     "no-case": "^3.0.4", | ||||
|     "normalize-path": "^3.0.0", | ||||
|     "now-and-later": "^3.0.0", | ||||
|     "object-inspect": "^1.13.4", | ||||
|     "object-is": "^1.1.6", | ||||
|     "object-keys": "^1.1.1", | ||||
|     "object.assign": "^4.1.7", | ||||
|     "object.defaults": "^1.1.0", | ||||
|     "object.pick": "^1.3.0", | ||||
|     "once": "^1.4.0", | ||||
|     "optionator": "^0.9.4", | ||||
|     "p-limit": "^3.1.0", | ||||
|     "p-locate": "^5.0.0", | ||||
|     "parent-module": "^1.0.1", | ||||
|     "parse-filepath": "^1.0.2", | ||||
|     "parse-passwd": "^1.0.0", | ||||
|     "pascal-case": "^3.1.2", | ||||
|     "path-exists": "^4.0.0", | ||||
|     "path-is-absolute": "^1.0.1", | ||||
|     "path-key": "^3.1.1", | ||||
|     "path-parse": "^1.0.7", | ||||
|     "path-root": "^0.1.1", | ||||
|     "path-root-regex": "^0.1.2", | ||||
|     "path-type": "^4.0.0", | ||||
|     "picomatch": "^2.3.1", | ||||
|     "pluralize": "^8.0.0", | ||||
|     "possible-typed-array-names": "^1.1.0", | ||||
|     "prelude-ls": "^1.2.1", | ||||
|     "proxy-from-env": "^1.1.0", | ||||
|     "punycode": "^2.3.1", | ||||
|     "queue-microtask": "^1.2.3", | ||||
|     "readable-stream": "^3.6.2", | ||||
|     "readdirp": "^3.6.0", | ||||
|     "recast": "^0.21.5", | ||||
|     "rechoir": "^0.8.0", | ||||
|     "regexp.prototype.flags": "^1.5.4", | ||||
|     "remove-trailing-separator": "^1.1.0", | ||||
|     "replace-ext": "^2.0.0", | ||||
|     "replace-homedir": "^2.0.0", | ||||
|     "require-directory": "^2.1.1", | ||||
|     "resolve": "^1.22.10", | ||||
|     "resolve-dir": "^1.0.1", | ||||
|     "resolve-from": "^4.0.0", | ||||
|     "resolve-options": "^2.0.0", | ||||
|     "reusify": "^1.1.0", | ||||
|     "rimraf": "^3.0.2", | ||||
|     "run-parallel": "^1.2.0", | ||||
|     "safe-buffer": "^5.2.1", | ||||
|     "safe-regex-test": "^1.1.0", | ||||
|     "safer-buffer": "^2.1.2", | ||||
|     "sax": "^1.4.1", | ||||
|     "semver": "^7.7.2", | ||||
|     "semver-greatest-satisfied-range": "^2.0.0", | ||||
|     "sentence-case": "^3.0.4", | ||||
|     "set-function-length": "^1.2.2", | ||||
|     "set-function-name": "^2.0.2", | ||||
|     "shebang-command": "^2.0.0", | ||||
|     "shebang-regex": "^3.0.0", | ||||
|     "side-channel": "^1.1.0", | ||||
|     "side-channel-list": "^1.0.0", | ||||
|     "side-channel-map": "^1.0.1", | ||||
|     "side-channel-weakmap": "^1.0.2", | ||||
|     "slash": "^3.0.0", | ||||
|     "source-map": "^0.6.1", | ||||
|     "sparkles": "^2.1.0", | ||||
|     "stop-iteration-iterator": "^1.1.0", | ||||
|     "stream-composer": "^1.0.2", | ||||
|     "stream-exhaust": "^1.0.2", | ||||
|     "streamx": "^2.22.1", | ||||
|     "string-width": "^4.2.3", | ||||
|     "string_decoder": "^1.3.0", | ||||
|     "strip-ansi": "^6.0.1", | ||||
|     "strip-json-comments": "^3.1.1", | ||||
|     "supports-color": "^7.2.0", | ||||
|     "supports-preserve-symlinks-flag": "^1.0.0", | ||||
|     "sver": "^1.8.4", | ||||
|     "teex": "^1.0.1", | ||||
|     "text-decoder": "^1.2.3", | ||||
|     "text-table": "^0.2.0", | ||||
|     "title-case": "^3.0.3", | ||||
|     "to-regex-range": "^5.0.1", | ||||
|     "to-through": "^3.0.0", | ||||
|     "transliteration": "^2.3.5", | ||||
|     "ts-api-utils": "^2.1.0", | ||||
|     "tslib": "^2.8.1", | ||||
|     "type-check": "^0.4.0", | ||||
|     "type-fest": "^0.20.2", | ||||
|     "unc-path-regex": "^0.1.2", | ||||
|     "undertaker": "^2.0.0", | ||||
|     "undertaker-registry": "^2.0.0", | ||||
|     "upper-case-first": "^2.0.2", | ||||
|     "uri-js": "^4.4.1", | ||||
|     "util": "^0.12.5", | ||||
|     "util-deprecate": "^1.0.2", | ||||
|     "v8flags": "^4.0.1", | ||||
|     "value-or-function": "^4.0.0", | ||||
|     "vinyl": "^3.0.1", | ||||
|     "vinyl-contents": "^2.0.0", | ||||
|     "vinyl-fs": "^4.0.2", | ||||
|     "vinyl-sourcemap": "^2.0.0", | ||||
|     "which": "^2.0.2", | ||||
|     "which-boxed-primitive": "^1.1.1", | ||||
|     "which-collection": "^1.0.2", | ||||
|     "which-typed-array": "^1.1.19", | ||||
|     "word-wrap": "^1.2.5", | ||||
|     "wrap-ansi": "^7.0.0", | ||||
|     "wrappy": "^1.0.2", | ||||
|     "xml2js": "^0.6.2", | ||||
|     "xmlbuilder": "^11.0.1", | ||||
|     "y18n": "^5.0.8", | ||||
|     "yargs": "^16.2.0", | ||||
|     "yargs-parser": "^20.2.9", | ||||
|     "yocto-queue": "^0.1.0", | ||||
|     "zod": "^3.24.1" | ||||
|   }, | ||||
|   "bugs": { | ||||
|     "url": "https://github.com/outgrow/n8n-nodes-outgrow/issues" | ||||
|   } | ||||
| } | ||||
|  | @ -3,8 +3,8 @@ | |||
| 		"strict": true, | ||||
| 		"module": "commonjs", | ||||
| 		"moduleResolution": "node", | ||||
| 		"target": "es2019", | ||||
| 		"lib": ["es2019", "es2020", "es2022.error"], | ||||
| 		"target": "ES2020", | ||||
| 		"lib": ["es2019", "es2020", "DOM","es2022.error"], | ||||
| 		"removeComments": true, | ||||
| 		"useUnknownInCatchVariables": false, | ||||
| 		"forceConsistentCasingInFileNames": true, | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue