2023-01-27 03:22:44 -08:00
|
|
|
import type {
|
2022-03-20 02:11:06 -07:00
|
|
|
ICredentialDataDecryptedObject,
|
|
|
|
ICredentialTestFunctions,
|
2022-02-11 08:20:41 -08:00
|
|
|
IDataObject,
|
2023-03-09 09:13:15 -08:00
|
|
|
IExecuteFunctions,
|
|
|
|
ILoadOptionsFunctions,
|
2022-02-11 08:20:41 -08:00
|
|
|
IHookFunctions,
|
|
|
|
IWebhookFunctions,
|
2024-12-16 09:39:25 -08:00
|
|
|
IHttpRequestOptions,
|
2022-02-11 08:20:41 -08:00
|
|
|
} from 'n8n-workflow';
|
2023-01-27 03:22:44 -08:00
|
|
|
import { NodeApiError } from 'n8n-workflow';
|
2022-02-11 08:20:41 -08:00
|
|
|
|
2023-06-16 07:26:35 -07:00
|
|
|
import get from 'lodash/get';
|
2022-03-20 02:11:06 -07:00
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
import { query } from './Queries';
|
2022-03-20 02:11:06 -07:00
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
export async function linearApiRequest(
|
|
|
|
this: IExecuteFunctions | IWebhookFunctions | IHookFunctions | ILoadOptionsFunctions,
|
2022-12-02 06:25:21 -08:00
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
body: any = {},
|
|
|
|
option: IDataObject = {},
|
|
|
|
): Promise<any> {
|
2022-02-11 08:20:41 -08:00
|
|
|
const endpoint = 'https://api.linear.app/graphql';
|
2023-09-20 02:19:12 -07:00
|
|
|
const authenticationMethod = this.getNodeParameter('authentication', 0, 'apiToken') as string;
|
2022-02-11 08:20:41 -08:00
|
|
|
|
2024-12-16 09:39:25 -08:00
|
|
|
let options: IHttpRequestOptions = {
|
2022-02-11 08:20:41 -08:00
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
},
|
|
|
|
method: 'POST',
|
|
|
|
body,
|
2024-12-16 09:39:25 -08:00
|
|
|
url: endpoint,
|
2022-02-11 08:20:41 -08:00
|
|
|
json: true,
|
|
|
|
};
|
|
|
|
options = Object.assign({}, options, option);
|
|
|
|
try {
|
2024-12-16 09:39:25 -08:00
|
|
|
const response = await this.helpers.httpRequestWithAuthentication.call(
|
2023-09-20 02:19:12 -07:00
|
|
|
this,
|
|
|
|
authenticationMethod === 'apiToken' ? 'linearApi' : 'linearOAuth2Api',
|
|
|
|
options,
|
|
|
|
);
|
2024-12-16 09:39:25 -08:00
|
|
|
|
|
|
|
if (response.errors) {
|
|
|
|
throw new NodeApiError(this.getNode(), response.errors, {
|
|
|
|
message: response.errors[0].message,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return response;
|
2022-02-11 08:20:41 -08:00
|
|
|
} catch (error) {
|
2024-12-16 09:39:25 -08:00
|
|
|
throw new NodeApiError(
|
|
|
|
this.getNode(),
|
|
|
|
{},
|
|
|
|
{
|
|
|
|
message: error.errorResponse
|
|
|
|
? error.errorResponse[0].message
|
|
|
|
: error.context.data.errors[0].message,
|
|
|
|
description: error.errorResponse
|
|
|
|
? error.errorResponse[0].extensions.userPresentableMessage
|
|
|
|
: error.context.data.errors[0].extensions.userPresentableMessage,
|
|
|
|
},
|
|
|
|
);
|
2022-02-11 08:20:41 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function capitalizeFirstLetter(data: string) {
|
|
|
|
return data.charAt(0).toUpperCase() + data.slice(1);
|
2022-03-20 02:11:06 -07:00
|
|
|
}
|
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
export async function linearApiRequestAllItems(
|
|
|
|
this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions,
|
|
|
|
propertyName: string,
|
|
|
|
body: any = {},
|
2024-05-08 02:24:08 -07:00
|
|
|
limit?: number,
|
2022-08-17 08:50:24 -07:00
|
|
|
): Promise<any> {
|
2022-03-20 02:11:06 -07:00
|
|
|
const returnData: IDataObject[] = [];
|
|
|
|
|
|
|
|
let responseData;
|
2024-05-08 02:24:08 -07:00
|
|
|
body.variables.first = limit && limit < 50 ? limit : 50;
|
2022-03-20 02:11:06 -07:00
|
|
|
body.variables.after = null;
|
|
|
|
|
2024-05-08 02:24:08 -07:00
|
|
|
const propertyPath = propertyName.split('.');
|
|
|
|
const nodesPath = [...propertyPath, 'nodes'];
|
|
|
|
const endCursorPath = [...propertyPath, 'pageInfo', 'endCursor'];
|
|
|
|
const hasNextPagePath = [...propertyPath, 'pageInfo', 'hasNextPage'];
|
|
|
|
|
2022-03-20 02:11:06 -07:00
|
|
|
do {
|
|
|
|
responseData = await linearApiRequest.call(this, body);
|
2024-05-08 02:24:08 -07:00
|
|
|
const nodes = get(responseData, nodesPath) as IDataObject[];
|
|
|
|
returnData.push(...nodes);
|
|
|
|
body.variables.after = get(responseData, endCursorPath);
|
|
|
|
if (limit && returnData.length >= limit) {
|
|
|
|
return returnData;
|
|
|
|
}
|
|
|
|
} while (get(responseData, hasNextPagePath));
|
|
|
|
|
2022-03-20 02:11:06 -07:00
|
|
|
return returnData;
|
|
|
|
}
|
|
|
|
|
2022-08-17 08:50:24 -07:00
|
|
|
export async function validateCredentials(
|
|
|
|
this: ICredentialTestFunctions,
|
|
|
|
decryptedCredentials: ICredentialDataDecryptedObject,
|
|
|
|
): Promise<any> {
|
2022-03-20 02:11:06 -07:00
|
|
|
const credentials = decryptedCredentials;
|
|
|
|
|
2024-12-16 09:39:25 -08:00
|
|
|
const options: IHttpRequestOptions = {
|
2022-03-20 02:11:06 -07:00
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
Authorization: credentials.apiKey,
|
|
|
|
},
|
|
|
|
method: 'POST',
|
|
|
|
body: {
|
|
|
|
query: query.getIssues(),
|
|
|
|
variables: {
|
|
|
|
first: 1,
|
|
|
|
},
|
|
|
|
},
|
2024-12-16 09:39:25 -08:00
|
|
|
url: 'https://api.linear.app/graphql',
|
2022-03-20 02:11:06 -07:00
|
|
|
json: true,
|
|
|
|
};
|
|
|
|
|
2024-01-17 07:08:50 -08:00
|
|
|
return await this.helpers.request(options);
|
2022-03-20 02:11:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//@ts-ignore
|
|
|
|
export const sort = (a, b) => {
|
2022-08-17 08:50:24 -07:00
|
|
|
if (a.name.toLocaleLowerCase() < b.name.toLocaleLowerCase()) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (a.name.toLocaleLowerCase() > b.name.toLocaleLowerCase()) {
|
|
|
|
return 1;
|
|
|
|
}
|
2022-03-20 02:11:06 -07:00
|
|
|
return 0;
|
|
|
|
};
|