mirror of
https://github.com/n8n-io/n8n.git
synced 2026-05-28 15:27:03 +02:00
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com> Co-authored-by: Michael Kret <88898367+michael-radency@users.noreply.github.com>
510 lines
15 KiB
TypeScript
510 lines
15 KiB
TypeScript
import jwt from 'jsonwebtoken';
|
|
import moment from 'moment-timezone';
|
|
import { DateTime } from 'luxon';
|
|
import type {
|
|
IExecuteFunctions,
|
|
ILoadOptionsFunctions,
|
|
ICredentialDataDecryptedObject,
|
|
IDataObject,
|
|
INodePropertyOptions,
|
|
JsonObject,
|
|
IHttpRequestMethods,
|
|
IRequestOptions,
|
|
IPollFunctions,
|
|
} from 'n8n-workflow';
|
|
import { NodeApiError } from 'n8n-workflow';
|
|
|
|
import { resolveAuthUrl } from '../../credentials/SalesforceJwtApi.credentials';
|
|
|
|
function getOptions(
|
|
this: IExecuteFunctions | ILoadOptionsFunctions | IPollFunctions,
|
|
method: IHttpRequestMethods,
|
|
endpoint: string,
|
|
|
|
body: any,
|
|
qs: IDataObject,
|
|
instanceUrl: string,
|
|
): IRequestOptions {
|
|
const options: IRequestOptions = {
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
method,
|
|
body,
|
|
qs,
|
|
uri: `${instanceUrl}/services/data/v59.0${endpoint}`,
|
|
json: true,
|
|
};
|
|
|
|
if (!Object.keys(options.body as IDataObject).length) {
|
|
delete options.body;
|
|
}
|
|
|
|
return options;
|
|
}
|
|
|
|
async function getAccessToken(
|
|
this: IExecuteFunctions | ILoadOptionsFunctions | IPollFunctions,
|
|
credentials: ICredentialDataDecryptedObject,
|
|
): Promise<IDataObject> {
|
|
const now = moment().unix();
|
|
const authUrl = resolveAuthUrl(credentials);
|
|
|
|
const signature = jwt.sign(
|
|
{
|
|
iss: credentials.clientId as string,
|
|
sub: credentials.username as string,
|
|
aud: authUrl,
|
|
exp: now + 3 * 60,
|
|
},
|
|
credentials.privateKey as string,
|
|
{
|
|
algorithm: 'RS256',
|
|
header: {
|
|
alg: 'RS256',
|
|
},
|
|
},
|
|
);
|
|
|
|
const options: IRequestOptions = {
|
|
headers: {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
},
|
|
method: 'POST',
|
|
form: {
|
|
grant_type: 'urn:ietf:params:oauth:grant-type:jwt-bearer',
|
|
assertion: signature,
|
|
},
|
|
uri: `${authUrl}/services/oauth2/token`,
|
|
json: true,
|
|
};
|
|
|
|
return await this.helpers.request(options);
|
|
}
|
|
|
|
export async function salesforceApiRequest(
|
|
this: IExecuteFunctions | ILoadOptionsFunctions | IPollFunctions,
|
|
method: IHttpRequestMethods,
|
|
endpoint: string,
|
|
|
|
body: any = {},
|
|
qs: IDataObject = {},
|
|
uri?: string,
|
|
option: IDataObject = {},
|
|
): Promise<any> {
|
|
const authenticationMethod = this.getNodeParameter('authentication', 0, 'oAuth2') as string;
|
|
try {
|
|
if (authenticationMethod === 'jwt') {
|
|
// https://help.salesforce.com/articleView?id=remoteaccess_oauth_jwt_flow.htm&type=5
|
|
const credentialsType = 'salesforceJwtApi';
|
|
const credentials = await this.getCredentials(credentialsType);
|
|
const response = await getAccessToken.call(this, credentials);
|
|
const { instance_url, access_token } = response;
|
|
const options = getOptions.call(
|
|
this,
|
|
method,
|
|
uri || endpoint,
|
|
body,
|
|
qs,
|
|
instance_url as string,
|
|
);
|
|
this.logger.debug(
|
|
`Authentication for "Salesforce" node is using "jwt". Invoking URI ${options.uri}`,
|
|
);
|
|
options.headers!.Authorization = `Bearer ${access_token}`;
|
|
Object.assign(options, option);
|
|
return await this.helpers.request(options);
|
|
} else {
|
|
// https://help.salesforce.com/articleView?id=remoteaccess_oauth_web_server_flow.htm&type=5
|
|
const credentialsType = 'salesforceOAuth2Api';
|
|
const credentials = await this.getCredentials<{
|
|
oauthTokenData: { instance_url: string };
|
|
}>(credentialsType);
|
|
const options = getOptions.call(
|
|
this,
|
|
method,
|
|
uri || endpoint,
|
|
body,
|
|
qs,
|
|
credentials.oauthTokenData.instance_url,
|
|
);
|
|
this.logger.debug(
|
|
`Authentication for "Salesforce" node is using "OAuth2". Invoking URI ${options.uri}`,
|
|
);
|
|
Object.assign(options, option);
|
|
|
|
return await this.helpers.requestOAuth2.call(this, credentialsType, options);
|
|
}
|
|
} catch (error) {
|
|
throw new NodeApiError(this.getNode(), error as JsonObject);
|
|
}
|
|
}
|
|
|
|
export async function salesforceApiRequestAllItems(
|
|
this: IExecuteFunctions | ILoadOptionsFunctions | IPollFunctions,
|
|
propertyName: string,
|
|
method: IHttpRequestMethods,
|
|
endpoint: string,
|
|
|
|
body: any = {},
|
|
query: IDataObject = {},
|
|
): Promise<any> {
|
|
const returnData: IDataObject[] = [];
|
|
|
|
let responseData;
|
|
let uri: string | undefined;
|
|
|
|
do {
|
|
responseData = await salesforceApiRequest.call(this, method, endpoint, body, query, uri);
|
|
uri = `${endpoint}/${responseData.nextRecordsUrl?.split('/')?.pop()}`;
|
|
returnData.push.apply(returnData, responseData[propertyName] as IDataObject[]);
|
|
} while (responseData.nextRecordsUrl !== undefined && responseData.nextRecordsUrl !== null);
|
|
|
|
return returnData;
|
|
}
|
|
|
|
/**
|
|
* Sorts the given options alphabetically
|
|
*
|
|
*/
|
|
export function sortOptions(options: INodePropertyOptions[]): void {
|
|
options.sort((a, b) => {
|
|
if (a.name < b.name) {
|
|
return -1;
|
|
}
|
|
if (a.name > b.name) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Escapes special characters in a string value for use in SOQL queries.
|
|
* SOQL requires escaping: single quotes, backslashes, and certain control characters.
|
|
* @see https://developer.salesforce.com/docs/atlas.en-us.soql_sosl.meta/soql_sosl/sforce_api_calls_soql_select_quotedstringescapes.htm
|
|
*/
|
|
export function escapeSoqlString(value: string): string {
|
|
return value
|
|
.replace(/\\/g, '\\\\') // Escape backslashes first
|
|
.replace(/'/g, "\\'") // Escape single quotes
|
|
.replace(/"/g, '\\"') // Escape double quotes
|
|
.replace(/\n/g, '\\n') // Escape newlines
|
|
.replace(/\r/g, '\\r') // Escape carriage returns
|
|
.replace(/\t/g, '\\t') // Escape tabs
|
|
.replace(/\f/g, '\\f') // Escape form feeds
|
|
.replace(/[\b]/g, '\\b'); // Escape backspaces
|
|
}
|
|
|
|
/**
|
|
* Validates that a field name is a valid Salesforce field identifier.
|
|
* Valid field names contain only alphanumeric characters, underscores, and can include
|
|
* relationship traversal dots for related object fields.
|
|
* @throws Error if the field name contains invalid characters
|
|
*/
|
|
export function validateSoqlFieldName(fieldName: string): string {
|
|
// Salesforce field names: alphanumeric, underscore, can have dots for relationships
|
|
// Examples: Name, Account__c, Account.Name, Custom_Field__c, Account__r, MyObject__Share
|
|
// Supports all Salesforce suffixes: __c, __r, __x, __e, __b, __mdt, __Share, __History, __Feed, etc.
|
|
const validFieldPattern =
|
|
/^[a-zA-Z][a-zA-Z0-9_]*(__[a-zA-Z]+)?(\.[a-zA-Z][a-zA-Z0-9_]*(__[a-zA-Z]+)?)*$/;
|
|
|
|
if (!validFieldPattern.test(fieldName)) {
|
|
throw new Error(`Invalid SOQL field name: ${fieldName}`);
|
|
}
|
|
|
|
return fieldName;
|
|
}
|
|
|
|
/**
|
|
* Validates and returns a valid SOQL comparison operator.
|
|
* @see https://developer.salesforce.com/docs/atlas.en-us.soql_sosl.meta/soql_sosl/sforce_api_calls_soql_select_comparisonoperators.htm
|
|
* @throws Error if the operator is invalid
|
|
*/
|
|
export function validateSoqlOperator(operation: string): string {
|
|
// Normalize whitespace: trim and replace multiple spaces with single space
|
|
const normalized = operation.trim().replace(/\s+/g, ' ').toUpperCase();
|
|
|
|
const validOperators: Record<string, string> = {
|
|
EQUAL: '=',
|
|
'=': '=',
|
|
'!=': '!=',
|
|
'<>': '<>',
|
|
'<': '<',
|
|
'<=': '<=',
|
|
'>': '>',
|
|
'>=': '>=',
|
|
LIKE: 'LIKE',
|
|
'NOT LIKE': 'NOT LIKE',
|
|
IN: 'IN',
|
|
'NOT IN': 'NOT IN',
|
|
INCLUDES: 'INCLUDES',
|
|
EXCLUDES: 'EXCLUDES',
|
|
};
|
|
|
|
const validOperator = validOperators[normalized] || validOperators[operation];
|
|
if (!validOperator) {
|
|
throw new Error(`Invalid SOQL operator: ${operation}`);
|
|
}
|
|
|
|
return validOperator;
|
|
}
|
|
|
|
/**
|
|
* Validates that an SObject name is a valid Salesforce object identifier.
|
|
* Standard objects: Account, Contact, Lead, etc.
|
|
* Custom objects: MyObject__c, Namespace__MyObject__c
|
|
* External objects: MyObject__x
|
|
* Platform events: MyEvent__e
|
|
* Big objects: MyBigObject__b
|
|
* Custom metadata types: MyMetadata__mdt
|
|
* @throws Error if the object name contains invalid characters
|
|
*/
|
|
export function validateSoqlObjectName(objectName: string): string {
|
|
// Salesforce object names: alphanumeric, underscore
|
|
// Can have namespace prefix like Namespace__ObjectName__c
|
|
// Standard objects: Account, Contact, Lead, etc.
|
|
// Suffixes (__c, __mdt, __Share, __ChangeEvent, etc.) are validated as English letters only,
|
|
// consistent with validateSoqlFieldName, to stay future-proof without enumerating every suffix.
|
|
const validObjectPattern = /^[A-Za-z][A-Za-z0-9_]*(?:__[A-Za-z][A-Za-z0-9_]*)*$/;
|
|
|
|
if (!validObjectPattern.test(objectName)) {
|
|
throw new Error(`Invalid SOQL object name: ${objectName}`);
|
|
}
|
|
|
|
return objectName;
|
|
}
|
|
|
|
/**
|
|
* Salesforce date literals that should not be quoted
|
|
* @see https://developer.salesforce.com/docs/atlas.en-us.soql_sosl.meta/soql_sosl/sforce_api_calls_soql_select_dateformats.htm
|
|
*/
|
|
const SALESFORCE_DATE_LITERALS = new Set([
|
|
'YESTERDAY',
|
|
'TODAY',
|
|
'TOMORROW',
|
|
'LAST_WEEK',
|
|
'THIS_WEEK',
|
|
'NEXT_WEEK',
|
|
'LAST_MONTH',
|
|
'THIS_MONTH',
|
|
'NEXT_MONTH',
|
|
'LAST_90_DAYS',
|
|
'NEXT_90_DAYS',
|
|
'THIS_QUARTER',
|
|
'LAST_QUARTER',
|
|
'NEXT_QUARTER',
|
|
'THIS_YEAR',
|
|
'LAST_YEAR',
|
|
'NEXT_YEAR',
|
|
'THIS_FISCAL_QUARTER',
|
|
'LAST_FISCAL_QUARTER',
|
|
'NEXT_FISCAL_QUARTER',
|
|
'THIS_FISCAL_YEAR',
|
|
'LAST_FISCAL_YEAR',
|
|
'NEXT_FISCAL_YEAR',
|
|
]);
|
|
|
|
export function getValue(value: any): string | number | boolean {
|
|
if (value === null || value === undefined) {
|
|
return 'null';
|
|
}
|
|
|
|
if (typeof value === 'boolean') {
|
|
return value;
|
|
}
|
|
|
|
if (typeof value === 'number') {
|
|
if (!Number.isFinite(value)) {
|
|
throw new Error('Invalid numeric value: must be a finite number');
|
|
}
|
|
return value;
|
|
}
|
|
|
|
// Handle arrays - convert to IN clause format with escaped values
|
|
if (Array.isArray(value)) {
|
|
const escapedValues = value.map((v) => {
|
|
if (typeof v === 'string') {
|
|
// Only escape strings, don't try to convert to numbers
|
|
return `'${escapeSoqlString(v)}'`;
|
|
}
|
|
if (typeof v === 'number' && Number.isFinite(v)) {
|
|
return v.toString();
|
|
}
|
|
if (typeof v === 'boolean') {
|
|
return v.toString();
|
|
}
|
|
throw new Error('Array values must be strings, numbers, or booleans');
|
|
});
|
|
return `(${escapedValues.join(',')})`;
|
|
}
|
|
|
|
if (typeof value === 'string') {
|
|
value = value.trim();
|
|
// Check for Salesforce date literals (e.g., TODAY, LAST_N_DAYS:7)
|
|
const upperValue = value.toUpperCase();
|
|
if (SALESFORCE_DATE_LITERALS.has(upperValue)) {
|
|
return upperValue;
|
|
}
|
|
|
|
// Check for LAST_N_DAYS, NEXT_N_DAYS, N_DAYS_AGO, etc. patterns
|
|
if (
|
|
/^(LAST|NEXT)_N_(DAYS|WEEKS|MONTHS|QUARTERS|YEARS|FISCAL_QUARTERS|FISCAL_YEARS):\d+$/.test(
|
|
upperValue,
|
|
)
|
|
) {
|
|
return upperValue;
|
|
}
|
|
|
|
// Check for N_DAYS_AGO, N_WEEKS_AGO, etc. patterns
|
|
if (
|
|
/^N_(DAYS|WEEKS|MONTHS|QUARTERS|YEARS|FISCAL_QUARTERS|FISCAL_YEARS)_AGO:\d+$/.test(upperValue)
|
|
) {
|
|
return upperValue;
|
|
}
|
|
|
|
// Check for Salesforce datetime format: YYYY-MM-DDTHH:mm:ss(.SSS)?(Z|[+-]HH:mm)
|
|
if (/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{3})?(Z|[+-]\d{2}:\d{2})?$/.test(value)) {
|
|
const luxonValue = DateTime.fromISO(value);
|
|
if (luxonValue.isValid) {
|
|
return value;
|
|
}
|
|
}
|
|
|
|
// Check for Salesforce date format: YYYY-MM-DD
|
|
if (/^\d{4}-\d{2}-\d{2}$/.test(value)) {
|
|
const luxonValue = DateTime.fromISO(value);
|
|
if (luxonValue.isValid) {
|
|
return value;
|
|
}
|
|
}
|
|
|
|
// Detect numeric strings and return them unquoted (leading zeros are preserved as strings)
|
|
if (/^-?(0|[1-9]\d*)(\.\d+)?$/.test(value)) {
|
|
const numericValue = Number(value);
|
|
if (Number.isFinite(numericValue)) {
|
|
return numericValue;
|
|
}
|
|
}
|
|
|
|
// All other strings are escaped and quoted
|
|
return `'${escapeSoqlString(value)}'`;
|
|
}
|
|
|
|
throw new Error(`Unsupported value type: ${typeof value}`);
|
|
}
|
|
|
|
export function getConditions(options: IDataObject): string | undefined {
|
|
const conditions = (options.conditionsUi as IDataObject)?.conditionValues as IDataObject[];
|
|
|
|
if (!Array.isArray(conditions) || conditions.length === 0) {
|
|
return undefined;
|
|
}
|
|
|
|
const conditionStrings = conditions.map((condition: IDataObject) => {
|
|
const field = validateSoqlFieldName(condition.field as string);
|
|
const operator = validateSoqlOperator(condition.operation as string);
|
|
const value = getValue(condition.value);
|
|
|
|
return `${field} ${operator} ${value}`;
|
|
});
|
|
|
|
return `WHERE ${conditionStrings.join(' AND ')}`;
|
|
}
|
|
|
|
export function getDefaultFields(sobject: string) {
|
|
return (
|
|
{
|
|
Account: 'id,name,type,LastModifiedDate',
|
|
Lead: 'id,company,firstname,lastname,street,postalCode,city,email,status,LastModifiedDate',
|
|
Contact: 'id,firstname,lastname,email,LastModifiedDate',
|
|
Opportunity: 'id,accountId,amount,probability,type,LastModifiedDate',
|
|
Case: 'id,accountId,contactId,priority,status,subject,type,LastModifiedDate',
|
|
Task: 'id,subject,status,priority,LastModifiedDate',
|
|
Attachment: 'id,name,LastModifiedDate',
|
|
User: 'id,name,email,LastModifiedDate',
|
|
} as IDataObject
|
|
)[sobject];
|
|
}
|
|
|
|
export function getQuery(options: IDataObject, sobject: string, returnAll: boolean, limit = 0) {
|
|
const validSobject = validateSoqlObjectName(sobject);
|
|
|
|
const fields: string[] = [];
|
|
if (options.fields) {
|
|
// options.fields is comma separated in standard Salesforce objects and array in custom Salesforce objects -- handle both cases
|
|
if (typeof options.fields === 'string') {
|
|
const fieldList = options.fields.split(',').map((f) => f.trim());
|
|
fields.push.apply(
|
|
fields,
|
|
fieldList.map((f) => validateSoqlFieldName(f)),
|
|
);
|
|
} else {
|
|
fields.push.apply(
|
|
fields,
|
|
(options.fields as string[]).map((f) => validateSoqlFieldName(f)),
|
|
);
|
|
}
|
|
} else {
|
|
fields.push.apply(
|
|
fields,
|
|
((getDefaultFields(validSobject) as string) || 'id,LastModifiedDate').split(','),
|
|
);
|
|
}
|
|
const conditions = getConditions(options);
|
|
|
|
let query = `SELECT ${fields.join(',')} FROM ${validSobject} ${conditions ? conditions : ''}`;
|
|
|
|
if (!returnAll) {
|
|
query = `SELECT ${fields.join(',')} FROM ${validSobject} ${
|
|
conditions ? conditions : ''
|
|
} LIMIT ${limit}`;
|
|
}
|
|
|
|
return query;
|
|
}
|
|
|
|
/**
|
|
* Calculates the polling start date with safety margin to account for Salesforce indexing delays
|
|
*/
|
|
export function getPollStartDate(lastTimeChecked: string | undefined): string {
|
|
if (!lastTimeChecked) {
|
|
return DateTime.now().toISO();
|
|
}
|
|
const safetyMarginMinutes = 15;
|
|
return DateTime.fromISO(lastTimeChecked).minus({ minutes: safetyMarginMinutes }).toISO();
|
|
}
|
|
|
|
/**
|
|
* Filters out already processed items and manages the processed IDs list
|
|
*/
|
|
export function filterAndManageProcessedItems(
|
|
responseData: IDataObject[],
|
|
processedIds: string[],
|
|
): { newItems: IDataObject[]; updatedProcessedIds: string[] } {
|
|
const processedIdsSet = new Set(processedIds);
|
|
|
|
const newItems: IDataObject[] = [];
|
|
const newItemIds: string[] = [];
|
|
|
|
for (const item of responseData) {
|
|
if (typeof item.Id !== 'string') continue;
|
|
|
|
const itemKey =
|
|
typeof item.LastModifiedDate === 'string' ? `${item.Id}_${item.LastModifiedDate}` : item.Id;
|
|
|
|
if (!processedIdsSet.has(itemKey)) {
|
|
newItems.push(item);
|
|
newItemIds.push(itemKey);
|
|
}
|
|
}
|
|
|
|
const remainingProcessedIds = Array.from(processedIdsSet);
|
|
const updatedProcessedIds = remainingProcessedIds.concat(newItemIds);
|
|
|
|
const MAX_IDS = 10000;
|
|
const trimmedProcessedIds = updatedProcessedIds.slice(-MAX_IDS);
|
|
|
|
return { newItems, updatedProcessedIds: trimmedProcessedIds };
|
|
}
|