n8n/packages/nodes-base/nodes/Asana/GenericFunctions.ts
Aaron Gutierrez 15ffd4fb9f
fix(Asana Node): Omit body from GET, HEAD, and DELETE requests (#8057)
Avoid unnecessarily including a request body with GET and HEAD requests.
Per RFC 7230 clients should not include a body for these requests, and
we (Asana) are rolling out an infrastructure change that will cause
these requests to fail.
2023-12-21 19:21:08 +01:00

166 lines
3.3 KiB
TypeScript

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<any> {
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,
};
const credentialType = authenticationMethod === 'accessToken' ? 'asanaApi' : 'asanaOAuth2Api';
return this.helpers.requestWithAuthentication.call(this, credentialType, options);
}
export async function asanaApiRequestAllItems(
this: IExecuteFunctions | ILoadOptionsFunctions,
method: IHttpRequestMethods,
endpoint: string,
body: any = {},
query: IDataObject = {},
): Promise<any> {
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<INodePropertyOptions[]> {
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',
];
}