import type { DeclarativeRestApiSettings, IDataObject, IExecuteFunctions, IExecutePaginationFunctions, IExecuteSingleFunctions, IHookFunctions, IHttpRequestMethods, IHttpRequestOptions, ILoadOptionsFunctions, IN8nHttpFullResponse, INodeExecutionData, INodePropertyOptions, IPollFunctions, IWebhookFunctions, } from 'n8n-workflow'; import { NodeApiError } from 'n8n-workflow'; import type { ToISOTimeOptions } from 'luxon'; import { DateTime } from 'luxon'; const VALID_EMAIL_REGEX = /^(([^<>()[\]\\.,;:\s@"]+(\.[^<>()[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/; const VALID_PHONE_REGEX = /((?:\+|00)[17](?: |\-)?|(?:\+|00)[1-9]\d{0,2}(?: |\-)?|(?:\+|00)1\-\d{3}(?: |\-)?)?(0\d|\([0-9]{3}\)|[1-9]{0,3})(?:((?: |\-)[0-9]{2}){4}|((?:[0-9]{2}){4})|((?: |\-)[0-9]{3}(?: |\-)[0-9]{4})|([0-9]{7}))/; export function isEmailValid(email: string): boolean { return VALID_EMAIL_REGEX.test(String(email).toLowerCase()); } export function isPhoneValid(phone: string): boolean { return VALID_PHONE_REGEX.test(String(phone)); } function dateToIsoSupressMillis(dateTime: string) { const options: ToISOTimeOptions = { suppressMilliseconds: true }; return DateTime.fromISO(dateTime).toISO(options); } export async function taskPostReceiceAction( this: IExecuteSingleFunctions, items: INodeExecutionData[], _response: IN8nHttpFullResponse, ): Promise { const contactId = this.getNodeParameter('contactId'); items.forEach((item) => (item.json.contactId = contactId)); return items; } export async function dueDatePreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { let dueDateParam = this.getNodeParameter('dueDate', null) as string; if (!dueDateParam) { const fields = this.getNodeParameter('updateFields') as { dueDate: string }; dueDateParam = fields.dueDate; } if (!dueDateParam) { throw new NodeApiError( this.getNode(), {}, { message: 'dueDate is required', description: 'dueDate is required' }, ); } const dueDate = dateToIsoSupressMillis(dueDateParam); requestOptions.body = (requestOptions.body || {}) as object; Object.assign(requestOptions.body, { dueDate }); return requestOptions; } export async function contactIdentifierPreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { requestOptions.body = (requestOptions.body || {}) as object; let identifier = this.getNodeParameter('contactIdentifier', null) as string; if (!identifier) { const fields = this.getNodeParameter('updateFields') as { contactIdentifier: string }; identifier = fields.contactIdentifier; } if (isEmailValid(identifier)) { Object.assign(requestOptions.body, { email: identifier }); } else if (isPhoneValid(identifier)) { Object.assign(requestOptions.body, { phone: identifier }); } else { Object.assign(requestOptions.body, { contactId: identifier }); } return requestOptions; } export async function validEmailAndPhonePreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { const body = (requestOptions.body || {}) as { email?: string; phone?: string }; if (body.email && !isEmailValid(body.email)) { const message = `email "${body.email}" has invalid format`; throw new NodeApiError(this.getNode(), {}, { message, description: message }); } if (body.phone && !isPhoneValid(body.phone)) { const message = `phone "${body.phone}" has invalid format`; throw new NodeApiError(this.getNode(), {}, { message, description: message }); } return requestOptions; } export async function dateTimeToEpochPreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { const qs = (requestOptions.qs || {}) as { startDate?: string | number; endDate?: string | number; }; const toEpoch = (dt: string) => new Date(dt).getTime(); if (qs.startDate) qs.startDate = toEpoch(qs.startDate as string); if (qs.endDate) qs.endDate = toEpoch(qs.endDate as string); return requestOptions; } export async function addLocationIdPreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { const { locationId } = ((await this.getCredentials('highLevelOAuth2Api'))?.oauthTokenData as IDataObject) ?? {}; const resource = this.getNodeParameter('resource') as string; const operation = this.getNodeParameter('operation') as string; if (resource === 'contact') { if (operation === 'getAll') { requestOptions.qs = requestOptions.qs || {}; Object.assign(requestOptions.qs, { locationId }); } if (operation === 'create') { requestOptions.body = requestOptions.body || {}; Object.assign(requestOptions.body, { locationId }); } } if (resource === 'opportunity') { if (operation === 'create') { requestOptions.body = requestOptions.body || {}; Object.assign(requestOptions.body, { locationId }); } if (operation === 'getAll') { requestOptions.qs = requestOptions.qs || {}; Object.assign(requestOptions.qs, { location_id: locationId }); } } return requestOptions; } export async function highLevelApiRequest( this: | IExecuteFunctions | IExecuteSingleFunctions | IWebhookFunctions | IPollFunctions | IHookFunctions | ILoadOptionsFunctions, method: IHttpRequestMethods, resource: string, body: IDataObject = {}, qs: IDataObject = {}, url?: string, option: IDataObject = {}, ) { let options: IHttpRequestOptions = { headers: { 'Content-Type': 'application/json', Version: '2021-07-28', }, method, body, qs, url: url || `https://services.leadconnectorhq.com${resource}`, json: true, }; if (!Object.keys(body).length) { delete options.body; } if (!Object.keys(qs).length) { delete options.qs; } options = Object.assign({}, options, option); return await this.helpers.httpRequestWithAuthentication.call(this, 'highLevelOAuth2Api', options); } export async function taskUpdatePreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { const body = (requestOptions.body || {}) as { title?: string; dueDate?: string }; if (!body.title || !body.dueDate) { const contactId = this.getNodeParameter('contactId'); const taskId = this.getNodeParameter('taskId'); const resource = `/contacts/${contactId}/tasks/${taskId}`; const responseData = await highLevelApiRequest.call(this, 'GET', resource); body.title = body.title || responseData.title; // the api response dueDate has to be formatted or it will error on update body.dueDate = body.dueDate || dateToIsoSupressMillis(responseData.dueDate as string); requestOptions.body = body; } return requestOptions; } export async function splitTagsPreSendAction( this: IExecuteSingleFunctions, requestOptions: IHttpRequestOptions, ): Promise { const body = (requestOptions.body || {}) as IDataObject; if (body.tags) { if (Array.isArray(body.tags)) return requestOptions; body.tags = (body.tags as string).split(',').map((tag) => tag.trim()); } return requestOptions; } export async function highLevelApiPagination( this: IExecutePaginationFunctions, requestData: DeclarativeRestApiSettings.ResultOptions, ): Promise { const responseData: INodeExecutionData[] = []; const resource = this.getNodeParameter('resource') as string; const returnAll = this.getNodeParameter('returnAll', false) as boolean; const resourceMapping: { [key: string]: string } = { contact: 'contacts', opportunity: 'opportunities', }; const rootProperty = resourceMapping[resource]; requestData.options.qs = requestData.options.qs || {}; if (returnAll) requestData.options.qs.limit = 100; let responseTotal = 0; do { const pageResponseData: INodeExecutionData[] = await this.makeRoutingRequest(requestData); const items = pageResponseData[0].json[rootProperty] as []; items.forEach((item) => responseData.push({ json: item })); const meta = pageResponseData[0].json.meta as IDataObject; const startAfterId = meta.startAfterId as string; const startAfter = meta.startAfter as number; requestData.options.qs = { startAfterId, startAfter }; responseTotal = (meta.total as number) || 0; } while (returnAll && responseTotal > responseData.length); return responseData; } export async function getPipelineStages( this: ILoadOptionsFunctions, ): Promise { const operation = this.getNodeParameter('operation') as string; let pipelineId = ''; if (operation === 'create') { pipelineId = this.getCurrentNodeParameter('pipelineId') as string; } if (operation === 'update') { pipelineId = this.getNodeParameter('updateFields.pipelineId') as string; } if (operation === 'getAll') { pipelineId = this.getNodeParameter('filters.pipelineId') as string; } const { locationId } = ((await this.getCredentials('highLevelOAuth2Api'))?.oauthTokenData as IDataObject) ?? {}; const pipelines = ( await highLevelApiRequest.call(this, 'GET', '/opportunities/pipelines', undefined, { locationId, }) ).pipelines as IDataObject[]; const pipeline = pipelines.find((p) => p.id === pipelineId); if (pipeline) { const options: INodePropertyOptions[] = (pipeline.stages as IDataObject[]).map((stage) => { const name = stage.name as string; const value = stage.id as string; return { name, value }; }); return options; } return []; } export async function getPipelines(this: ILoadOptionsFunctions): Promise { const { locationId } = ((await this.getCredentials('highLevelOAuth2Api'))?.oauthTokenData as IDataObject) ?? {}; const responseData = await highLevelApiRequest.call( this, 'GET', '/opportunities/pipelines', undefined, { locationId }, ); const pipelines = responseData.pipelines as [{ id: string; name: string; email: string }]; const options: INodePropertyOptions[] = pipelines.map((pipeline) => { const name = pipeline.name; const value = pipeline.id; return { name, value }; }); return options; } export async function getContacts(this: ILoadOptionsFunctions): Promise { const { locationId } = ((await this.getCredentials('highLevelOAuth2Api'))?.oauthTokenData as IDataObject) ?? {}; const responseData = await highLevelApiRequest.call(this, 'GET', '/contacts/', undefined, { locationId, }); const contacts = responseData.contacts as [{ id: string; name: string; email: string }]; const options: INodePropertyOptions[] = contacts.map((contact) => { const name = contact.email; const value = contact.id; return { name, value }; }); return options; } export async function getUsers(this: ILoadOptionsFunctions): Promise { const { locationId } = ((await this.getCredentials('highLevelOAuth2Api'))?.oauthTokenData as IDataObject) ?? {}; const responseData = await highLevelApiRequest.call(this, 'GET', '/users/', undefined, { locationId, }); const users = responseData.users as [{ id: string; name: string; email: string }]; const options: INodePropertyOptions[] = users.map((user) => { const name = user.name; const value = user.id; return { name, value }; }); return options; } export async function getTimezones(this: ILoadOptionsFunctions): Promise { const responseData = await highLevelApiRequest.call(this, 'GET', '/timezones'); const timezones = responseData.timezones as string[]; return timezones.map((zone) => ({ name: zone, value: zone, })) as INodePropertyOptions[]; }