mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 01:56:46 +00:00
✨ Add Formstack Trigger (#2066)
* add initial formstack integration configuration * implement getForms for formstack integration * implement oauth2 authentication for formstack * fix formstack color * enable formstack webhook registration * remove typeform types * implement formstack fetch * ⚡ Improvements to #1673 * ⚡ Improvements * ⚡ Make work with new async credentials * ⚡ Improvements Co-authored-by: Aniruddha Adhikary <aniruddha@adhikary.net> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
206
packages/nodes-base/nodes/Formstack/GenericFunctions.ts
Normal file
206
packages/nodes-base/nodes/Formstack/GenericFunctions.ts
Normal file
@@ -0,0 +1,206 @@
|
||||
import {
|
||||
IExecuteFunctions,
|
||||
IHookFunctions,
|
||||
ILoadOptionsFunctions,
|
||||
IWebhookFunctions,
|
||||
} from 'n8n-core';
|
||||
|
||||
import {
|
||||
IDataObject,
|
||||
INodePropertyOptions,
|
||||
NodeApiError,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import {
|
||||
OptionsWithUri,
|
||||
} from 'request';
|
||||
|
||||
export interface IFormstackFieldDefinitionType {
|
||||
id: string;
|
||||
label: string;
|
||||
description: string;
|
||||
name: string;
|
||||
type: string;
|
||||
options: unknown;
|
||||
required: string;
|
||||
uniq: string;
|
||||
hidden: string;
|
||||
readonly: string;
|
||||
colspan: string;
|
||||
label_position: string;
|
||||
num_columns: string;
|
||||
date_format: string;
|
||||
time_format: string;
|
||||
}
|
||||
|
||||
export interface IFormstackWebhookResponseBody {
|
||||
FormID: string;
|
||||
UniqueID: string;
|
||||
}
|
||||
|
||||
export interface IFormstackSubmissionFieldContainer {
|
||||
field: string;
|
||||
value: string;
|
||||
}
|
||||
|
||||
export enum FormstackFieldFormat {
|
||||
ID = 'id',
|
||||
Label = 'label',
|
||||
Name = 'name',
|
||||
}
|
||||
|
||||
/**
|
||||
* Make an API request to Formstack
|
||||
*
|
||||
* @param {IHookFunctions} this
|
||||
* @param {string} method
|
||||
* @param {string} url
|
||||
* @param {object} body
|
||||
* @returns {Promise<any>}
|
||||
*/
|
||||
export async function apiRequest(this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions | IWebhookFunctions, method: string, endpoint: string, body: IDataObject = {}, query: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any
|
||||
const authenticationMethod = this.getNodeParameter('authentication', 0);
|
||||
|
||||
const options: OptionsWithUri = {
|
||||
headers: {},
|
||||
method,
|
||||
body,
|
||||
qs: query || {},
|
||||
uri: `https://www.formstack.com/api/v2/${endpoint}`,
|
||||
json: true,
|
||||
};
|
||||
|
||||
if (!Object.keys(body).length) {
|
||||
delete options.body;
|
||||
}
|
||||
|
||||
try {
|
||||
if (authenticationMethod === 'accessToken') {
|
||||
const credentials = await this.getCredentials('formstackApi') as IDataObject;
|
||||
|
||||
if (credentials === undefined) {
|
||||
throw new Error('No credentials got returned!');
|
||||
}
|
||||
|
||||
options.headers!['Authorization'] = `Bearer ${credentials.accessToken}`;
|
||||
return await this.helpers.request!(options);
|
||||
} else {
|
||||
return await this.helpers.requestOAuth2!.call(this, 'formstackOAuth2Api', options);
|
||||
}
|
||||
} catch (error) {
|
||||
throw new NodeApiError(this.getNode(), error);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Make an API request to paginated Formstack 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 apiRequestAllItems(this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions | IWebhookFunctions, method: string, endpoint: string, body: IDataObject, dataKey: string, query?: IDataObject): Promise<any> { // tslint:disable-line:no-any
|
||||
|
||||
if (query === undefined) {
|
||||
query = {};
|
||||
}
|
||||
|
||||
query.per_page = 200;
|
||||
query.page = 0;
|
||||
|
||||
const returnData = {
|
||||
items: [] as IDataObject[],
|
||||
};
|
||||
|
||||
let responseData;
|
||||
|
||||
do {
|
||||
query.page += 1;
|
||||
|
||||
responseData = await apiRequest.call(this, method, endpoint, body, query);
|
||||
returnData.items.push.apply(returnData.items, responseData[dataKey]);
|
||||
} while (
|
||||
responseData.total !== undefined &&
|
||||
Math.ceil(responseData.total / query.per_page) > query.page
|
||||
);
|
||||
|
||||
return returnData;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns all the available forms
|
||||
*
|
||||
* @export
|
||||
* @param {ILoadOptionsFunctions} this
|
||||
* @returns {Promise<INodePropertyOptions[]>}
|
||||
*/
|
||||
export async function getForms(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
|
||||
const endpoint = 'form.json';
|
||||
const responseData = await apiRequestAllItems.call(this, 'GET', endpoint, {}, 'forms', { folders: false });
|
||||
|
||||
if (responseData.items === undefined) {
|
||||
throw new Error('No data got returned');
|
||||
}
|
||||
const returnData: INodePropertyOptions[] = [];
|
||||
for (const baseData of responseData.items) {
|
||||
returnData.push({
|
||||
name: baseData.name,
|
||||
value: baseData.id,
|
||||
});
|
||||
}
|
||||
return returnData;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns all the fields of a form
|
||||
*
|
||||
* @export
|
||||
* @param {ILoadOptionsFunctions} this
|
||||
* @param {string} formID
|
||||
* @returns {Promise<IFormstackFieldDefinitionType[]>}
|
||||
*/
|
||||
export async function getFields(this: IWebhookFunctions, formID: string): Promise<Record<string, IFormstackFieldDefinitionType>> {
|
||||
const endpoint = `form/${formID}.json`;
|
||||
const responseData = await apiRequestAllItems.call(this, 'GET', endpoint, {}, 'fields');
|
||||
|
||||
if (responseData.items === undefined) {
|
||||
throw new Error('No form fields meta data got returned');
|
||||
}
|
||||
|
||||
const fields = responseData.items as IFormstackFieldDefinitionType[];
|
||||
const fieldMap: Record<string, IFormstackFieldDefinitionType> = {};
|
||||
|
||||
fields.forEach(field => {
|
||||
fieldMap[field.id] = field;
|
||||
});
|
||||
|
||||
return fieldMap;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns all the fields of a form
|
||||
*
|
||||
* @export
|
||||
* @param {ILoadOptionsFunctions} this
|
||||
* @param {string} uniqueId
|
||||
* @returns {Promise<IFormstackFieldDefinitionType[]>}
|
||||
*/
|
||||
export async function getSubmission(this: ILoadOptionsFunctions | IWebhookFunctions, uniqueId: string): Promise<IFormstackSubmissionFieldContainer[]> {
|
||||
const endpoint = `submission/${uniqueId}.json`;
|
||||
const responseData = await apiRequestAllItems.call(this, 'GET', endpoint, {}, 'data');
|
||||
|
||||
if (responseData.items === undefined) {
|
||||
throw new Error('No form fields meta data got returned');
|
||||
}
|
||||
|
||||
return responseData.items as IFormstackSubmissionFieldContainer[];
|
||||
}
|
||||
Reference in New Issue
Block a user