mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-24 04:04:06 -08:00
✨ Add Grist node (#2158)
* Implement Grist node with List/Append/Update/Delete operations * 🔨 Refactor Grist node * 🔨 Make API key required * 🔨 Complete create/upate operations * 🔨 Fix item index in docId and tableId * 🔨 Simplify continueOnFail item * 👕 Nodelinter pass * 👕 Fix lint * 👕 Resort imports * ⚡ Improvements * 🔨 Simplify with optional access operator * 🔨 Simplify row ID processing in deletion * 🚧 Add stub for cred test Pending change to core * ⚡ Add workaround for cred test * 🔥 Remove excess items check * ✏️ Rename fields * 🐛 Fix numeric filter * ✏️ Add feedback from Product * 🔥 Remove superfluous key * ⚡ Small change * ⚡ Fix subtitle and improve how data gets returned Co-authored-by: Alex Hall <alex.mojaki@gmail.com> Co-authored-by: ricardo <ricardoespinoza105@gmail.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
parent
a144a8e315
commit
4bce33a530
50
packages/nodes-base/credentials/GristApi.credentials.ts
Normal file
50
packages/nodes-base/credentials/GristApi.credentials.ts
Normal file
|
@ -0,0 +1,50 @@
|
|||
import {
|
||||
ICredentialType,
|
||||
INodeProperties,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
export class GristApi implements ICredentialType {
|
||||
name = 'gristApi';
|
||||
displayName = 'Grist API';
|
||||
documentationUrl = 'grist';
|
||||
properties: INodeProperties[] = [
|
||||
{
|
||||
displayName: 'API Key',
|
||||
name: 'apiKey',
|
||||
type: 'string',
|
||||
default: '',
|
||||
required: true,
|
||||
},
|
||||
{
|
||||
displayName: 'Plan Type',
|
||||
name: 'planType',
|
||||
type: 'options',
|
||||
default: 'free',
|
||||
options: [
|
||||
{
|
||||
name: 'Free',
|
||||
value: 'free',
|
||||
},
|
||||
{
|
||||
name: 'Paid',
|
||||
value: 'paid',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Custom Subdomain',
|
||||
name: 'customSubdomain',
|
||||
type: 'string',
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'Custom subdomain of your team',
|
||||
displayOptions: {
|
||||
show: {
|
||||
planType: [
|
||||
'paid',
|
||||
],
|
||||
},
|
||||
},
|
||||
},
|
||||
];
|
||||
}
|
109
packages/nodes-base/nodes/Grist/GenericFunctions.ts
Normal file
109
packages/nodes-base/nodes/Grist/GenericFunctions.ts
Normal file
|
@ -0,0 +1,109 @@
|
|||
import {
|
||||
IExecuteFunctions,
|
||||
ILoadOptionsFunctions,
|
||||
} from 'n8n-core';
|
||||
|
||||
import {
|
||||
OptionsWithUri,
|
||||
} from 'request';
|
||||
|
||||
import {
|
||||
IDataObject,
|
||||
NodeApiError,
|
||||
NodeOperationError,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import {
|
||||
GristCredentials,
|
||||
GristDefinedFields,
|
||||
GristFilterProperties,
|
||||
GristSortProperties,
|
||||
} from './types';
|
||||
|
||||
export async function gristApiRequest(
|
||||
this: IExecuteFunctions | ILoadOptionsFunctions,
|
||||
method: string,
|
||||
endpoint: string,
|
||||
body: IDataObject | number[] = {},
|
||||
qs: IDataObject = {},
|
||||
) {
|
||||
const {
|
||||
apiKey,
|
||||
planType,
|
||||
customSubdomain,
|
||||
} = await this.getCredentials('gristApi') as GristCredentials;
|
||||
|
||||
const subdomain = planType === 'free' ? 'docs' : customSubdomain;
|
||||
|
||||
const options: OptionsWithUri = {
|
||||
headers: {
|
||||
Authorization: `Bearer ${apiKey}`,
|
||||
},
|
||||
method,
|
||||
uri: `https://${subdomain}.getgrist.com/api${endpoint}`,
|
||||
qs,
|
||||
body,
|
||||
json: true,
|
||||
};
|
||||
|
||||
if (!Object.keys(body).length) {
|
||||
delete options.body;
|
||||
}
|
||||
|
||||
if (!Object.keys(qs).length) {
|
||||
delete options.qs;
|
||||
}
|
||||
|
||||
try {
|
||||
return await this.helpers.request!(options);
|
||||
} catch (error) {
|
||||
throw new NodeApiError(this.getNode(), error);
|
||||
}
|
||||
}
|
||||
|
||||
export function parseSortProperties(sortProperties: GristSortProperties) {
|
||||
return sortProperties.reduce((acc, cur, curIdx) => {
|
||||
if (cur.direction === 'desc') acc += '-';
|
||||
acc += cur.field;
|
||||
if (curIdx !== sortProperties.length - 1) acc += ',';
|
||||
return acc;
|
||||
}, '');
|
||||
}
|
||||
|
||||
export function parseFilterProperties(filterProperties: GristFilterProperties) {
|
||||
return filterProperties.reduce<{ [key: string]: Array<string | number>; }>((acc, cur) => {
|
||||
acc[cur.field] = acc[cur.field] ?? [];
|
||||
const values = isNaN(Number(cur.values)) ? cur.values : Number(cur.values);
|
||||
acc[cur.field].push(values);
|
||||
return acc;
|
||||
}, {});
|
||||
}
|
||||
|
||||
export function parseDefinedFields(fieldsToSendProperties: GristDefinedFields) {
|
||||
return fieldsToSendProperties.reduce<{ [key: string]: string; }>((acc, cur) => {
|
||||
acc[cur.fieldId] = cur.fieldValue;
|
||||
return acc;
|
||||
}, {});
|
||||
}
|
||||
|
||||
export function parseAutoMappedInputs(
|
||||
incomingKeys: string[],
|
||||
inputsToIgnore: string[],
|
||||
item: any, // tslint:disable-line:no-any
|
||||
) {
|
||||
return incomingKeys.reduce<{ [key: string]: any; }>((acc, curKey) => { // tslint:disable-line:no-any
|
||||
if (inputsToIgnore.includes(curKey)) return acc;
|
||||
acc = { ...acc, [curKey]: item[curKey] };
|
||||
return acc;
|
||||
}, {});
|
||||
}
|
||||
|
||||
export function throwOnZeroDefinedFields(this: IExecuteFunctions, fields: GristDefinedFields) {
|
||||
if (!fields?.length) {
|
||||
throw new NodeOperationError(
|
||||
this.getNode(),
|
||||
'No defined data found. Please specify the data to send in \'Fields to Send\'.',
|
||||
);
|
||||
}
|
||||
}
|
||||
|
20
packages/nodes-base/nodes/Grist/Grist.node.json
Normal file
20
packages/nodes-base/nodes/Grist/Grist.node.json
Normal file
|
@ -0,0 +1,20 @@
|
|||
{
|
||||
"node": "n8n-nodes-base.grist",
|
||||
"nodeVersion": "1.0",
|
||||
"codexVersion": "1.0",
|
||||
"categories": [
|
||||
"Data & Storage"
|
||||
],
|
||||
"resources": {
|
||||
"credentialDocumentation": [
|
||||
{
|
||||
"url": "https://docs.n8n.io/credentials/grist"
|
||||
}
|
||||
],
|
||||
"primaryDocumentation": [
|
||||
{
|
||||
"url": "https://docs.n8n.io/nodes/n8n-nodes-base.grist/"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
280
packages/nodes-base/nodes/Grist/Grist.node.ts
Normal file
280
packages/nodes-base/nodes/Grist/Grist.node.ts
Normal file
|
@ -0,0 +1,280 @@
|
|||
import {
|
||||
IExecuteFunctions
|
||||
} from 'n8n-core';
|
||||
|
||||
import {
|
||||
ICredentialsDecrypted,
|
||||
ICredentialTestFunctions,
|
||||
IDataObject,
|
||||
ILoadOptionsFunctions,
|
||||
INodeExecutionData,
|
||||
INodeType,
|
||||
INodeTypeDescription,
|
||||
NodeCredentialTestResult,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import {
|
||||
OptionsWithUri,
|
||||
} from 'request';
|
||||
|
||||
import {
|
||||
gristApiRequest,
|
||||
parseAutoMappedInputs,
|
||||
parseDefinedFields,
|
||||
parseFilterProperties,
|
||||
parseSortProperties,
|
||||
throwOnZeroDefinedFields,
|
||||
} from './GenericFunctions';
|
||||
|
||||
import {
|
||||
operationFields,
|
||||
} from './OperationDescription';
|
||||
|
||||
import {
|
||||
FieldsToSend,
|
||||
GristColumns,
|
||||
GristCreateRowPayload,
|
||||
GristCredentials,
|
||||
GristGetAllOptions,
|
||||
GristUpdateRowPayload,
|
||||
SendingOptions,
|
||||
} from './types';
|
||||
|
||||
export class Grist implements INodeType {
|
||||
description: INodeTypeDescription = {
|
||||
displayName: 'Grist',
|
||||
name: 'grist',
|
||||
icon: 'file:grist.svg',
|
||||
subtitle: '={{$parameter["operation"]}}',
|
||||
group: ['input'],
|
||||
version: 1,
|
||||
description: 'Consume the Grist API',
|
||||
defaults: {
|
||||
name: 'Grist',
|
||||
color: '#394650',
|
||||
},
|
||||
inputs: ['main'],
|
||||
outputs: ['main'],
|
||||
credentials: [
|
||||
{
|
||||
name: 'gristApi',
|
||||
required: true,
|
||||
testedBy: 'gristApiTest',
|
||||
},
|
||||
],
|
||||
properties: operationFields,
|
||||
};
|
||||
|
||||
methods = {
|
||||
loadOptions: {
|
||||
async getTableColumns(this: ILoadOptionsFunctions) {
|
||||
const docId = this.getNodeParameter('docId', 0) as string;
|
||||
const tableId = this.getNodeParameter('tableId', 0) as string;
|
||||
const endpoint = `/docs/${docId}/tables/${tableId}/columns`;
|
||||
|
||||
const { columns } = await gristApiRequest.call(this, 'GET', endpoint) as GristColumns;
|
||||
return columns.map(({ id }) => ({ name: id, value: id }));
|
||||
},
|
||||
},
|
||||
|
||||
credentialTest: {
|
||||
async gristApiTest(
|
||||
this: ICredentialTestFunctions,
|
||||
credential: ICredentialsDecrypted,
|
||||
): Promise<NodeCredentialTestResult> {
|
||||
const {
|
||||
apiKey,
|
||||
planType,
|
||||
customSubdomain,
|
||||
} = credential.data as GristCredentials;
|
||||
|
||||
const subdomain = planType === 'free' ? 'docs' : customSubdomain;
|
||||
|
||||
const endpoint = '/orgs';
|
||||
|
||||
const options: OptionsWithUri = {
|
||||
headers: {
|
||||
Authorization: `Bearer ${apiKey}`,
|
||||
},
|
||||
method: 'GET',
|
||||
uri: `https://${subdomain}.getgrist.com/api${endpoint}`,
|
||||
qs: { limit: 1 },
|
||||
json: true,
|
||||
};
|
||||
|
||||
try {
|
||||
await this.helpers.request(options);
|
||||
return {
|
||||
status: 'OK',
|
||||
message: 'Authentication successful',
|
||||
};
|
||||
} catch (error) {
|
||||
return {
|
||||
status: 'Error',
|
||||
message: error.message,
|
||||
};
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||
const items = this.getInputData();
|
||||
let responseData;
|
||||
const returnData: IDataObject[] = [];
|
||||
|
||||
const operation = this.getNodeParameter('operation', 0) as string;
|
||||
|
||||
for (let i = 0; i < items.length; i++) {
|
||||
|
||||
try {
|
||||
|
||||
if (operation === 'create') {
|
||||
|
||||
// ----------------------------------
|
||||
// create
|
||||
// ----------------------------------
|
||||
|
||||
// https://support.getgrist.com/api/#tag/records/paths/~1docs~1{docId}~1tables~1{tableId}~1records/post
|
||||
|
||||
const body = { records: [] } as GristCreateRowPayload;
|
||||
|
||||
const dataToSend = this.getNodeParameter('dataToSend', 0) as SendingOptions;
|
||||
|
||||
if (dataToSend === 'autoMapInputs') {
|
||||
|
||||
const incomingKeys = Object.keys(items[i].json);
|
||||
const rawInputsToIgnore = this.getNodeParameter('inputsToIgnore', i) as string;
|
||||
const inputsToIgnore = rawInputsToIgnore.split(',').map(c => c.trim());
|
||||
const fields = parseAutoMappedInputs(incomingKeys, inputsToIgnore, items[i].json);
|
||||
body.records.push({ fields });
|
||||
|
||||
} else if (dataToSend === 'defineInNode') {
|
||||
|
||||
const { properties } = this.getNodeParameter('fieldsToSend', i, []) as FieldsToSend;
|
||||
throwOnZeroDefinedFields.call(this, properties);
|
||||
body.records.push({ fields: parseDefinedFields(properties) });
|
||||
|
||||
}
|
||||
|
||||
const docId = this.getNodeParameter('docId', 0) as string;
|
||||
const tableId = this.getNodeParameter('tableId', 0) as string;
|
||||
const endpoint = `/docs/${docId}/tables/${tableId}/records`;
|
||||
|
||||
responseData = await gristApiRequest.call(this, 'POST', endpoint, body);
|
||||
responseData = {
|
||||
id: responseData.records[0].id,
|
||||
...body.records[0].fields,
|
||||
};
|
||||
|
||||
} else if (operation === 'delete') {
|
||||
|
||||
// ----------------------------------
|
||||
// delete
|
||||
// ----------------------------------
|
||||
|
||||
// https://support.getgrist.com/api/#tag/data/paths/~1docs~1{docId}~1tables~1{tableId}~1data~1delete/post
|
||||
|
||||
const docId = this.getNodeParameter('docId', 0) as string;
|
||||
const tableId = this.getNodeParameter('tableId', 0) as string;
|
||||
const endpoint = `/docs/${docId}/tables/${tableId}/data/delete`;
|
||||
|
||||
const rawRowIds = (this.getNodeParameter('rowId', i) as string).toString();
|
||||
const body = rawRowIds.split(',').map(c => c.trim()).map(Number);
|
||||
|
||||
await gristApiRequest.call(this, 'POST', endpoint, body);
|
||||
responseData = { success: true };
|
||||
|
||||
} else if (operation === 'update') {
|
||||
|
||||
// ----------------------------------
|
||||
// update
|
||||
// ----------------------------------
|
||||
|
||||
// https://support.getgrist.com/api/#tag/records/paths/~1docs~1{docId}~1tables~1{tableId}~1records/patch
|
||||
|
||||
const body = { records: [] } as GristUpdateRowPayload;
|
||||
|
||||
const rowId = this.getNodeParameter('rowId', i) as string;
|
||||
const dataToSend = this.getNodeParameter('dataToSend', 0) as SendingOptions;
|
||||
|
||||
if (dataToSend === 'autoMapInputs') {
|
||||
|
||||
const incomingKeys = Object.keys(items[i].json);
|
||||
const rawInputsToIgnore = this.getNodeParameter('inputsToIgnore', i) as string;
|
||||
const inputsToIgnore = rawInputsToIgnore.split(',').map(c => c.trim());
|
||||
const fields = parseAutoMappedInputs(incomingKeys, inputsToIgnore, items[i].json);
|
||||
body.records.push({ id: Number(rowId), fields });
|
||||
|
||||
} else if (dataToSend === 'defineInNode') {
|
||||
|
||||
const { properties } = this.getNodeParameter('fieldsToSend', i, []) as FieldsToSend;
|
||||
throwOnZeroDefinedFields.call(this, properties);
|
||||
const fields = parseDefinedFields(properties);
|
||||
body.records.push({ id: Number(rowId), fields });
|
||||
|
||||
}
|
||||
|
||||
const docId = this.getNodeParameter('docId', 0) as string;
|
||||
const tableId = this.getNodeParameter('tableId', 0) as string;
|
||||
const endpoint = `/docs/${docId}/tables/${tableId}/records`;
|
||||
|
||||
await gristApiRequest.call(this, 'PATCH', endpoint, body);
|
||||
responseData = {
|
||||
id: rowId,
|
||||
...body.records[0].fields,
|
||||
};
|
||||
|
||||
} else if (operation === 'getAll') {
|
||||
|
||||
// ----------------------------------
|
||||
// getAll
|
||||
// ----------------------------------
|
||||
|
||||
// https://support.getgrist.com/api/#tag/records
|
||||
|
||||
const docId = this.getNodeParameter('docId', 0) as string;
|
||||
const tableId = this.getNodeParameter('tableId', 0) as string;
|
||||
const endpoint = `/docs/${docId}/tables/${tableId}/records`;
|
||||
|
||||
const qs: IDataObject = {};
|
||||
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
|
||||
if (!returnAll) {
|
||||
qs.limit = this.getNodeParameter('limit', i) as number;
|
||||
}
|
||||
|
||||
const { sort, filter } = this.getNodeParameter('additionalOptions', i) as GristGetAllOptions;
|
||||
|
||||
if (sort?.sortProperties.length) {
|
||||
qs.sort = parseSortProperties(sort.sortProperties);
|
||||
}
|
||||
|
||||
if (filter?.filterProperties.length) {
|
||||
const parsed = parseFilterProperties(filter.filterProperties);
|
||||
qs.filter = JSON.stringify(parsed);
|
||||
}
|
||||
|
||||
responseData = await gristApiRequest.call(this, 'GET', endpoint, {}, qs);
|
||||
responseData = responseData.records.map((data: IDataObject) => {
|
||||
return { id: data.id, ...(data.fields as object) };
|
||||
});
|
||||
}
|
||||
|
||||
} catch (error) {
|
||||
if (this.continueOnFail()) {
|
||||
returnData.push({ error: error.message });
|
||||
continue;
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
Array.isArray(responseData)
|
||||
? returnData.push(...responseData)
|
||||
: returnData.push(responseData);
|
||||
}
|
||||
|
||||
return [this.helpers.returnJsonArray(returnData)];
|
||||
}
|
||||
}
|
338
packages/nodes-base/nodes/Grist/OperationDescription.ts
Normal file
338
packages/nodes-base/nodes/Grist/OperationDescription.ts
Normal file
|
@ -0,0 +1,338 @@
|
|||
import {
|
||||
INodeProperties,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
export const operationFields: INodeProperties[] = [
|
||||
{
|
||||
displayName: 'Operation',
|
||||
name: 'operation',
|
||||
type: 'options',
|
||||
noDataExpression: true,
|
||||
options: [
|
||||
{
|
||||
name: 'Create Row',
|
||||
value: 'create',
|
||||
description: 'Create rows in a table',
|
||||
},
|
||||
{
|
||||
name: 'Delete Row',
|
||||
value: 'delete',
|
||||
description: 'Delete rows from a table',
|
||||
},
|
||||
{
|
||||
name: 'Get All Rows',
|
||||
value: 'getAll',
|
||||
description: 'Read rows from a table',
|
||||
},
|
||||
{
|
||||
name: 'Update Row',
|
||||
value: 'update',
|
||||
description: 'Update rows in a table',
|
||||
},
|
||||
],
|
||||
default: 'getAll',
|
||||
},
|
||||
|
||||
// ----------------------------------
|
||||
// shared
|
||||
// ----------------------------------
|
||||
{
|
||||
displayName: 'Document ID',
|
||||
name: 'docId',
|
||||
type: 'string',
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'In your document, click your profile icon, then Document Settings, then copy the value under "This document\'s ID"',
|
||||
},
|
||||
{
|
||||
displayName: 'Table ID',
|
||||
name: 'tableId',
|
||||
type: 'string',
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'ID of table to operate on. If unsure, look at the Code View.',
|
||||
},
|
||||
|
||||
// ----------------------------------
|
||||
// delete
|
||||
// ----------------------------------
|
||||
{
|
||||
displayName: 'Row ID',
|
||||
name: 'rowId',
|
||||
type: 'string',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'delete',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: '',
|
||||
description: 'ID of the row to delete, or comma-separated list of row IDs to delete',
|
||||
required: true,
|
||||
},
|
||||
|
||||
// ----------------------------------
|
||||
// getAll
|
||||
// ----------------------------------
|
||||
{
|
||||
displayName: 'Return All',
|
||||
name: 'returnAll',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
description: 'Whether to return all results or only up to a given limit',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'getAll',
|
||||
],
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
displayName: 'Limit',
|
||||
name: 'limit',
|
||||
type: 'number',
|
||||
typeOptions: {
|
||||
minValue: 1,
|
||||
},
|
||||
default: 50,
|
||||
description: 'Max number of results to return',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'getAll',
|
||||
],
|
||||
returnAll: [
|
||||
false,
|
||||
],
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
displayName: 'Additional Options',
|
||||
name: 'additionalOptions',
|
||||
type: 'collection',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'getAll',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: {},
|
||||
placeholder: 'Add Option',
|
||||
options: [
|
||||
{
|
||||
displayName: 'Filter',
|
||||
name: 'filter',
|
||||
placeholder: 'Add Filter',
|
||||
description: 'Only return rows matching all of the given filters. For complex filters, create a formula column and filter for the value "true".',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: true,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
displayName: 'Filter Properties',
|
||||
name: 'filterProperties',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Column',
|
||||
name: 'field',
|
||||
type: 'options',
|
||||
typeOptions: {
|
||||
loadOptionsDependsOn: [
|
||||
'docId',
|
||||
'tableId',
|
||||
],
|
||||
loadOptionsMethod: 'getTableColumns',
|
||||
},
|
||||
default: '',
|
||||
description: 'Column to apply the filter in',
|
||||
required: true,
|
||||
},
|
||||
{
|
||||
displayName: 'Values',
|
||||
name: 'values',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Comma-separated list of values to search for in the filtered column',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Sort Order',
|
||||
name: 'sort',
|
||||
placeholder: 'Add Field',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: true,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
displayName: 'Sort Properties',
|
||||
name: 'sortProperties',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Column',
|
||||
name: 'field',
|
||||
type: 'options',
|
||||
typeOptions: {
|
||||
loadOptionsDependsOn: [
|
||||
'docId',
|
||||
'tableId',
|
||||
],
|
||||
loadOptionsMethod: 'getTableColumns',
|
||||
},
|
||||
default: '',
|
||||
required: true,
|
||||
description: 'Column to sort on',
|
||||
},
|
||||
{
|
||||
displayName: 'Direction',
|
||||
name: 'direction',
|
||||
type: 'options',
|
||||
options: [
|
||||
{
|
||||
name: 'Ascending',
|
||||
value: 'asc',
|
||||
},
|
||||
{
|
||||
name: 'Descending',
|
||||
value: 'desc',
|
||||
},
|
||||
],
|
||||
default: 'asc',
|
||||
description: 'Direction to sort in',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
|
||||
// ----------------------------------
|
||||
// update
|
||||
// ----------------------------------
|
||||
{
|
||||
displayName: 'Row ID',
|
||||
name: 'rowId',
|
||||
type: 'string',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'update',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: '',
|
||||
description: 'ID of the row to update',
|
||||
required: true,
|
||||
},
|
||||
|
||||
// ----------------------------------
|
||||
// create + update
|
||||
// ----------------------------------
|
||||
{
|
||||
displayName: 'Data to Send',
|
||||
name: 'dataToSend',
|
||||
type: 'options',
|
||||
options: [
|
||||
{
|
||||
name: 'Auto-Map Input Data to Columns',
|
||||
value: 'autoMapInputs',
|
||||
description: 'Use when node input properties match destination column names',
|
||||
},
|
||||
{
|
||||
name: 'Define Below for Each Column',
|
||||
value: 'defineInNode',
|
||||
description: 'Set the value for each destination column',
|
||||
},
|
||||
],
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'create',
|
||||
'update',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: 'defineInNode',
|
||||
description: 'Whether to insert the input data this node receives in the new row',
|
||||
},
|
||||
{
|
||||
displayName: 'Inputs to Ignore',
|
||||
name: 'inputsToIgnore',
|
||||
type: 'string',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'create',
|
||||
'update',
|
||||
],
|
||||
dataToSend: [
|
||||
'autoMapInputs',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: '',
|
||||
required: false,
|
||||
description: 'List of input properties to avoid sending, separated by commas. Leave empty to send all properties.',
|
||||
placeholder: 'Enter properties...',
|
||||
},
|
||||
{
|
||||
displayName: 'Fields to Send',
|
||||
name: 'fieldsToSend',
|
||||
placeholder: 'Add Field',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValueButtonText: 'Add Field to Send',
|
||||
multipleValues: true,
|
||||
},
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: [
|
||||
'create',
|
||||
'update',
|
||||
],
|
||||
dataToSend: [
|
||||
'defineInNode',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
displayName: 'Properties',
|
||||
name: 'properties',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Column Name/ID',
|
||||
name: 'fieldId',
|
||||
description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/nodes/expressions.html#expressions">expression</a>',
|
||||
type: 'options',
|
||||
typeOptions: {
|
||||
loadOptionsDependsOn: [
|
||||
'tableId',
|
||||
],
|
||||
loadOptionsMethod: 'getTableColumns',
|
||||
},
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Field Value',
|
||||
name: 'fieldValue',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
];
|
1
packages/nodes-base/nodes/Grist/grist.svg
Normal file
1
packages/nodes-base/nodes/Grist/grist.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 21 KiB |
46
packages/nodes-base/nodes/Grist/types.d.ts
vendored
Normal file
46
packages/nodes-base/nodes/Grist/types.d.ts
vendored
Normal file
|
@ -0,0 +1,46 @@
|
|||
export type GristCredentials = {
|
||||
apiKey: string;
|
||||
planType: 'free' | 'paid';
|
||||
customSubdomain?: string;
|
||||
}
|
||||
|
||||
export type GristColumns = {
|
||||
columns: Array<{ id: string }>;
|
||||
};
|
||||
|
||||
export type GristSortProperties = Array<{
|
||||
field: string;
|
||||
direction: 'asc' | 'desc';
|
||||
}>;
|
||||
|
||||
export type GristFilterProperties = Array<{
|
||||
field: string;
|
||||
values: string;
|
||||
}>;
|
||||
|
||||
export type GristGetAllOptions = {
|
||||
sort?: { sortProperties: GristSortProperties };
|
||||
filter?: { filterProperties: GristFilterProperties };
|
||||
};
|
||||
|
||||
export type GristDefinedFields = Array<{
|
||||
fieldId: string;
|
||||
fieldValue: string;
|
||||
}>;
|
||||
|
||||
export type GristCreateRowPayload = {
|
||||
records: Array<{
|
||||
fields: { [key: string]: any };
|
||||
}>
|
||||
};
|
||||
|
||||
export type GristUpdateRowPayload = {
|
||||
records: Array<{
|
||||
id: number;
|
||||
fields: { [key: string]: any };
|
||||
}>
|
||||
}
|
||||
|
||||
export type SendingOptions = 'defineInNode' | 'autoMapInputs';
|
||||
|
||||
export type FieldsToSend = { properties: GristDefinedFields; };
|
|
@ -126,6 +126,7 @@
|
|||
"dist/credentials/GoogleTranslateOAuth2Api.credentials.js",
|
||||
"dist/credentials/GotifyApi.credentials.js",
|
||||
"dist/credentials/GoToWebinarOAuth2Api.credentials.js",
|
||||
"dist/credentials/GristApi.credentials.js",
|
||||
"dist/credentials/YouTubeOAuth2Api.credentials.js",
|
||||
"dist/credentials/GumroadApi.credentials.js",
|
||||
"dist/credentials/HarvestApi.credentials.js",
|
||||
|
@ -430,6 +431,7 @@
|
|||
"dist/nodes/Gotify/Gotify.node.js",
|
||||
"dist/nodes/GoToWebinar/GoToWebinar.node.js",
|
||||
"dist/nodes/GraphQL/GraphQL.node.js",
|
||||
"dist/nodes/Grist/Grist.node.js",
|
||||
"dist/nodes/Gumroad/GumroadTrigger.node.js",
|
||||
"dist/nodes/HackerNews/HackerNews.node.js",
|
||||
"dist/nodes/Harvest/Harvest.node.js",
|
||||
|
|
Loading…
Reference in a new issue