import type { IDataObject, IExecuteFunctions, IHookFunctions, ILoadOptionsFunctions, IHttpRequestMethods, IHttpRequestOptions, INodePropertyOptions, } from 'n8n-workflow'; import get from 'lodash/get'; /** * Make an API request to Asana * */ export async function asanaApiRequest( this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions, method: IHttpRequestMethods, endpoint: `/${string}`, body: object, query?: IDataObject, uri?: string | undefined, ): Promise { const authenticationMethod = this.getNodeParameter('authentication', 0) as string; const options: IHttpRequestOptions = { headers: {}, method, body: method === 'GET' || method === 'HEAD' || method === 'DELETE' ? null : { data: body }, qs: query, url: uri || `https://app.asana.com/api/1.0${endpoint}`, json: true, }; if (options.body === null) { delete options.body; } const credentialType = authenticationMethod === 'accessToken' ? 'asanaApi' : 'asanaOAuth2Api'; return await this.helpers.requestWithAuthentication.call(this, credentialType, options); } export async function asanaApiRequestAllItems( this: IExecuteFunctions | ILoadOptionsFunctions, method: IHttpRequestMethods, endpoint: `/${string}`, body: any = {}, query: IDataObject = {}, ): Promise { const returnData: IDataObject[] = []; let responseData; let uri: string | undefined; query.limit = 100; do { responseData = await asanaApiRequest.call( this, method, endpoint, body as IDataObject, query, uri, ); uri = get(responseData, 'next_page.uri'); returnData.push.apply(returnData, responseData.data as IDataObject[]); } while (responseData.next_page !== null); return returnData; } export async function getWorkspaces(this: ILoadOptionsFunctions): Promise { const endpoint = '/workspaces'; const responseData = await asanaApiRequestAllItems.call(this, 'GET', endpoint, {}); const returnData: INodePropertyOptions[] = []; for (const workspaceData of responseData) { if (workspaceData.resource_type !== 'workspace') { // Not sure if for some reason also ever other resources // get returned but just in case filter them out continue; } returnData.push({ name: workspaceData.name, value: workspaceData.gid, }); } returnData.sort((a, b) => { if (a.name < b.name) { return -1; } if (a.name > b.name) { return 1; } return 0; }); return returnData; } export function getColorOptions(): INodePropertyOptions[] { return [ 'dark-blue', 'dark-brown', 'dark-green', 'dark-orange', 'dark-pink', 'dark-purple', 'dark-red', 'dark-teal', 'dark-warm-gray', 'light-blue', 'light-green', 'light-orange', 'light-pink', 'light-purple', 'light-red', 'light-teal', 'light-warm-gray', 'light-yellow', 'none', ].map((value) => { return { name: value, value, }; }); } export function getTaskFields() { return [ '*', 'GID', 'Resource Type', 'name', 'Approval Status', 'Assignee Status', 'Completed', 'Completed At', 'Completed By', 'Created At', 'Dependencies', 'Dependents', 'Due At', 'Due On', 'External', 'HTML Notes', 'Liked', 'Likes', 'Memberships', 'Modified At', 'Notes', 'Num Likes', 'Resource Subtype', 'Start On', 'Assignee', 'Custom Fields', 'Followers', 'Parent', 'Permalink URL', 'Projects', 'Tags', 'Workspace', ]; }