mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-02 08:27:29 -08:00
0da398b0e4
* ✨ change FE to handle new object type * 🚸 improve UX of handling invalid credentials * 🚧 WIP * 🎨 fix typescript issues * 🐘 add migrations for all supported dbs * ✏️ add description to migrations * ⚡ add credential update on import * ⚡ resolve after merge issues * 👕 fix lint issues * ⚡ check credentials on workflow create/update * update interface * 👕 fix ts issues * ⚡ adaption to new credentials UI * 🐛 intialize cache on BE for credentials check * 🐛 fix undefined oldCredentials * 🐛 fix deleting credential * 🐛 fix check for undefined keys * 🐛 fix disabling edit in execution * 🎨 just show credential name on execution view * ✏️ remove TODO * ⚡ implement review suggestions * ⚡ add cache to getCredentialsByType * ⏪ use getter instead of cache * ✏️ fix variable name typo * 🐘 include waiting nodes to migrations * 🐛 fix reverting migrations command * ⚡ update typeorm command * ✨ create db:revert command * 👕 fix lint error * ✨ Add optional authenticate method to credentials * ⚡ Simplify code and add authentication support to MattermostApi * 👕 Fix lint issue * ⚡ Add support to own-mode * 👕 Fix lint issue * ✨ Add support for predefined auth types bearer and headerAuth * ⚡ Make sure that DateTime Node always returns strings * ⚡ Add support for moment types to If Node * ⚡ Make it possible for HTTP Request Node to use all credential types * ✨ Add basicAuth support * Add a new dropcontact node * ✨ First basic implementation of mainly JSON based nodes * ✨ Add fixedCollection support, added value parameter and expression support for value and property * Improvements to #2389 * ⚡ Add credentials verification * ⚡ Small improvement * ⚡ set default time to 45 seconds * ✨ Add support for preSend and postReceive methods * ➕ Add lodash merge and set depedency to workflow * 👕 Fix lint issue * ⚡ Improvements * ⚡ Improvements * ⚡ Improvements * ⚡ Improvements * ⚡ Improvements * 🐛 Set siren and language correctly * ⚡ Add support for requestDefaults * ⚡ Add support for baseURL to httpRequest * ⚡ Move baseURL to correct location * ✨ Add support for options loading * 🐛 Fix error with fullAccess nodes * ✨ Add credential test functionality * 🐛 Fix issue with OAuth autentication and lint issue * ⚡ Fix build issue * 🐛 Fix issue that url got always overwritten to empty * ✨ Add pagination support * ⚡ Code fix required after merge * ⚡ Remove not needed imports * ⚡ Fix credential test * ✨ Add expression support for request properties and $self support on properties * ⚡ Rename $self to $value * 👕 Fix lint issue * ⚡ Add example how to send data in path * ✨ Make it possible to not sent in dot notation * ✨ Add support for postReceive:rootProperty * ⚡ Fix typo * ✨ Add support for postReceive:set * ⚡ Some fixes * ⚡ Small improvement * ;zap: Separate RoutingNode code * ⚡ Simplify code and fix bug * ⚡ Remove unused code * ✨ Make it possible to define "request" and "requestProperty" on options * 👕 Fix lint issue * ⚡ Change $credentials variables name * ✨ Enable expressions and access to credentials in requestDefaults * ⚡ Make parameter option loading use RoutingNode.makeRoutingRequest * ✨ Allow requestOperations overwrite on LoadOptions * ✨ Make it possible to access current node parameters in loadOptions * ⚡ Rename parameters variable to make future proof * ⚡ Make it possible to use offset-pagination with body * ✨ Add support for queryAuth * ⚡ Never return more items than requested * ✨ Make it possible to overwrite requestOperations on parameter and option level * 👕 Fix lint issue * ✨ Allow simplified auth also with regular nodes * ✨ Add support for receiving binary data * 🐛 Fix example node * ⚡ Rename property "name" to "displayName" in loadOptions * ⚡ Send data by default as "query" if nothing is set * ⚡ Rename $self to $parent * ⚡ Change to work with INodeExecutionData instead of IDataObject * ⚡ Improve binaryData handling * ⚡ Property design improvements * ⚡ Fix property name * 🚨 Add some tests * ⚡ Add also test for request * ⚡ Improve test and fix issues * ⚡ Improvements to loadOptions * ⚡ Normalize loadOptions with rest of code * ⚡ Add info text * ✨ Add support for $value in postReceive * 🚨 Add tests for RoutingNode.runNode * ⚡ Remove TODOs and make url property optional * ⚡ Fix bug and lint issue * 🐛 Fix bug that not the correct property got used * 🚨 Add tests for CredentialsHelper.authenticate * ⚡ Improve code and resolve expressions also everywhere for loadOptions and credential test requests * ✨ Make it possible to define multiple preSend and postReceive actions * ✨ Allow to define tests on credentials * ⚡ Remove test data * ⬆️ Update package-lock.json file * ⚡ Remove old not longer used code Co-authored-by: Ben Hesseldieck <b.hesseldieck@gmail.com> Co-authored-by: Mutasem <mutdmour@gmail.com> Co-authored-by: PaulineDropcontact <pauline@dropcontact.io> Co-authored-by: ricardo <ricardoespinoza105@gmail.com>
264 lines
7.1 KiB
TypeScript
264 lines
7.1 KiB
TypeScript
import {
|
|
IExecuteFunctions,
|
|
IHookFunctions,
|
|
ILoadOptionsFunctions,
|
|
} from 'n8n-core';
|
|
|
|
import {
|
|
IDataObject,
|
|
INodePropertyOptions,
|
|
NodeApiError,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
OptionsWithUri,
|
|
} from 'request';
|
|
|
|
export interface ICustomInterface {
|
|
name: string;
|
|
key: string;
|
|
options?: Array<{
|
|
id: number;
|
|
label: string;
|
|
}>;
|
|
}
|
|
|
|
export interface ICustomProperties {
|
|
[key: string]: ICustomInterface;
|
|
}
|
|
|
|
/**
|
|
* Make an API request to Pipedrive
|
|
*
|
|
* @param {IHookFunctions} this
|
|
* @param {string} method
|
|
* @param {string} url
|
|
* @param {object} body
|
|
* @returns {Promise<any>}
|
|
*/
|
|
export async function pipedriveApiRequest(this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions, method: string, endpoint: string, body: IDataObject, query: IDataObject = {}, formData?: IDataObject, downloadFile?: boolean): Promise<any> { // tslint:disable-line:no-any
|
|
const authenticationMethod = this.getNodeParameter('authentication', 0);
|
|
|
|
const options: OptionsWithUri = {
|
|
headers: {
|
|
Accept: 'application/json',
|
|
},
|
|
method,
|
|
qs: query,
|
|
uri: `https://api.pipedrive.com/v1${endpoint}`,
|
|
};
|
|
|
|
if (downloadFile === true) {
|
|
options.encoding = null;
|
|
} else {
|
|
options.json = true;
|
|
}
|
|
|
|
if (Object.keys(body).length !== 0) {
|
|
options.body = body;
|
|
}
|
|
|
|
if (formData !== undefined && Object.keys(formData).length !== 0) {
|
|
options.formData = formData;
|
|
}
|
|
|
|
if (query === undefined) {
|
|
query = {};
|
|
}
|
|
|
|
try {
|
|
const credentialType = authenticationMethod === 'apiToken' ? 'pipedriveApi' : 'pipedriveOAuth2Api';
|
|
const responseData = await this.helpers.requestWithAuthentication.call(this, credentialType, options);
|
|
|
|
if (downloadFile === true) {
|
|
return {
|
|
data: responseData,
|
|
};
|
|
}
|
|
|
|
if (responseData.success === false) {
|
|
throw new NodeApiError(this.getNode(), responseData);
|
|
}
|
|
|
|
return {
|
|
additionalData: responseData.additional_data,
|
|
data: (responseData.data === null) ? [] : responseData.data,
|
|
};
|
|
} catch (error) {
|
|
throw new NodeApiError(this.getNode(), error);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Make an API request to paginated Pipedrive endpoint
|
|
* and return all results
|
|
*
|
|
* @export
|
|
* @param {(IHookFunctions | IExecuteFunctions)} this
|
|
* @param {string} method
|
|
* @param {string} endpoint
|
|
* @param {IDataObject} body
|
|
* @param {IDataObject} [query]
|
|
* @returns {Promise<any>}
|
|
*/
|
|
export async function pipedriveApiRequestAllItems(this: IHookFunctions | IExecuteFunctions, method: string, endpoint: string, body: IDataObject, query?: IDataObject): Promise<any> { // tslint:disable-line:no-any
|
|
|
|
if (query === undefined) {
|
|
query = {};
|
|
}
|
|
query.limit = 100;
|
|
query.start = 0;
|
|
|
|
const returnData: IDataObject[] = [];
|
|
|
|
let responseData;
|
|
|
|
do {
|
|
responseData = await pipedriveApiRequest.call(this, method, endpoint, body, query);
|
|
// the search path returns data diferently
|
|
if (responseData.data.items) {
|
|
returnData.push.apply(returnData, responseData.data.items);
|
|
} else {
|
|
returnData.push.apply(returnData, responseData.data);
|
|
}
|
|
|
|
query.start = responseData.additionalData.pagination.next_start;
|
|
} while (
|
|
responseData.additionalData !== undefined &&
|
|
responseData.additionalData.pagination !== undefined &&
|
|
responseData.additionalData.pagination.more_items_in_collection === true
|
|
);
|
|
|
|
return {
|
|
data: returnData,
|
|
};
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Gets the custom properties from Pipedrive
|
|
*
|
|
* @export
|
|
* @param {(IHookFunctions | IExecuteFunctions)} this
|
|
* @param {string} resource
|
|
* @returns {Promise<ICustomProperties>}
|
|
*/
|
|
export async function pipedriveGetCustomProperties(this: IHookFunctions | IExecuteFunctions, resource: string): Promise<ICustomProperties> {
|
|
|
|
const endpoints: { [key: string]: string } = {
|
|
'activity': '/activityFields',
|
|
'deal': '/dealFields',
|
|
'organization': '/organizationFields',
|
|
'person': '/personFields',
|
|
'product': '/productFields',
|
|
};
|
|
|
|
if (endpoints[resource] === undefined) {
|
|
throw new NodeOperationError(this.getNode(), `The resource "${resource}" is not supported for resolving custom values!`);
|
|
}
|
|
|
|
const requestMethod = 'GET';
|
|
|
|
const body = {};
|
|
const qs = {};
|
|
// Get the custom properties and their values
|
|
const responseData = await pipedriveApiRequest.call(this, requestMethod, endpoints[resource], body, qs);
|
|
|
|
const customProperties: ICustomProperties = {};
|
|
|
|
for (const customPropertyData of responseData.data) {
|
|
customProperties[customPropertyData.key] = customPropertyData;
|
|
}
|
|
|
|
return customProperties;
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Converts names and values of custom properties from their actual values to the
|
|
* Pipedrive internal ones
|
|
*
|
|
* @export
|
|
* @param {ICustomProperties} customProperties
|
|
* @param {IDataObject} item
|
|
*/
|
|
export function pipedriveEncodeCustomProperties(customProperties: ICustomProperties, item: IDataObject): void {
|
|
let customPropertyData;
|
|
|
|
for (const key of Object.keys(item)) {
|
|
customPropertyData = Object.values(customProperties).find(customPropertyData => customPropertyData.name === key);
|
|
|
|
if (customPropertyData !== undefined) {
|
|
// Is a custom property
|
|
|
|
// Check if also the value has to be resolved or just the key
|
|
if (item[key] !== null && item[key] !== undefined && customPropertyData.options !== undefined && Array.isArray(customPropertyData.options)) {
|
|
// Has an option key so get the actual option-value
|
|
const propertyOption = customPropertyData.options.find(option => option.label.toString() === item[key]!.toString());
|
|
|
|
if (propertyOption !== undefined) {
|
|
item[customPropertyData.key as string] = propertyOption.id;
|
|
delete item[key];
|
|
}
|
|
} else {
|
|
// Does already represent the actual value or is null
|
|
item[customPropertyData.key as string] = item[key];
|
|
delete item[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Converts names and values of custom properties to their actual values
|
|
*
|
|
* @export
|
|
* @param {ICustomProperties} customProperties
|
|
* @param {IDataObject} item
|
|
*/
|
|
export function pipedriveResolveCustomProperties(customProperties: ICustomProperties, item: IDataObject): void {
|
|
let customPropertyData;
|
|
|
|
// Itterate over all keys and replace the custom ones
|
|
for (const key of Object.keys(item)) {
|
|
if (customProperties[key] !== undefined) {
|
|
// Is a custom property
|
|
customPropertyData = customProperties[key];
|
|
|
|
// Check if also the value has to be resolved or just the key
|
|
if (item[key] !== null && item[key] !== undefined && customPropertyData.options !== undefined && Array.isArray(customPropertyData.options)) {
|
|
// Has an option key so get the actual option-value
|
|
const propertyOption = customPropertyData.options.find(option => option.id.toString() === item[key]!.toString());
|
|
|
|
if (propertyOption !== undefined) {
|
|
item[customPropertyData.name as string] = propertyOption.label;
|
|
delete item[key];
|
|
}
|
|
} else {
|
|
// Does already represent the actual value or is null
|
|
item[customPropertyData.name as string] = item[key];
|
|
delete item[key];
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
export function sortOptionParameters(optionParameters: INodePropertyOptions[]): INodePropertyOptions[] {
|
|
optionParameters.sort((a, b) => {
|
|
const aName = a.name.toLowerCase();
|
|
const bName = b.name.toLowerCase();
|
|
if (aName < bName) { return -1; }
|
|
if (aName > bName) { return 1; }
|
|
return 0;
|
|
});
|
|
|
|
return optionParameters;
|
|
}
|