mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-07 10:57:29 -08:00
7ce7285f7a
* Changes to types so that credentials can be always loaded from DB This first commit changes all return types from the execute functions and calls to get credentials to be async so we can use await. This is a first step as previously credentials were loaded in memory and always available. We will now be loading them from the DB which requires turning the whole call chain async. * Fix updated files * Removed unnecessary credential loading to improve performance * Fix typo * ⚡ Fix issue * Updated new nodes to load credentials async * ⚡ Remove not needed comment Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
328 lines
7.6 KiB
TypeScript
328 lines
7.6 KiB
TypeScript
import { IExecuteFunctions } from 'n8n-core';
|
|
import {
|
|
IDataObject,
|
|
INodeExecutionData,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
NodeApiError,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import { OptionsWithUri } from 'request';
|
|
import { RequestPromiseOptions } from 'request-promise-native';
|
|
|
|
export class GraphQL implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'GraphQL',
|
|
name: 'graphql',
|
|
icon: 'file:graphql.png',
|
|
group: ['input'],
|
|
version: 1,
|
|
description: 'Makes a GraphQL request and returns the received data',
|
|
defaults: {
|
|
name: 'GraphQL',
|
|
color: '#E10098',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
{
|
|
name: 'httpHeaderAuth',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
authentication: [
|
|
'headerAuth',
|
|
],
|
|
},
|
|
},
|
|
},
|
|
],
|
|
properties: [
|
|
{
|
|
displayName: 'Authentication',
|
|
name: 'authentication',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Header Auth',
|
|
value: 'headerAuth',
|
|
},
|
|
{
|
|
name: 'None',
|
|
value: 'none',
|
|
},
|
|
],
|
|
default: 'none',
|
|
description: 'The way to authenticate.',
|
|
},
|
|
{
|
|
displayName: 'HTTP Request Method',
|
|
name: 'requestMethod',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'GET',
|
|
value: 'GET',
|
|
},
|
|
{
|
|
name: 'POST',
|
|
value: 'POST',
|
|
},
|
|
],
|
|
default: 'POST',
|
|
description: 'The underlying HTTP request method to use.',
|
|
},
|
|
{
|
|
displayName: 'Endpoint',
|
|
name: 'endpoint',
|
|
type: 'string',
|
|
default: '',
|
|
placeholder: 'http://example.com/graphql',
|
|
description: 'The GraphQL endpoint.',
|
|
required: true,
|
|
},
|
|
{
|
|
displayName: 'Ignore SSL Issues',
|
|
name: 'allowUnauthorizedCerts',
|
|
type: 'boolean',
|
|
default: false,
|
|
description: 'Still fetch the response even if SSL certificate validation is not possible.',
|
|
},
|
|
{
|
|
displayName: 'Request Format',
|
|
name: 'requestFormat',
|
|
type: 'options',
|
|
required: true,
|
|
options: [
|
|
{
|
|
name: 'GraphQL (raw)',
|
|
value: 'graphql',
|
|
},
|
|
{
|
|
name: 'JSON',
|
|
value: 'json',
|
|
},
|
|
],
|
|
displayOptions: {
|
|
show: {
|
|
requestMethod: [
|
|
'POST',
|
|
],
|
|
},
|
|
},
|
|
default: 'graphql',
|
|
description: 'The format for the query payload',
|
|
},
|
|
{
|
|
displayName: 'Query',
|
|
name: 'query',
|
|
type: 'json',
|
|
default: '',
|
|
description: 'GraphQL query',
|
|
required: true,
|
|
},
|
|
{
|
|
displayName: 'Variables',
|
|
name: 'variables',
|
|
type: 'json',
|
|
default: '',
|
|
description: 'Query variables',
|
|
displayOptions: {
|
|
show: {
|
|
requestFormat: [
|
|
'json',
|
|
],
|
|
requestMethod: [
|
|
'POST',
|
|
],
|
|
},
|
|
},
|
|
},
|
|
{
|
|
displayName: 'Operation Name',
|
|
name: 'operationName',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Name of operation to execute',
|
|
displayOptions: {
|
|
show: {
|
|
requestFormat: [
|
|
'json',
|
|
],
|
|
requestMethod: [
|
|
'POST',
|
|
],
|
|
},
|
|
},
|
|
},
|
|
{
|
|
displayName: 'Response Format',
|
|
name: 'responseFormat',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'JSON',
|
|
value: 'json',
|
|
},
|
|
{
|
|
name: 'String',
|
|
value: 'string',
|
|
},
|
|
],
|
|
default: 'json',
|
|
description: 'The format in which the data gets returned from the URL.',
|
|
},
|
|
{
|
|
displayName: 'Response Data Property Name',
|
|
name: 'dataPropertyName',
|
|
type: 'string',
|
|
default: 'data',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
responseFormat: [
|
|
'string',
|
|
],
|
|
},
|
|
},
|
|
description: 'Name of the property to which to write the response data.',
|
|
},
|
|
|
|
// Header Parameters
|
|
{
|
|
displayName: 'Headers',
|
|
name: 'headerParametersUi',
|
|
placeholder: 'Add Header',
|
|
type: 'fixedCollection',
|
|
typeOptions: {
|
|
multipleValues: true,
|
|
},
|
|
description: 'The headers to send.',
|
|
default: {},
|
|
options: [
|
|
{
|
|
name: 'parameter',
|
|
displayName: 'Header',
|
|
values: [
|
|
{
|
|
displayName: 'Name',
|
|
name: 'name',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Name of the header.',
|
|
},
|
|
{
|
|
displayName: 'Value',
|
|
name: 'value',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Value to set for the header.',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
};
|
|
|
|
|
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
|
|
const items = this.getInputData();
|
|
const httpHeaderAuth = await this.getCredentials('httpHeaderAuth');
|
|
|
|
let requestOptions: OptionsWithUri & RequestPromiseOptions;
|
|
|
|
const returnItems: INodeExecutionData[] = [];
|
|
for (let itemIndex = 0; itemIndex < items.length; itemIndex++) {
|
|
try {
|
|
const requestMethod = this.getNodeParameter('requestMethod', itemIndex, 'POST') as string;
|
|
const endpoint = this.getNodeParameter('endpoint', itemIndex, '') as string;
|
|
const requestFormat = this.getNodeParameter('requestFormat', itemIndex, 'graphql') as string;
|
|
const responseFormat = this.getNodeParameter('responseFormat', 0) as string;
|
|
|
|
const { parameter }: { parameter?: Array<{ name: string, value: string }> } = this
|
|
.getNodeParameter('headerParametersUi', itemIndex, {}) as IDataObject;
|
|
const headerParameters = (parameter || []).reduce((result, item) => ({
|
|
...result,
|
|
[item.name]: item.value,
|
|
}), {});
|
|
|
|
requestOptions = {
|
|
headers: {
|
|
'content-type': `application/${requestFormat}`,
|
|
...headerParameters,
|
|
},
|
|
method: requestMethod,
|
|
uri: endpoint,
|
|
simple: false,
|
|
rejectUnauthorized: !this.getNodeParameter('allowUnauthorizedCerts', itemIndex, false) as boolean,
|
|
};
|
|
|
|
// Add credentials if any are set
|
|
if (httpHeaderAuth !== undefined) {
|
|
requestOptions.headers![httpHeaderAuth.name as string] = httpHeaderAuth.value;
|
|
}
|
|
|
|
const gqlQuery = this.getNodeParameter('query', itemIndex, '') as string;
|
|
if (requestMethod === 'GET') {
|
|
requestOptions.qs = {
|
|
query: gqlQuery,
|
|
};
|
|
} else {
|
|
if (requestFormat === 'json') {
|
|
requestOptions.body = {
|
|
query: gqlQuery,
|
|
variables: this.getNodeParameter('variables', itemIndex, {}) as object,
|
|
operationName: this.getNodeParameter('operationName', itemIndex) as string,
|
|
};
|
|
if (typeof requestOptions.body.variables === 'string') {
|
|
try {
|
|
requestOptions.body.variables = JSON.parse(requestOptions.body.variables || '{}');
|
|
} catch (error) {
|
|
throw new NodeOperationError(this.getNode(), 'Using variables failed:\n' + requestOptions.body.variables + '\n\nWith error message:\n' + error);
|
|
}
|
|
}
|
|
if (requestOptions.body.operationName === '') {
|
|
requestOptions.body.operationName = null;
|
|
}
|
|
requestOptions.json = true;
|
|
} else {
|
|
requestOptions.body = gqlQuery;
|
|
}
|
|
}
|
|
|
|
const response = await this.helpers.request(requestOptions);
|
|
if (responseFormat === 'string') {
|
|
const dataPropertyName = this.getNodeParameter('dataPropertyName', 0) as string;
|
|
|
|
returnItems.push({
|
|
json: {
|
|
[dataPropertyName]: response,
|
|
},
|
|
});
|
|
} else {
|
|
if (typeof response === 'string') {
|
|
try {
|
|
returnItems.push({ json: JSON.parse(response) });
|
|
} catch (error) {
|
|
throw new NodeOperationError(this.getNode(), 'Response body is not valid JSON. Change "Response Format" to "String"');
|
|
}
|
|
} else {
|
|
returnItems.push({ json: response });
|
|
}
|
|
}
|
|
} catch (error) {
|
|
if (this.continueOnFail()) {
|
|
returnItems.push({ json: { error: error.message } });
|
|
continue;
|
|
}
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
return this.prepareOutputData(returnItems);
|
|
}
|
|
}
|