joel-plutio #1
					 12 changed files with 6108 additions and 12744 deletions
				
			
		
							
								
								
									
										17
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | |||
| root = true | ||||
| 
 | ||||
| [*] | ||||
| charset = utf-8 | ||||
| indent_style = tab | ||||
| indent_size = 2 | ||||
| end_of_line = lf | ||||
| insert_final_newline = true | ||||
| trim_trailing_whitespace = true | ||||
| 
 | ||||
| [package.json] | ||||
| indent_style = space | ||||
| indent_size = 2 | ||||
| 
 | ||||
| [*.yml] | ||||
| indent_style = space | ||||
| indent_size = 2 | ||||
							
								
								
									
										51
									
								
								.eslintrc.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								.eslintrc.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,51 @@ | |||
| module.exports = { | ||||
| 	root: true, | ||||
| 
 | ||||
| 	env: { | ||||
| 		browser: true, | ||||
| 		es6: true, | ||||
| 		node: true, | ||||
| 	}, | ||||
| 
 | ||||
| 	parser: '@typescript-eslint/parser', | ||||
| 	parserOptions: { | ||||
| 		project: ['./tsconfig.json'], | ||||
| 		sourceType: 'module', | ||||
| 		extraFileExtensions: ['.json'], | ||||
| 	}, | ||||
| 	ignorePatterns: [ | ||||
| 		'.eslintrc.js', | ||||
| 		'**/*.js', | ||||
| 		'**/node_modules/**', | ||||
| 		'**/dist/**', | ||||
| 	], | ||||
| 
 | ||||
| 	overrides: [ | ||||
| 		{ | ||||
| 			files: ['package.json'], | ||||
| 			plugins: ['eslint-plugin-n8n-nodes-base'], | ||||
| 			extends: ['plugin:n8n-nodes-base/community'], | ||||
| 		}, | ||||
| 		{ | ||||
| 			files: ['./credentials/**/*.ts'], | ||||
| 			plugins: ['eslint-plugin-n8n-nodes-base'], | ||||
| 			extends: ['plugin:n8n-nodes-base/credentials'], | ||||
| 			rules: { | ||||
| 				'n8n-nodes-base/cred-class-field-documentation-url-missing': 'off', | ||||
| 				'n8n-nodes-base/cred-class-field-documentation-url-miscased': 'off', | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			files: ['./nodes/**/*.ts'], | ||||
| 			plugins: ['eslint-plugin-n8n-nodes-base'], | ||||
| 			extends: ['plugin:n8n-nodes-base/nodes'], | ||||
| 			rules: { | ||||
| 				'n8n-nodes-base/node-execute-block-missing-continue-on-fail': 'off', | ||||
| 				'n8n-nodes-base/node-resource-description-filename-against-convention': 'off', | ||||
| 				'n8n-nodes-base/node-param-fixed-collection-type-unsorted-items': 'off', | ||||
| 				'n8n-nodes-base/node-execute-block-operation-missing-singular-pairing': 'off', | ||||
| 				'n8n-nodes-base/node-execute-block-operation-missing-plural-pairing': 'off', | ||||
| 			}, | ||||
| 		}, | ||||
| 	], | ||||
| }; | ||||
							
								
								
									
										126
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										126
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -1,120 +1,8 @@ | |||
| # ---> Node | ||||
| # Logs | ||||
| logs | ||||
| *.log | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| lerna-debug.log* | ||||
| .pnpm-debug.log* | ||||
| 
 | ||||
| # Diagnostic reports (https://nodejs.org/api/report.html) | ||||
| report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json | ||||
| 
 | ||||
| # Runtime data | ||||
| pids | ||||
| *.pid | ||||
| *.seed | ||||
| *.pid.lock | ||||
| 
 | ||||
| # Directory for instrumented libs generated by jscoverage/JSCover | ||||
| lib-cov | ||||
| 
 | ||||
| # Coverage directory used by tools like istanbul | ||||
| coverage | ||||
| *.lcov | ||||
| 
 | ||||
| # nyc test coverage | ||||
| .nyc_output | ||||
| 
 | ||||
| # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) | ||||
| .grunt | ||||
| 
 | ||||
| # Bower dependency directory (https://bower.io/) | ||||
| bower_components | ||||
| 
 | ||||
| # node-waf configuration | ||||
| .lock-wscript | ||||
| 
 | ||||
| # Compiled binary addons (https://nodejs.org/api/addons.html) | ||||
| build/Release | ||||
| 
 | ||||
| # Dependency directories | ||||
| node_modules/ | ||||
| jspm_packages/ | ||||
| 
 | ||||
| # Snowpack dependency directory (https://snowpack.dev/) | ||||
| web_modules/ | ||||
| 
 | ||||
| # TypeScript cache | ||||
| *.tsbuildinfo | ||||
| 
 | ||||
| # Optional npm cache directory | ||||
| .npm | ||||
| 
 | ||||
| # Optional eslint cache | ||||
| .eslintcache | ||||
| 
 | ||||
| # Microbundle cache | ||||
| .rpt2_cache/ | ||||
| .rts2_cache_cjs/ | ||||
| .rts2_cache_es/ | ||||
| .rts2_cache_umd/ | ||||
| 
 | ||||
| # Optional REPL history | ||||
| .node_repl_history | ||||
| 
 | ||||
| # Output of 'npm pack' | ||||
| *.tgz | ||||
| 
 | ||||
| # Yarn Integrity file | ||||
| .yarn-integrity | ||||
| 
 | ||||
| # dotenv environment variables file | ||||
| .env | ||||
| .env.test | ||||
| .env.production | ||||
| 
 | ||||
| # parcel-bundler cache (https://parceljs.org/) | ||||
| .cache | ||||
| .parcel-cache | ||||
| 
 | ||||
| # Next.js build output | ||||
| .next | ||||
| out | ||||
| 
 | ||||
| # Nuxt.js build / generate output | ||||
| .nuxt | ||||
| node_modules | ||||
| .DS_Store | ||||
| .tmp | ||||
| tmp | ||||
| dist | ||||
| 
 | ||||
| # Gatsby files | ||||
| .cache/ | ||||
| # Comment in the public line in if your project uses Gatsby and not Next.js | ||||
| # https://nextjs.org/blog/next-9-1#public-directory-support | ||||
| # public | ||||
| 
 | ||||
| # vuepress build output | ||||
| .vuepress/dist | ||||
| 
 | ||||
| # Serverless directories | ||||
| .serverless/ | ||||
| 
 | ||||
| # FuseBox cache | ||||
| .fusebox/ | ||||
| 
 | ||||
| # DynamoDB Local files | ||||
| .dynamodb/ | ||||
| 
 | ||||
| # TernJS port file | ||||
| .tern-port | ||||
| 
 | ||||
| # Stores VSCode versions used for testing VSCode extensions | ||||
| .vscode-test | ||||
| 
 | ||||
| # yarn v2 | ||||
| .yarn/cache | ||||
| .yarn/unplugged | ||||
| .yarn/build-state.yml | ||||
| .yarn/install-state.gz | ||||
| .pnp.* | ||||
| 
 | ||||
| npm-debug.log* | ||||
| yarn.lock | ||||
| .vscode/launch.json | ||||
|  |  | |||
							
								
								
									
										51
									
								
								.prettierrc.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								.prettierrc.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,51 @@ | |||
| module.exports = { | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#semicolons
 | ||||
| 	 */ | ||||
| 	semi: true, | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#trailing-commas
 | ||||
| 	 */ | ||||
| 	trailingComma: 'all', | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#bracket-spacing
 | ||||
| 	 */ | ||||
| 	bracketSpacing: true, | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#tabs
 | ||||
| 	 */ | ||||
| 	useTabs: true, | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#tab-width
 | ||||
| 	 */ | ||||
| 	tabWidth: 2, | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#arrow-function-parentheses
 | ||||
| 	 */ | ||||
| 	arrowParens: 'always', | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#quotes
 | ||||
| 	 */ | ||||
| 	singleQuote: true, | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#quote-props
 | ||||
| 	 */ | ||||
| 	quoteProps: 'as-needed', | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#end-of-line
 | ||||
| 	 */ | ||||
| 	endOfLine: 'lf', | ||||
| 
 | ||||
| 	/** | ||||
| 	 * https://prettier.io/docs/en/options.html#print-width
 | ||||
| 	 */ | ||||
| 	printWidth: 100, | ||||
| }; | ||||
							
								
								
									
										7
									
								
								.vscode/extensions.json
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.vscode/extensions.json
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | |||
| { | ||||
| 	"recommendations": [ | ||||
| 		"dbaeumer.vscode-eslint", | ||||
| 		"EditorConfig.EditorConfig", | ||||
| 		"esbenp.prettier-vscode", | ||||
| 	] | ||||
| } | ||||
							
								
								
									
										32
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										32
									
								
								README.md
									
										
									
									
									
								
							|  | @ -4,9 +4,33 @@ This repo contains the N8N Plutio node created by Chykalophia. | |||
| We created this N8N module so that we could internet our internal tools together. This module connects via the Plutio API to manage: | ||||
| 
 | ||||
| - Tasks | ||||
|   - Get (single) Task | ||||
| 	- Delete Task | ||||
| 	- Update Task | ||||
| 	- Create Task | ||||
| 	- Copy Task | ||||
| 	- Move Task | ||||
| - Projects | ||||
|   - Create Project | ||||
| 	- Get (single) Project | ||||
| 	  - By user email | ||||
| 		- By user ID | ||||
| 		- By Project name | ||||
| 		- By Project ID | ||||
| 	- Update Project | ||||
| 	- Delete Project | ||||
| - Comments | ||||
|   - Create Comment | ||||
| 	- Get (single) Comment | ||||
| 	- Update Comment | ||||
| 	- Delete Comment | ||||
| - Invoices | ||||
|   - Create Invoice | ||||
| 	- Get (single) Invoice | ||||
| 	- Update Invoice | ||||
| 	- Delete Invoice | ||||
| 
 | ||||
| This module uses Plutio API version 1.10 | ||||
| 
 | ||||
| # About Chykalophia | ||||
| 
 | ||||
|  | @ -18,8 +42,12 @@ We are a team of dedicated nerds and creatived working with women-led brands to | |||
| 
 | ||||
| ## Using this Custom node & Installtion | ||||
| 
 | ||||
| Run `npm i n8n-nodes-plutio` from inside the docker container or your n8n folder for modules. | ||||
| Update N8N to version 0.193.5 or higher. | ||||
| 
 | ||||
| Go to N8N Settings -> Community nodes -> Install a community node -> write `n8n-nodes-plutio` | ||||
| 
 | ||||
| Note: You need to have owner/administrator privilege to install community node. | ||||
| 
 | ||||
| ## License | ||||
| 
 | ||||
| [MIT](https://github.com/n8n-io/n8n-nodes-starter/blob/master/LICENSE.md) | ||||
| [MIT](https://github.com/n8n-io/n8n-nodes-starter/blob/master/LICENSE.md) | ||||
|  |  | |||
|  | @ -1,12 +1,43 @@ | |||
| import { | ||||
| 	ICredentialType, | ||||
| 	INodeProperties, | ||||
| 	ICredentialDataDecryptedObject, | ||||
| 	IAuthenticateGeneric, | ||||
| 	IExecuteFunctions, | ||||
| 	ICredentialTestRequest, | ||||
| 	NodeApiError, | ||||
| 	IDataObject, | ||||
| } from 'n8n-workflow'; | ||||
| 
 | ||||
| export class PlutioApi implements ICredentialType { | ||||
| 	name = 'plutioApi'; | ||||
| 	displayName = 'Plutio API'; | ||||
| 	extends = ['oAuth2Api']; | ||||
| 	properties: INodeProperties[] = [ | ||||
| 		// {
 | ||||
| 		// 	displayName: 'Grant Type',
 | ||||
| 		// 	name: 'grantType',
 | ||||
| 		// 	type: 'hidden',
 | ||||
| 		// 	default: 'client_credentials',
 | ||||
| 		// },
 | ||||
| 		// {
 | ||||
| 		// 	displayName: 'Scope',
 | ||||
| 		// 	name: 'scope',
 | ||||
| 		// 	type: 'hidden',
 | ||||
| 		// 	default: '*',
 | ||||
| 		// },
 | ||||
| 		// {
 | ||||
| 		// 	displayName: 'Authentication',
 | ||||
| 		// 	name: 'authentication',
 | ||||
| 		// 	type: 'hidden',
 | ||||
| 		// 	default: 'header',
 | ||||
| 		// },
 | ||||
| 		// {
 | ||||
| 		// 		displayName: 'Access Token URL',
 | ||||
| 		// 		name: 'accessTokenUrl',
 | ||||
| 		// 		type: 'hidden',
 | ||||
| 		// 		default: 'https://api.plutio.com/v1.10/oauth/token',
 | ||||
| 		// },
 | ||||
| 		{ | ||||
| 				displayName: 'Client ID', | ||||
| 				name: 'clientId', | ||||
|  | @ -29,4 +60,57 @@ export class PlutioApi implements ICredentialType { | |||
| 			default: '', | ||||
| 		}, | ||||
| 	]; | ||||
| 
 | ||||
| 	// async preAuthentication(this: IExecuteFunctions, credentials: ICredentialDataDecryptedObject) {
 | ||||
| 	// 	const endpoint = 'api.plutio.com/v1.10';
 | ||||
| 	// 	const returnData: IDataObject[] = [];
 | ||||
| 	// 	let access_token;
 | ||||
| 	// 	try {
 | ||||
| 	// 		access_token = await this.helpers.httpRequest({
 | ||||
| 	// 			method: 'POST',
 | ||||
| 	// 			url: `https://${endpoint}/oauth/token`,
 | ||||
| 	// 			headers: {
 | ||||
| 	// 				'Content-Type': 'application/x-www-form-urlencoded',
 | ||||
| 	// 				'business': `${credentials.business}`,
 | ||||
| 	// 			},
 | ||||
| 	// 			body: {
 | ||||
| 	// 				'client_id': `${credentials.clientId}`,
 | ||||
| 	// 				'client_secret': `${credentials.clientSecret}`,
 | ||||
| 	// 				'grant_type': 'client_credentials',
 | ||||
| 	// 			},
 | ||||
| 	// 			json: true,
 | ||||
| 	// 		}) as {access_token: IDataObject};
 | ||||
| 	// 		if (Array.isArray(access_token)) {
 | ||||
| 	// 			returnData.push.apply(returnData, access_token as IDataObject[]);
 | ||||
| 	// 		} else {
 | ||||
| 	// 			if (access_token === undefined) {
 | ||||
| 	// 				access_token = {
 | ||||
| 	// 					success: true,
 | ||||
| 	// 				};
 | ||||
| 	// 			}
 | ||||
| 	// 			returnData.push(access_token as IDataObject);
 | ||||
| 	// 		}
 | ||||
| 	// 		if (returnData[0].accessToken) {
 | ||||
| 	// 			return {accessToken: returnData[0].accessToken};
 | ||||
| 	// 		}
 | ||||
| 	// 	} catch (error) {
 | ||||
| 	// 		throw new NodeApiError(this.getNode(), error);
 | ||||
| 	// 	}
 | ||||
| 	// }
 | ||||
| 	// authenticate: IAuthenticateGeneric = {
 | ||||
| 	// 	type: 'generic',
 | ||||
| 	// 	properties: {
 | ||||
| 	// 		headers: {
 | ||||
| 	// 			'Content-Type': 'application/json',
 | ||||
| 	// 			'Business': '={{$credentials.business}}',
 | ||||
| 	// 			'Authorization': '=Bearer {{$credentials.accessToken}}',
 | ||||
| 	// 		},
 | ||||
| 	// 	},
 | ||||
| 	// };
 | ||||
| 	// test: ICredentialTestRequest = {
 | ||||
| 	// 	request: {
 | ||||
| 	// 		baseURL: 'https://api.plutio.com/v1.10',
 | ||||
| 	// 		url: '/templates',
 | ||||
| 	// 	},
 | ||||
| 	// };
 | ||||
| } | ||||
|  |  | |||
|  | @ -17,7 +17,11 @@ async function plutioApiRequestToken(this: IExecuteFunctions | ILoadOptionsFunct | |||
| 	const clientId = `${credentials.clientId}`; | ||||
| 	const clientSecret = `${credentials.clientSecret}`; | ||||
| 	const business = `${credentials.business}`; | ||||
| <<<<<<< HEAD | ||||
| 	const endpoint = 'api.plutio.com/v1.10'; | ||||
| ======= | ||||
| 	const endpoint = 'api.plutio.com/v1.9'; | ||||
| >>>>>>> master | ||||
| 	const returnData: IDataObject[] = []; | ||||
| 
 | ||||
| 	let responseData; | ||||
|  | @ -58,7 +62,11 @@ async function plutioApiRequestToken(this: IExecuteFunctions | ILoadOptionsFunct | |||
| 
 | ||||
| // Rest API function for plutio node.
 | ||||
| export async function plutioApiRequest(this: IExecuteFunctions | ILoadOptionsFunctions, method: string, resource: string, body: any = {}, query: IDataObject = {}, uri?: string, option: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any
 | ||||
| <<<<<<< HEAD | ||||
| 	const endpoint = 'api.plutio.com/v1.10'; | ||||
| ======= | ||||
| 	const endpoint = 'api.plutio.com/v1.9'; | ||||
| >>>>>>> master | ||||
| 	const credentials = await this.getCredentials('plutioApi'); | ||||
| 	const plutioApiToken = await plutioApiRequestToken.call(this); | ||||
| 	const business = `${credentials.business}`; | ||||
|  |  | |||
|  | @ -50,6 +50,7 @@ interface ICreatePlutioBody { | |||
| 	currency?: string; | ||||
| 	discount?: string; | ||||
| 	index?: number; | ||||
| 	contributors?: [string]; | ||||
| } | ||||
| 
 | ||||
| export class Plutio implements INodeType { | ||||
|  | @ -93,7 +94,7 @@ export class Plutio implements INodeType { | |||
| 						value: 'invoice', | ||||
| 					}, | ||||
| 					{ | ||||
| 						name: 'Project-CKLPH', | ||||
| 						name: 'Project', | ||||
| 						value: 'project', | ||||
| 					}, | ||||
| 				], | ||||
|  | @ -208,6 +209,21 @@ export class Plutio implements INodeType { | |||
| 				return returnData; | ||||
| 			}, | ||||
| 
 | ||||
| 			async getProjectTemplateId(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { | ||||
| 				const returnData: INodePropertyOptions[] = []; | ||||
| 				const ids = await plutioApiRequest.call(this, 'GET', '/templates', {}, {'entityType': 'project'}); | ||||
| 				for (const id of ids) { | ||||
| 					const templateName = id.title; | ||||
| 					const templateId = id._id; | ||||
| 
 | ||||
| 					returnData.push({ | ||||
| 						name: templateName, | ||||
| 						value: templateId, | ||||
| 					}); | ||||
| 				} | ||||
| 				return returnData; | ||||
| 			}, | ||||
| 
 | ||||
| 			async getProjectId(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { | ||||
| 				const returnData: INodePropertyOptions[] = []; | ||||
| 				const ids = await plutioApiRequest.call(this, 'GET', '/projects'); | ||||
|  | @ -227,7 +243,7 @@ export class Plutio implements INodeType { | |||
| 			// select them easily
 | ||||
| 			async getCustomFieldTitle(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { | ||||
| 				const returnData: INodePropertyOptions[] = []; | ||||
| 				const fields = await plutioApiRequest.call(this, 'GET', '/custom-fields'); | ||||
| 				const fields = await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, {'entityType': 'task'}); | ||||
| 				for (const field of fields) { | ||||
| 					if ('task' === field.entityType) { | ||||
| 						const fieldName = field.title; | ||||
|  | @ -241,6 +257,21 @@ export class Plutio implements INodeType { | |||
| 				} | ||||
| 				return returnData; | ||||
| 			}, | ||||
| 
 | ||||
| 			async getProjectCustomField(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { | ||||
| 				const returnData: INodePropertyOptions[] = []; | ||||
| 				const fields = await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, {'entityType': 'project'}); | ||||
| 				for (const field of fields) { | ||||
| 					const fieldName = field.title; | ||||
| 					const fieldValue = field._id; | ||||
| 
 | ||||
| 					returnData.push({ | ||||
| 						name: fieldName, | ||||
| 						value: fieldValue, | ||||
| 					}); | ||||
| 				} | ||||
| 				return returnData; | ||||
| 			}, | ||||
| 		}, | ||||
| 	}; | ||||
| 
 | ||||
|  | @ -326,20 +357,22 @@ export class Plutio implements INodeType { | |||
| 						} | ||||
| 						if (options.customFields) { | ||||
| 							const metadata = (options.customFields as IDataObject).customField as IDataObject[]; | ||||
| 							const customQs = {"$or":[{"inputType": "select"}, {"inputType": "multi"}],"entityType":"task"}; | ||||
| 							await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, customQs).then(responses => { | ||||
| 								for (const data of metadata) { | ||||
| 									for (const response of responses) { | ||||
| 										if (response._id === data._id) { | ||||
| 											for (const option of response.options) { | ||||
| 												if (option.name === data.value) { | ||||
| 													data.value = (option.name as string).replace(/^[(a-zA-Z\s)]*$/g, option._id); | ||||
| 							if (metadata) { | ||||
| 								const customQs = {"$or":[{"inputType": "select"}, {"inputType": "multi"}],"entityType":"task"}; | ||||
| 								await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, customQs).then(responses => { | ||||
| 									for (const data of metadata) { | ||||
| 										for (const response of responses) { | ||||
| 											if (response._id === data._id) { | ||||
| 												for (const option of response.options) { | ||||
| 													if (option.name === data.value) { | ||||
| 														data.value = (option.name as string).replace(/^[(a-zA-Z\s)]*$/g, option._id); | ||||
| 													} | ||||
| 												} | ||||
| 											} | ||||
| 										} | ||||
| 									} | ||||
| 								} | ||||
| 							}); | ||||
| 								}); | ||||
| 							} | ||||
| 							body.customFields = metadata; | ||||
| 							delete options.customFields; | ||||
| 						} | ||||
|  | @ -813,6 +846,159 @@ export class Plutio implements INodeType { | |||
| 							responseData = {'name': 'default'}; | ||||
| 						} | ||||
| 					} | ||||
| 					if ('get' === operation) { | ||||
| 						const options = this.getNodeParameter('options', i) as IDataObject; | ||||
| 
 | ||||
| 						if (options.contributor) { | ||||
| 							let contributorId; | ||||
| 							const users = await plutioApiRequest.call(this, 'GET', '/people'); | ||||
| 							for (const user of users) { | ||||
| 								if (user.contactEmails[0].address === options.contributor || options.contributor === user._id) { | ||||
| 									contributorId = user._id; | ||||
| 								} | ||||
| 							} | ||||
| 							qs.contributors = contributorId as string; | ||||
| 						} | ||||
| 						if (options._id) { | ||||
| 							qs._id = options._id as string; | ||||
| 						} | ||||
| 						responseData = await plutioApiRequest.call(this, 'GET', '/projects', {}, qs); | ||||
| 					} | ||||
| 					if ('create' === operation) { | ||||
| 						const options = this.getNodeParameter('options', i) as IDataObject; | ||||
| 						const customFields = this.getNodeParameter('customFields', i) as IDataObject; | ||||
| 						const contributors = this.getNodeParameter('contributorsUi', i) as IDataObject; | ||||
| 
 | ||||
| 						if (options.name) { | ||||
| 							body.name = options.name as string; | ||||
| 						} | ||||
| 						if (options.templateId) { | ||||
| 							body.templateId = options.templateId as string; | ||||
| 						} | ||||
| 						if (customFields) { | ||||
| 							const metadata = (customFields as IDataObject).customField as IDataObject[]; | ||||
| 							if (metadata) { | ||||
| 								const customQs = {"$or":[{"inputType": "select"}, {"inputType": "multi"}],"entityType":"project"}; | ||||
| 								await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, customQs).then(responses => { | ||||
| 									for (const data of metadata) { | ||||
| 										for (const response of responses) { | ||||
| 											if (response._id === data._id) { | ||||
| 												for (const option of response.options) { | ||||
| 													if (option.name === data.value) { | ||||
| 														data.value = (option.name as string).replace(/^[(a-zA-Z\s)]*$/g, option._id); | ||||
| 													} | ||||
| 												} | ||||
| 											} | ||||
| 										} | ||||
| 									} | ||||
| 								}); | ||||
| 							} | ||||
| 							body.customFields = metadata as IDataObject[]; | ||||
| 						} | ||||
| 						if (contributors) { | ||||
| 							const metadata = (contributors as IDataObject).contributors as IDataObject[]; | ||||
| 							let contributor: IProperyId[] = []; | ||||
| 							const users: IProperyId[] = []; | ||||
| 
 | ||||
| 							if (metadata) { | ||||
| 								for (const data of metadata) { | ||||
| 									contributor.push(data.value as IDataObject); | ||||
| 								} | ||||
| 
 | ||||
| 								// flatten contributor array.
 | ||||
| 								contributor = contributor.flatMap(a => a); | ||||
| 								const customQs = {'status': 'active'}; | ||||
| 
 | ||||
| 								await plutioApiRequest.call(this, 'GET', '/people', {}, customQs).then(people => { | ||||
| 									for (const id of contributor) { | ||||
| 										for (const person of people) { | ||||
| 											const userName = (person.name.last) ? `${person.name.first} ${person.name.last}` : `${person.name.first}`; | ||||
| 											if (id === person._id || id === userName) { | ||||
| 												users.push(person._id); | ||||
| 											} | ||||
| 										} | ||||
| 									} | ||||
| 								}); | ||||
| 							} | ||||
| 
 | ||||
| 							// remove duplicates.
 | ||||
| 							contributor = users.filter((c, index) => { | ||||
| 								return users.indexOf(c) === index; | ||||
| 							}); | ||||
| 
 | ||||
| 							body.contributors = contributor as [string]; | ||||
| 						} | ||||
| 						responseData = await plutioApiRequest.call(this, 'POST', '/projects', body); | ||||
| 					} | ||||
| 					if ('update' === operation) { | ||||
| 						const _id = this.getNodeParameter('_id', 0) as string; | ||||
| 						const options = this.getNodeParameter('options', i) as IDataObject; | ||||
| 						const customFields = this.getNodeParameter('customFields', i) as IDataObject; | ||||
| 						const contributors = this.getNodeParameter('contributorsUi', i) as IDataObject; | ||||
| 
 | ||||
| 						if (_id) { | ||||
| 							body._id = _id as string; | ||||
| 						} | ||||
| 						if (options.name) { | ||||
| 							body.name = options.name as string; | ||||
| 						} | ||||
| 						if (options.templateId) { | ||||
| 							body.templateId = options.templateId as string; | ||||
| 						} | ||||
| 						if (customFields) { | ||||
| 							const metadata = (customFields as IDataObject).customField as IDataObject[]; | ||||
| 							if (metadata) { | ||||
| 								const customQs = {"$or":[{"inputType": "select"}, {"inputType": "multi"}],"entityType":"project"}; | ||||
| 								await plutioApiRequest.call(this, 'GET', '/custom-fields', {}, customQs).then(responses => { | ||||
| 									for (const data of metadata) { | ||||
| 										for (const response of responses) { | ||||
| 											if (response._id === data._id) { | ||||
| 												for (const option of response.options) { | ||||
| 													if (option.name === data.value) { | ||||
| 														data.value = (option.name as string).replace(/^[(a-zA-Z\s)]*$/g, option._id); | ||||
| 													} | ||||
| 												} | ||||
| 											} | ||||
| 										} | ||||
| 									} | ||||
| 								}); | ||||
| 							} | ||||
| 							body.customFields = metadata as IDataObject[]; | ||||
| 						} | ||||
| 						if (contributors) { | ||||
| 							const metadata = (contributors as IDataObject).contributors as IDataObject[]; | ||||
| 							let contributor: IProperyId[] = []; | ||||
| 							const users: IProperyId[] = []; | ||||
| 							if (metadata) { | ||||
| 								for (const data of metadata) { | ||||
| 									contributor.push(data.value as IDataObject); | ||||
| 								} | ||||
| 
 | ||||
| 								// flatten contributor array.
 | ||||
| 								contributor = contributor.flatMap(a => a); | ||||
| 								const customQs = {'status': 'active'}; | ||||
| 
 | ||||
| 								await plutioApiRequest.call(this, 'GET', '/people', {}, customQs).then(people => { | ||||
| 									for (const id of contributor) { | ||||
| 										for (const person of people) { | ||||
| 											const userName = (person.name.last) ? `${person.name.first} ${person.name.last}` : `${person.name.first}`; | ||||
| 											if (id === person._id || id === userName) { | ||||
| 												users.push(person._id); | ||||
| 											} | ||||
| 										} | ||||
| 									} | ||||
| 								}); | ||||
| 							} | ||||
| 
 | ||||
| 							// remove duplicates.
 | ||||
| 							contributor = users.filter((c, index) => { | ||||
| 								return users.indexOf(c) === index; | ||||
| 							}); | ||||
| 
 | ||||
| 							body.contributors = contributor as [string]; | ||||
| 						} | ||||
| 						responseData = await plutioApiRequest.call(this, 'PUT', '/projects', body);; | ||||
| 					} | ||||
| 					if ('move' === operation) { | ||||
| 						const _id = this.getNodeParameter('_id', 0) as string; | ||||
| 						const index = this.getNodeParameter('index', 0) as number; | ||||
|  |  | |||
|  | @ -41,12 +41,12 @@ export const projectOperations: INodeProperties[] = [ | |||
| 				description: 'Get projects', | ||||
| 				action: 'Get a project', | ||||
| 			}, | ||||
| 			{ | ||||
| 				name: 'Get CKLPH', | ||||
| 				value: 'getCklph', | ||||
| 				description: 'Get projects by contributor\'s email for CKLPH', | ||||
| 				action: 'Get a project', | ||||
| 			}, | ||||
| 			// {
 | ||||
| 			// 	name: 'Chykalophia',
 | ||||
| 			// 	value: 'getCklph',
 | ||||
| 			// 	description: 'Get projects by contributor\'s email for CKLPH',
 | ||||
| 			// 	action: 'Get a project',
 | ||||
| 			// },
 | ||||
| 			{ | ||||
| 				name: 'Move', | ||||
| 				value: 'move', | ||||
|  | @ -85,6 +85,130 @@ export const projectDescription: INodeProperties[] = [ | |||
| 		}, | ||||
| 		description: 'ID of Project', | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Options', | ||||
| 		name: 'options', | ||||
| 		type: 'collection', | ||||
| 		placeholder: 'Add Option', | ||||
| 		default: {}, | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: [ | ||||
| 					'project', | ||||
| 				], | ||||
| 				operation: [ | ||||
| 					'update', | ||||
| 					'create', | ||||
| 				], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				displayName: 'Project Name', | ||||
| 				name: 'name', | ||||
| 				type: 'string', | ||||
| 				default: '', | ||||
| 				description: 'Name of Project', | ||||
| 			}, | ||||
| 			{ | ||||
| 				displayName: 'Template Name or ID', | ||||
| 				name: 'templateId', | ||||
| 				type: 'options', | ||||
| 				description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>', | ||||
| 				default: '', | ||||
| 				typeOptions: { | ||||
| 					loadOptionsMethod: 'getProjectTemplateId', | ||||
| 				}, | ||||
| 			}, | ||||
| 		], | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Custom Fields', | ||||
| 		name: 'customFields', | ||||
| 		type: 'fixedCollection', | ||||
| 		placeholder: 'Add Custom Field', | ||||
| 		typeOptions: { | ||||
| 			multipleValues: true, | ||||
| 		}, | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: [ | ||||
| 					'project', | ||||
| 				], | ||||
| 				operation: [ | ||||
| 					'create', | ||||
| 					'update', | ||||
| 				], | ||||
| 			}, | ||||
| 		}, | ||||
| 		description: 'Key value pairs containing the name and value of the custom field. Only dates in the format YYYY-MM-DD are accepted as input for custom date fields.', | ||||
| 		default: [], | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				displayName: 'Custom Field', | ||||
| 				name: 'customField', | ||||
| 				values: [ | ||||
| 					{ | ||||
| 						displayName: 'Custom Field Name or ID', | ||||
| 						name: '_id', | ||||
| 						type: 'options', | ||||
| 						description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>', | ||||
| 						default: '', | ||||
| 						typeOptions: { | ||||
| 							loadOptionsMethod: 'getProjectCustomField', | ||||
| 						}, | ||||
| 					}, | ||||
| 					{ | ||||
| 						displayName: 'Value Name or ID', | ||||
| 						name: 'value', | ||||
| 						type: 'string', | ||||
| 						default: '', | ||||
| 						description: 'Custom Field\'s values', | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 		], | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Contributors', | ||||
| 		name: 'contributorsUi', | ||||
| 		type: 'fixedCollection', | ||||
| 		default: [], | ||||
| 		placeholder: 'Add Person', | ||||
| 		typeOptions: { | ||||
| 			multipleValues: true, | ||||
| 		}, | ||||
| 		displayOptions: { | ||||
| 			show: { | ||||
| 				resource: [ | ||||
| 					'project', | ||||
| 				], | ||||
| 				operation: [ | ||||
| 					'update', | ||||
| 					'create', | ||||
| 				], | ||||
| 			}, | ||||
| 		}, | ||||
| 		options: [ | ||||
| 			{ | ||||
| 				displayName: 'Contributors', | ||||
| 				name: 'contributors', | ||||
| 				values: [ | ||||
| 					{ | ||||
| 						displayName: 'Contributors: Name or ID', | ||||
| 						name: 'value', | ||||
| 						type: 'options', | ||||
| 						description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>', | ||||
| 						default: '', | ||||
| 						typeOptions: { | ||||
| 							loadOptionsMethod: 'getUsers', | ||||
| 						}, | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 		], | ||||
| 		description: 'Name or ID of the user to whom the task has been assigned. Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/nodes/expressions.html#expressions">expression</a>.', | ||||
| 	}, | ||||
| 	{ | ||||
| 		displayName: 'Index', | ||||
| 		name: 'index', | ||||
|  | @ -132,17 +256,7 @@ export const projectDescription: INodeProperties[] = [ | |||
| 			}, | ||||
| 			{ | ||||
| 				displayName: 'Project Name or ID', | ||||
| 				name: 'name', | ||||
| 				type: 'options', | ||||
| 				description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>', | ||||
| 				default: '', | ||||
| 				typeOptions: { | ||||
| 					loadOptionsMethod: 'getProjectId', | ||||
| 				}, | ||||
| 			}, | ||||
| 			{ | ||||
| 				displayName: 'Client Name or ID', | ||||
| 				name: 'name', | ||||
| 				name: '_id', | ||||
| 				type: 'options', | ||||
| 				description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/code-examples/expressions/">expression</a>', | ||||
| 				default: '', | ||||
|  |  | |||
							
								
								
									
										18090
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										18090
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										28
									
								
								package.json
									
										
									
									
									
								
							
							
						
						
									
										28
									
								
								package.json
									
										
									
									
									
								
							|  | @ -1,25 +1,25 @@ | |||
| { | ||||
|   "name": "n8n-nodes-plutio", | ||||
|   "version": "1.0.0", | ||||
|   "description": "Custom Plutio node module for n8n.", | ||||
|   "version": "0.1.0", | ||||
|   "description": "Custom n8n node module for Plutio.", | ||||
|   "keywords": [ | ||||
|     "n8n-community-node-package" | ||||
|   ], | ||||
|   "license": "MIT", | ||||
|   "homepage": "https://chykalophia.com", | ||||
|   "contributors": | ||||
|   [ | ||||
|     { | ||||
|       "name": "Peter Krzyzek", | ||||
|       "email": "peter@chykalophia.com", | ||||
|       "url": "https://chykalophia.com" | ||||
|     }, | ||||
|     { | ||||
|       "name": "Joel Sanguenza", | ||||
|       "email": "joel@chykalophia.com", | ||||
|       "url": "https://chykalophia.com" | ||||
|     } | ||||
|   ], | ||||
|     [ | ||||
|       { | ||||
|         "name": "Peter Krzyzek", | ||||
|         "email": "peter@chykalophia.com", | ||||
|         "url": "https://chykalophia.com" | ||||
|       }, | ||||
|       { | ||||
|         "name": "Joel Sanguenza", | ||||
|         "email": "joel@chykalophia.com", | ||||
|         "url": "https://chykalophia.com" | ||||
|       } | ||||
|     ], | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://git.cklph.dev/Chykalophia/n8n-nodes-plutio.git" | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue