diff --git a/packages/editor-ui/src/components/mixins/moveNodeWorkflow.ts b/packages/editor-ui/src/components/mixins/moveNodeWorkflow.ts index b1250003fd..7aca13623d 100644 --- a/packages/editor-ui/src/components/mixins/moveNodeWorkflow.ts +++ b/packages/editor-ui/src/components/mixins/moveNodeWorkflow.ts @@ -33,7 +33,7 @@ export const moveNodeWorkflow = mixins( const nodeViewOffsetPositionX = offsetPosition[0] + (position.x - this.moveLastPosition[0]); const nodeViewOffsetPositionY = offsetPosition[1] + (position.y - this.moveLastPosition[1]); - this.$store.commit('setNodeViewOffsetPosition', {newOffset: [nodeViewOffsetPositionX, nodeViewOffsetPositionY], setStateDirty: true}); + this.$store.commit('setNodeViewOffsetPosition', {newOffset: [nodeViewOffsetPositionX, nodeViewOffsetPositionY]}); // Update the last position this.moveLastPosition[0] = position.x; @@ -101,7 +101,7 @@ export const moveNodeWorkflow = mixins( const offsetPosition = this.$store.getters.getNodeViewOffsetPosition; const nodeViewOffsetPositionX = offsetPosition[0] - normalized.pixelX; const nodeViewOffsetPositionY = offsetPosition[1] - normalized.pixelY; - this.$store.commit('setNodeViewOffsetPosition', {newOffset: [nodeViewOffsetPositionX, nodeViewOffsetPositionY], setStateDirty: true}); + this.$store.commit('setNodeViewOffsetPosition', {newOffset: [nodeViewOffsetPositionX, nodeViewOffsetPositionY]}); }, }, }); diff --git a/packages/editor-ui/src/components/mixins/workflowRun.ts b/packages/editor-ui/src/components/mixins/workflowRun.ts index 08ac66a110..ea375f7399 100644 --- a/packages/editor-ui/src/components/mixins/workflowRun.ts +++ b/packages/editor-ui/src/components/mixins/workflowRun.ts @@ -29,7 +29,6 @@ export const workflowRun = mixins( // because then it can not receive the data as it executes. throw new Error('No active connection to server. It is maybe down.'); } - const workflow = this.getWorkflow(); this.$store.commit('addActiveAction', 'workflowRunning'); diff --git a/packages/editor-ui/src/store.ts b/packages/editor-ui/src/store.ts index 0ad38d81f3..8275908fc0 100644 --- a/packages/editor-ui/src/store.ts +++ b/packages/editor-ui/src/store.ts @@ -96,7 +96,6 @@ export const store = new Vuex.Store({ // Active Executions addActiveExecution (state, newActiveExecution: IExecutionsCurrentSummaryExtended) { - state.stateIsDirty = true; // Check if the execution exists already const activeExecution = state.activeExecutions.find(execution => { return execution.idActive === newActiveExecution.idActive; @@ -113,7 +112,6 @@ export const store = new Vuex.Store({ state.activeExecutions.unshift(newActiveExecution); }, finishActiveExecution (state, finishedActiveExecution: IPushDataExecutionFinished) { - state.stateIsDirty = true; // Find the execution to set to finished const activeExecution = state.activeExecutions.find(execution => { return execution.idActive === finishedActiveExecution.executionIdActive; @@ -132,7 +130,6 @@ export const store = new Vuex.Store({ Vue.set(activeExecution, 'stoppedAt', finishedActiveExecution.data.stoppedAt); }, setActiveExecutions (state, newActiveExecutions: IExecutionsCurrentSummaryExtended[]) { - state.stateIsDirty = true; Vue.set(state, 'activeExecutions', newActiveExecutions); }, @@ -165,7 +162,6 @@ export const store = new Vuex.Store({ state.selectedNodes.push(node); }, removeNodeFromSelection (state, node: INodeUi) { - state.stateIsDirty = true; let index; for (index in state.selectedNodes) { if (state.selectedNodes[index].name === node.name) { @@ -377,7 +373,6 @@ export const store = new Vuex.Store({ // Set/Overwrite the value Vue.set(node.issues!, nodeIssueData.type, nodeIssueData.value); - state.stateIsDirty = true; } return true; @@ -466,7 +461,6 @@ export const store = new Vuex.Store({ state.nodeIndex.push(nodeName); }, setNodeIndex (state, newData: { index: number, name: string | null}) { - state.stateIsDirty = true; state.nodeIndex[newData.index] = newData.name; }, resetNodeIndex (state) { @@ -478,9 +472,6 @@ export const store = new Vuex.Store({ state.nodeViewMoveInProgress = value; }, setNodeViewOffsetPosition (state, data) { - if (data.setStateDirty === true) { - state.stateIsDirty = true; - } state.nodeViewOffsetPosition = data.newOffset; }, @@ -541,16 +532,6 @@ export const store = new Vuex.Store({ Vue.set(state, 'oauthCallbackUrls', urls); }, - addNodeType (state, typeData: INodeTypeDescription) { - if (!typeData.hasOwnProperty('name')) { - // All node-types have to have a name - // TODO: Check if there is an error or whatever that is supposed to be returned - return; - } - state.stateIsDirty = true; - state.nodeTypes.push(typeData); - }, - setActiveNode (state, nodeName: string) { state.activeNode = nodeName; }, @@ -573,7 +554,6 @@ export const store = new Vuex.Store({ if (state.workflowExecutionData.data.resultData.runData[pushData.nodeName] === undefined) { Vue.set(state.workflowExecutionData.data.resultData.runData, pushData.nodeName, []); } - state.stateIsDirty = true; state.workflowExecutionData.data.resultData.runData[pushData.nodeName].push(pushData.data); }, diff --git a/packages/editor-ui/src/views/NodeView.vue b/packages/editor-ui/src/views/NodeView.vue index 2cf41645eb..fff03cb087 100644 --- a/packages/editor-ui/src/views/NodeView.vue +++ b/packages/editor-ui/src/views/NodeView.vue @@ -1006,6 +1006,8 @@ export default mixins( await this.addNodes([newNodeData]); + this.$store.commit('setStateDirty', true); + // Automatically deselect all nodes and select the current one and also active // current node this.deselectAllNodes(); @@ -1500,6 +1502,8 @@ export default mixins( await this.addNodes([newNodeData]); + this.$store.commit('setStateDirty', true); + // Automatically deselect all nodes and select the current one and also active // current node this.deselectAllNodes(); @@ -1834,6 +1838,8 @@ export default mixins( // Add the nodes with the changed node names, expressions and connections await this.addNodes(Object.values(tempWorkflow.nodes), tempWorkflow.connectionsBySourceNode); + this.$store.commit('setStateDirty', true); + return { nodes: Object.values(tempWorkflow.nodes), connections: tempWorkflow.connectionsBySourceNode, diff --git a/packages/nodes-base/credentials/FacebookGraphAppApi.credentials.ts b/packages/nodes-base/credentials/FacebookGraphAppApi.credentials.ts new file mode 100644 index 0000000000..271921397a --- /dev/null +++ b/packages/nodes-base/credentials/FacebookGraphAppApi.credentials.ts @@ -0,0 +1,22 @@ +import { + ICredentialType, + NodePropertyTypes, +} from 'n8n-workflow'; + +export class FacebookGraphAppApi implements ICredentialType { + name = 'facebookGraphAppApi'; + displayName = 'Facebook Graph API (App)'; + documentationUrl = 'facebookGraphApp'; + extends = [ + 'facebookGraphApi', + ]; + properties = [ + { + displayName: 'App Secret', + name: 'appSecret', + type: 'string' as NodePropertyTypes, + default: '', + description: '(Optional) When the app secret is set the node will verify this signature to validate the integrity and origin of the payload.', + }, + ]; +} diff --git a/packages/nodes-base/credentials/ShopifyApi.credentials.ts b/packages/nodes-base/credentials/ShopifyApi.credentials.ts index e921b9fa23..7d5740a232 100644 --- a/packages/nodes-base/credentials/ShopifyApi.credentials.ts +++ b/packages/nodes-base/credentials/ShopifyApi.credentials.ts @@ -28,6 +28,7 @@ export class ShopifyApi implements ICredentialType { required: true, type: 'string' as NodePropertyTypes, default: '', + description: 'Only the subdomain without .myshopify.com', }, { displayName: 'Shared Secret', diff --git a/packages/nodes-base/credentials/StoryblokContentApi.credentials.ts b/packages/nodes-base/credentials/StoryblokContentApi.credentials.ts new file mode 100644 index 0000000000..71c32e5358 --- /dev/null +++ b/packages/nodes-base/credentials/StoryblokContentApi.credentials.ts @@ -0,0 +1,18 @@ +import { + ICredentialType, + NodePropertyTypes, +} from 'n8n-workflow'; + +export class StoryblokContentApi implements ICredentialType { + name = 'storyblokContentApi'; + displayName = 'Storyblok Content API'; + documentationUrl = 'storyblok'; + properties = [ + { + displayName: 'API Key', + name: 'apiKey', + type: 'string' as NodePropertyTypes, + default: '', + }, + ]; +} diff --git a/packages/nodes-base/credentials/StoryblokManagementApi.credentials.ts b/packages/nodes-base/credentials/StoryblokManagementApi.credentials.ts new file mode 100644 index 0000000000..de390da8e4 --- /dev/null +++ b/packages/nodes-base/credentials/StoryblokManagementApi.credentials.ts @@ -0,0 +1,18 @@ +import { + ICredentialType, + NodePropertyTypes, +} from 'n8n-workflow'; + +export class StoryblokManagementApi implements ICredentialType { + name = 'storyblokManagementApi'; + displayName = 'Storyblok Management API'; + documentationUrl = 'storyblok'; + properties = [ + { + displayName: 'Personal Access Token', + name: 'accessToken', + type: 'string' as NodePropertyTypes, + default: '', + }, + ]; +} diff --git a/packages/nodes-base/nodes/Facebook/FacebookGraphApi.node.ts b/packages/nodes-base/nodes/Facebook/FacebookGraphApi.node.ts index 42d13cc873..d8a9aa2d3c 100644 --- a/packages/nodes-base/nodes/Facebook/FacebookGraphApi.node.ts +++ b/packages/nodes-base/nodes/Facebook/FacebookGraphApi.node.ts @@ -10,7 +10,9 @@ import { INodeTypeDescription, } from 'n8n-workflow'; -import { OptionsWithUri } from 'request'; +import { + OptionsWithUri, +} from 'request'; export class FacebookGraphApi implements INodeType { description: INodeTypeDescription = { @@ -22,7 +24,7 @@ export class FacebookGraphApi implements INodeType { description: 'Interacts with Facebook using the Graph API', defaults: { name: 'Facebook Graph API', - color: '#772244', + color: '#3B5998', }, inputs: ['main'], outputs: ['main'], diff --git a/packages/nodes-base/nodes/Facebook/FacebookTrigger.node.ts b/packages/nodes-base/nodes/Facebook/FacebookTrigger.node.ts new file mode 100644 index 0000000000..278ca729bc --- /dev/null +++ b/packages/nodes-base/nodes/Facebook/FacebookTrigger.node.ts @@ -0,0 +1,251 @@ +import { + IHookFunctions, + IWebhookFunctions, +} from 'n8n-core'; + +import { + IDataObject, + INodeType, + INodeTypeDescription, + IWebhookResponseData, +} from 'n8n-workflow'; + +import * as uuid from 'uuid/v4'; + +import { + snakeCase, +} from 'change-case'; + +import { + facebookApiRequest, +} from './GenericFunctions'; + +import { + createHmac, +} from 'crypto'; + +export class FacebookTrigger implements INodeType { + description: INodeTypeDescription = { + displayName: 'Facebook Trigger', + name: 'facebookTrigger', + icon: 'file:facebook.png', + group: ['trigger'], + version: 1, + subtitle: '={{$parameter["appId"] +"/"+ $parameter["object"]}}', + description: 'Starts the workflow when a Facebook events occurs.', + defaults: { + name: 'Facebook Trigger', + color: '#3B5998', + }, + inputs: [], + outputs: ['main'], + credentials: [ + { + name: 'facebookGraphAppApi', + required: true, + }, + ], + webhooks: [ + { + name: 'setup', + httpMethod: 'GET', + responseMode: 'onReceived', + path: 'webhook', + }, + { + name: 'default', + httpMethod: 'POST', + responseMode: 'onReceived', + path: 'webhook', + }, + ], + properties: [ + { + displayName: 'Object', + name: 'object', + type: 'options', + options: [ + { + name: 'Ad Account', + value: 'adAccount', + description: 'Get updates about Ad Account', + }, + { + name: 'Application', + value: 'application', + description: 'Get updates about the app', + }, + { + name: 'Certificate Transparency', + value: 'certificateTransparency', + description: 'Get updates about Certificate Transparency', + }, + { + name: 'Group', + value: 'group', + description: 'Get updates about activity in groups and events in groups for Workplace', + }, + { + name: 'Instagram', + value: 'instagram', + description: 'Get updates about comments on your media', + }, + { + name: 'Link', + value: 'link', + description: 'Get updates about links for rich previews by an external provider', + }, + { + name: 'Page', + value: 'page', + description: 'Page updates', + }, + { + name: 'Permissions', + value: 'permissions', + description: 'Updates regarding granting or revoking permissions', + }, + { + name: 'User', + value: 'user', + description: 'User profile updates', + }, + { + name: 'Whatsapp Business Account', + value: 'whatsappBusinessAccount', + description: 'Get updates about Whatsapp business account', + }, + { + name: 'Workplace Security', + value: 'workplaceSecurity', + description: 'Get updates about Workplace Security', + }, + ], + required: true, + default: 'user', + description: 'The object to subscribe to', + }, + { + displayName: 'App ID', + name: 'appId', + type: 'string', + required: true, + default: '', + description: 'Facebook APP ID', + }, + { + displayName: 'Options', + name: 'options', + type: 'collection', + default: {}, + placeholder: 'Add option', + options: [ + { + displayName: 'Include values', + name: 'includeValues', + type: 'boolean', + default: true, + description: 'Indicates if change notifications should include the new values.', + }, + ], + }, + ], + }; + + // @ts-ignore (because of request) + webhookMethods = { + default: { + async checkExists(this: IHookFunctions): Promise<boolean> { + const webhookUrl = this.getNodeWebhookUrl('default') as string; + const object = this.getNodeParameter('object') as string; + const appId = this.getNodeParameter('appId') as string; + + const { data } = await facebookApiRequest.call(this, 'GET', `/${appId}/subscriptions`, {}); + + for (const webhook of data) { + if (webhook.target === webhookUrl && webhook.object === object && webhook.status === true) { + return true; + } + } + return false; + }, + async create(this: IHookFunctions): Promise<boolean> { + const webhookData = this.getWorkflowStaticData('node'); + const webhookUrl = this.getNodeWebhookUrl('default') as string; + const object = this.getNodeParameter('object') as string; + const appId = this.getNodeParameter('appId') as string; + const options = this.getNodeParameter('options') as IDataObject; + + const body = { + object: snakeCase(object), + callback_url: webhookUrl, + verify_token: uuid(), + } as IDataObject; + + if (options.includeValues !== undefined) { + body.include_values = options.includeValues; + } + + const responseData = await facebookApiRequest.call(this, 'POST', `/${appId}/subscriptions`, body); + + webhookData.verifyToken = body.verify_token; + + if (responseData.success !== true) { + // Facebook did not return success, so something went wrong + throw new Error('Facebook webhook creation response did not contain the expected data.'); + } + return true; + }, + async delete(this: IHookFunctions): Promise<boolean> { + const appId = this.getNodeParameter('appId') as string; + const object = this.getNodeParameter('object') as string; + + try { + await facebookApiRequest.call(this, 'DELETE', `/${appId}/subscriptions`, { object: snakeCase(object) }); + } catch (e) { + return false; + } + return true; + }, + }, + }; + + async webhook(this: IWebhookFunctions): Promise<IWebhookResponseData> { + const bodyData = this.getBodyData() as IDataObject; + const query = this.getQueryData() as IDataObject; + const res = this.getResponseObject(); + const req = this.getRequestObject(); + const headerData = this.getHeaderData() as IDataObject; + const credentials = this.getCredentials('facebookGraphAppApi') as IDataObject; + // Check if we're getting facebook's challenge request (https://developers.facebook.com/docs/graph-api/webhooks/getting-started) + if (this.getWebhookName() === 'setup') { + if (query['hub.challenge']) { + //TODO + //compare hub.verify_token with the saved token + //const webhookData = this.getWorkflowStaticData('node'); + // if (webhookData.verifyToken !== query['hub.verify_token']) { + // return {}; + // } + res.status(200).send(query['hub.challenge']).end(); + return { + noWebhookResponse: true, + }; + } + } + + // validate signature if app secret is set + if (credentials.appSecret !== '') { + //@ts-ignore + const computedSignature = createHmac('sha1', credentials.appSecret as string).update(req.rawBody).digest('hex'); + if (headerData['x-hub-signature'] !== `sha1=${computedSignature}`) { + return {}; + } + } + + return { + workflowData: [ + this.helpers.returnJsonArray(bodyData.entry as IDataObject[]), + ], + }; + } +} diff --git a/packages/nodes-base/nodes/Facebook/GenericFunctions.ts b/packages/nodes-base/nodes/Facebook/GenericFunctions.ts new file mode 100644 index 0000000000..20e16ecf49 --- /dev/null +++ b/packages/nodes-base/nodes/Facebook/GenericFunctions.ts @@ -0,0 +1,53 @@ +import { + OptionsWithUri, +} from 'request'; + +import { + IExecuteFunctions, + IExecuteSingleFunctions, + IHookFunctions, + ILoadOptionsFunctions, + IWebhookFunctions, +} from 'n8n-core'; + +import { + IDataObject, +} from 'n8n-workflow'; + +export async function facebookApiRequest(this: IHookFunctions | IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions | IWebhookFunctions, method: string, resource: string, body: any = {}, qs: IDataObject = {}, uri?: string, option: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any + + let credentials; + + if (this.getNode().name.includes('Trigger')) { + credentials = this.getCredentials('facebookGraphAppApi') as IDataObject; + } else { + credentials = this.getCredentials('facebookGraphApi') as IDataObject; + } + + qs.access_token = credentials!.accessToken; + + const options: OptionsWithUri = { + headers: { + accept: 'application/json,text/*;q=0.99', + }, + method, + qs, + body, + gzip: true, + uri: uri ||`https://graph.facebook.com/v8.0${resource}`, + json: true, + }; + + try { + return await this.helpers.request!(options); + } catch (error) { + + if (error.response.body && error.response.body.error) { + const message = error.response.body.error.message; + throw new Error( + `Facebook Trigger error response [${error.statusCode}]: ${message}`, + ); + } + throw new Error(error); + } +} diff --git a/packages/nodes-base/nodes/Facebook/facebook.png b/packages/nodes-base/nodes/Facebook/facebook.png index e0cc044609..85e6670eeb 100644 Binary files a/packages/nodes-base/nodes/Facebook/facebook.png and b/packages/nodes-base/nodes/Facebook/facebook.png differ diff --git a/packages/nodes-base/nodes/Storyblok/GenericFunctions.ts b/packages/nodes-base/nodes/Storyblok/GenericFunctions.ts new file mode 100644 index 0000000000..71ff1857b9 --- /dev/null +++ b/packages/nodes-base/nodes/Storyblok/GenericFunctions.ts @@ -0,0 +1,92 @@ +import { + OptionsWithUri, +} from 'request'; + +import { + IExecuteFunctions, + IExecuteSingleFunctions, + IHookFunctions, + ILoadOptionsFunctions, +} from 'n8n-core'; + +import { + IDataObject, +} from 'n8n-workflow'; + +export async function storyblokApiRequest(this: IHookFunctions | IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions, method: string, resource: string, body: any = {}, qs: IDataObject = {}, option: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any + const authenticationMethod = this.getNodeParameter('source', 0) as string; + + let options: OptionsWithUri = { + headers: { + 'Content-Type': 'application/json', + }, + method, + qs, + body, + uri: '', + json: true, + }; + + options = Object.assign({}, options, option); + + if (Object.keys(options.body).length === 0) { + delete options.body; + } + + if (authenticationMethod === 'contentApi') { + const credentials = this.getCredentials('storyblokContentApi') as IDataObject; + + options.uri = `https://api.storyblok.com${resource}`; + + Object.assign(options.qs, { token: credentials.apiKey }); + } else { + const credentials = this.getCredentials('storyblokManagementApi') as IDataObject; + + options.uri = `https://mapi.storyblok.com${resource}`; + + Object.assign(options.headers, { 'Authorization': credentials.accessToken }); + } + + try { + return this.helpers.request!(options); + } catch (error) { + if (error.response && error.response.body && error.response.body.message) { + // Try to return the error prettier + const errorBody = error.response.body; + throw new Error(`Storyblok error response [${error.statusCode}]: ${errorBody.message}`); + } + + // Expected error data did not get returned so throw the actual error + throw error; + } +} + +export async function storyblokApiRequestAllItems(this: IHookFunctions | ILoadOptionsFunctions | IExecuteFunctions, propertyName: string, method: string, resource: string, body: any = {}, query: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any + const returnData: IDataObject[] = []; + + let responseData; + + query.per_page = 100; + + query.page = 1; + + do { + responseData = await storyblokApiRequest.call(this, method, resource, body, query); + query.page++; + returnData.push.apply(returnData, responseData[propertyName]); + } while ( + responseData[propertyName].length !== 0 + ); + + return returnData; +} + +export function validateJSON(json: string | undefined): any { // tslint:disable-line:no-any + let result; + try { + result = JSON.parse(json!); + } catch (exception) { + result = undefined; + } + return result; +} diff --git a/packages/nodes-base/nodes/Storyblok/StoryContentDescription.ts b/packages/nodes-base/nodes/Storyblok/StoryContentDescription.ts new file mode 100644 index 0000000000..48b78e9d54 --- /dev/null +++ b/packages/nodes-base/nodes/Storyblok/StoryContentDescription.ts @@ -0,0 +1,143 @@ +import { + INodeProperties, +} from 'n8n-workflow'; + +export const storyContentOperations = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + displayOptions: { + show: { + source: [ + 'contentApi', + ], + resource: [ + 'story', + ], + }, + }, + options: [ + { + name: 'Get', + value: 'get', + description: 'Get a story', + }, + { + name: 'Get All', + value: 'getAll', + description: 'Get all stories', + }, + ], + default: 'get', + description: 'The operation to perform.', + }, +] as INodeProperties[]; + +export const storyContentFields = [ + + /* -------------------------------------------------------------------------- */ + /* story:get */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Identifier', + name: 'identifier', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'contentApi', + ], + resource: [ + 'story', + ], + operation: [ + 'get', + ], + }, + }, + description: 'The ID or slug of the story to get.', + }, + + /* -------------------------------------------------------------------------- */ + /* story:getAll */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Return All', + name: 'returnAll', + type: 'boolean', + displayOptions: { + show: { + source: [ + 'contentApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + }, + }, + default: false, + description: 'Returns a list of your user contacts.', + }, + { + displayName: 'Limit', + name: 'limit', + type: 'number', + displayOptions: { + show: { + source: [ + 'contentApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + returnAll: [ + false, + ], + }, + }, + typeOptions: { + minValue: 1, + maxValue: 100, + }, + default: 50, + description: 'How many results to return.', + }, + { + displayName: 'Filters', + name: 'filters', + type: 'collection', + placeholder: 'Add Filter', + default: {}, + displayOptions: { + show: { + source: [ + 'contentApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + }, + }, + options: [ + { + displayName: 'Starts With', + name: 'starts_with', + type: 'string', + default: '', + description: 'Filter by slug.', + }, + ], + }, +] as INodeProperties[]; diff --git a/packages/nodes-base/nodes/Storyblok/StoryManagementDescription.ts b/packages/nodes-base/nodes/Storyblok/StoryManagementDescription.ts new file mode 100644 index 0000000000..9e0acb49c3 --- /dev/null +++ b/packages/nodes-base/nodes/Storyblok/StoryManagementDescription.ts @@ -0,0 +1,647 @@ +import { + INodeProperties, +} from 'n8n-workflow'; + +export const storyManagementOperations = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + }, + }, + options: [ + // { + // name: 'Create', + // value: 'create', + // description: 'Create a story', + // }, + { + name: 'Delete', + value: 'delete', + description: 'Delete a story', + }, + { + name: 'Get', + value: 'get', + description: 'Get a story', + }, + { + name: 'Get All', + value: 'getAll', + description: 'Get all stories', + }, + { + name: 'Publish', + value: 'publish', + description: 'Publish a story', + }, + { + name: 'Unpublish', + value: 'unpublish', + description: 'Unpublish a story', + }, + ], + default: 'get', + description: 'The operation to perform.', + }, +] as INodeProperties[]; + +export const storyManagementFields = [ + + // /* -------------------------------------------------------------------------- */ + // /* story:create */ + // /* -------------------------------------------------------------------------- */ + // { + // displayName: 'Space ID', + // name: 'space', + // type: 'options', + // typeOptions: { + // loadOptionsMethod: 'getSpaces', + // }, + // default: '', + // required: true, + // displayOptions: { + // show: { + // source: [ + // 'managementApi', + // ], + // resource: [ + // 'story', + // ], + // operation: [ + // 'create', + // ], + // }, + // }, + // description: 'The name of the space.', + // }, + // { + // displayName: 'Name', + // name: 'name', + // type: 'string', + // default: '', + // required: true, + // displayOptions: { + // show: { + // source: [ + // 'managementApi', + // ], + // resource: [ + // 'story', + // ], + // operation: [ + // 'create', + // ], + // }, + // }, + // description: 'The name you give this story.', + // }, + // { + // displayName: 'Slug', + // name: 'slug', + // type: 'string', + // default: '', + // required: true, + // displayOptions: { + // show: { + // source: [ + // 'managementApi', + // ], + // resource: [ + // 'story', + // ], + // operation: [ + // 'create', + // ], + // }, + // }, + // description: 'The slug/path you give this story.', + // }, + // { + // displayName: 'JSON Parameters', + // name: 'jsonParameters', + // type: 'boolean', + // default: false, + // description: '', + // displayOptions: { + // show: { + // source: [ + // 'managementApi', + // ], + // resource: [ + // 'story', + // ], + // operation: [ + // 'create', + // ], + // }, + // }, + // }, + // { + // displayName: 'Additional Fields', + // name: 'additionalFields', + // type: 'collection', + // placeholder: 'Add Field', + // displayOptions: { + // show: { + // source: [ + // 'managementApi', + // ], + // resource: [ + // 'story', + // ], + // operation: [ + // 'create', + // ], + // }, + // }, + // default: {}, + // options: [ + // { + // displayName: 'Content', + // name: 'contentUi', + // type: 'fixedCollection', + // description: 'Add Content', + // typeOptions: { + // multipleValues: false, + // }, + // displayOptions: { + // show: { + // '/jsonParameters': [ + // false, + // ], + // }, + // }, + // placeholder: 'Add Content', + // default: '', + // options: [ + // { + // displayName: 'Content Data', + // name: 'contentValue', + // values: [ + // { + // displayName: 'Component', + // name: 'component', + // type: 'options', + // typeOptions: { + // loadOptionsMethod: 'getComponents', + // loadOptionsDependsOn: [ + // 'space', + // ], + // }, + // default: '', + // }, + // { + // displayName: 'Elements', + // name: 'elementUi', + // type: 'fixedCollection', + // description: 'Add Body', + // typeOptions: { + // multipleValues: true, + // }, + // placeholder: 'Add Element', + // default: '', + // options: [ + // { + // displayName: 'Element', + // name: 'elementValues', + // values: [ + // { + // displayName: 'Component', + // name: 'component', + // type: 'options', + // typeOptions: { + // loadOptionsMethod: 'getComponents', + // loadOptionsDependsOn: [ + // 'space', + // ], + // }, + // default: '', + // }, + // { + // displayName: 'Element Data', + // name: 'dataUi', + // type: 'fixedCollection', + // description: 'Add Data', + // typeOptions: { + // multipleValues: true, + // }, + // placeholder: 'Add Data', + // default: '', + // options: [ + // { + // displayName: 'Data', + // name: 'dataValues', + // values: [ + // { + // displayName: 'Key', + // name: 'key', + // type: 'string', + // default: '', + // }, + // { + // displayName: 'Value', + // name: 'value', + // type: 'string', + // default: '', + // }, + // ], + // }, + // ], + // }, + // ], + // }, + // ], + // }, + // ], + // }, + // ], + // }, + // { + // displayName: 'Content (JSON)', + // name: 'contentJson', + // type: 'string', + // displayOptions: { + // show: { + // '/jsonParameters': [ + // true, + // ], + // }, + // }, + // default: '', + // }, + // { + // displayName: 'Parent ID', + // name: 'parentId', + // type: 'string', + // default: '', + // description: 'Parent story/folder numeric ID.', + // }, + // { + // displayName: 'Path', + // name: 'path', + // type: 'string', + // default: '', + // description: 'Given real path, used in the preview editor.', + // }, + // { + // displayName: 'Is Startpage', + // name: 'isStartpage', + // type: 'boolean', + // default: false, + // description: 'Is startpage of current folder.', + // }, + // { + // displayName: 'First Published At', + // name: 'firstPublishedAt', + // type: 'dateTime', + // default: '', + // description: 'First publishing date.', + // }, + // ], + // }, + + /* -------------------------------------------------------------------------- */ + /* story:delete */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Space ID', + name: 'space', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getSpaces', + }, + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'delete', + ], + }, + }, + description: 'The name of the space.', + }, + { + displayName: 'Story ID', + name: 'storyId', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'delete', + ], + }, + }, + description: 'Numeric ID of the story.', + }, + + /* -------------------------------------------------------------------------- */ + /* story:get */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Space ID', + name: 'space', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getSpaces', + }, + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'get', + ], + }, + }, + description: 'The name of the space.', + }, + { + displayName: 'Story ID', + name: 'storyId', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'get', + ], + }, + }, + description: 'Numeric ID of the story.', + }, + + /* -------------------------------------------------------------------------- */ + /* story:getAll */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Space ID', + name: 'space', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getSpaces', + }, + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + }, + }, + description: 'The name of the space', + }, + { + displayName: 'Return All', + name: 'returnAll', + type: 'boolean', + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + }, + }, + default: false, + description: 'Returns a list of your user contacts.', + }, + { + displayName: 'Limit', + name: 'limit', + type: 'number', + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + returnAll: [ + false, + ], + }, + }, + typeOptions: { + minValue: 1, + maxValue: 100, + }, + default: 50, + description: 'How many results to return.', + }, + { + displayName: 'Filters', + name: 'filters', + type: 'collection', + placeholder: 'Add Filter', + default: {}, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'getAll', + ], + }, + }, + options: [ + { + displayName: 'Starts With', + name: 'starts_with', + type: 'string', + default: '', + description: 'Filter by slug.', + }, + ], + }, + + /* -------------------------------------------------------------------------- */ + /* story:publish */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Space ID', + name: 'space', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getSpaces', + }, + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'publish', + ], + }, + }, + description: 'The name of the space.', + }, + { + displayName: 'Story ID', + name: 'storyId', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'publish', + ], + }, + }, + description: 'Numeric ID of the story.', + }, + { + displayName: 'Options', + name: 'options', + type: 'collection', + placeholder: 'Add Option', + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'publish', + ], + }, + }, + default: {}, + options: [ + { + displayName: 'Release ID', + name: 'releaseId', + type: 'string', + default: '', + description: 'Numeric ID of release.', + }, + { + displayName: 'Language', + name: 'language', + type: 'string', + default: '', + description: 'Language code to publish the story individually (must be enabled in the space settings).', + }, + ], + }, + + /* -------------------------------------------------------------------------- */ + /* story:unpublish */ + /* -------------------------------------------------------------------------- */ + { + displayName: 'Space ID', + name: 'space', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getSpaces', + }, + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'unpublish', + ], + }, + }, + description: 'The name of the space.', + }, + { + displayName: 'Story ID', + name: 'storyId', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + resource: [ + 'story', + ], + operation: [ + 'unpublish', + ], + }, + }, + description: 'Numeric ID of the story.', + }, +] as INodeProperties[]; diff --git a/packages/nodes-base/nodes/Storyblok/Storyblok.node.ts b/packages/nodes-base/nodes/Storyblok/Storyblok.node.ts new file mode 100644 index 0000000000..fac9331865 --- /dev/null +++ b/packages/nodes-base/nodes/Storyblok/Storyblok.node.ts @@ -0,0 +1,334 @@ +import { + IExecuteFunctions, +} from 'n8n-core'; + +import { + IDataObject, + ILoadOptionsFunctions, + INodeExecutionData, + INodePropertyOptions, + INodeType, + INodeTypeDescription, +} from 'n8n-workflow'; + +import { + storyblokApiRequest, + storyblokApiRequestAllItems, + validateJSON, +} from './GenericFunctions'; + +import { + storyContentFields, + storyContentOperations, +} from './StoryContentDescription'; + +import { + storyManagementFields, + storyManagementOperations, +} from './StoryManagementDescription'; + +import { v4 as uuidv4 } from 'uuid'; + +export class Storyblok implements INodeType { + description: INodeTypeDescription = { + displayName: 'Storyblok', + name: 'storyblok', + icon: 'file:storyblok.svg', + group: ['output'], + version: 1, + subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}', + description: 'Consume Storyblok API', + defaults: { + name: 'Storyblok', + color: '#09b3af', + }, + inputs: ['main'], + outputs: ['main'], + credentials: [ + { + name: 'storyblokContentApi', + required: true, + displayOptions: { + show: { + source: [ + 'contentApi', + ], + }, + }, + }, + { + name: 'storyblokManagementApi', + required: true, + displayOptions: { + show: { + source: [ + 'managementApi', + ], + }, + }, + }, + ], + properties: [ + { + displayName: 'Source', + name: 'source', + type: 'options', + default: 'contentApi', + description: 'Pick where your data comes from, Content or Management API', + options: [ + { + name: 'Content API', + value: 'contentApi', + }, + { + name: 'Management API', + value: 'managementApi', + }, + ], + }, + // Resources: Content API + { + displayName: 'Resource', + name: 'resource', + type: 'options', + options: [ + { + name: 'Story', + value: 'story', + }, + ], + default: 'story', + description: 'Resource to consume.', + displayOptions: { + show: { + source: [ + 'contentApi', + ], + }, + }, + }, + // Resources: Management API + { + displayName: 'Resource', + name: 'resource', + type: 'options', + options: [ + { + name: 'Story', + value: 'story', + }, + ], + default: 'story', + description: 'Resource to consume.', + displayOptions: { + show: { + source: [ + 'managementApi', + ], + }, + }, + }, + // Content API - Story + ...storyContentOperations, + ...storyContentFields, + // Management API - Story + ...storyManagementOperations, + ...storyManagementFields, + ], + }; + + methods = { + loadOptions: { + async getSpaces( + this: ILoadOptionsFunctions, + ): Promise<INodePropertyOptions[]> { + const returnData: INodePropertyOptions[] = []; + const { spaces } = await storyblokApiRequest.call( + this, + 'GET', + '/v1/spaces', + ); + for (const space of spaces) { + returnData.push({ + name: space.name, + value: space.id, + }); + } + return returnData; + }, + async getComponents( + this: ILoadOptionsFunctions, + ): Promise<INodePropertyOptions[]> { + const returnData: INodePropertyOptions[] = []; + const space = this.getCurrentNodeParameter('space') as string; + const { components } = await storyblokApiRequest.call( + this, + 'GET', + `/v1/spaces/${space}/components`, + ); + for (const component of components) { + returnData.push({ + name: `${component.name} ${(component.is_root ? '(root)' : '')}`, + value: component.name, + }); + } + return returnData; + }, + }, + }; + + async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> { + const items = this.getInputData(); + const returnData: IDataObject[] = []; + const length = items.length as unknown as number; + const qs: IDataObject = {}; + let responseData; + const source = this.getNodeParameter('source', 0) as string; + const resource = this.getNodeParameter('resource', 0) as string; + const operation = this.getNodeParameter('operation', 0) as string; + for (let i = 0; i < length; i++) { + if (source === 'contentApi') { + if (resource === 'story') { + if (operation === 'get') { + const identifier = this.getNodeParameter('identifier', i) as string; + + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/cdn/stories/${identifier}`); + responseData = responseData.story; + } + if (operation === 'getAll') { + const filters = this.getNodeParameter('filters', i) as string; + const returnAll = this.getNodeParameter('returnAll', i) as boolean; + Object.assign(qs, filters); + + if (returnAll) { + responseData = await storyblokApiRequestAllItems.call(this, 'stories', 'GET', '/v1/cdn/stories', {}, qs); + } else { + const limit = this.getNodeParameter('limit', i) as number; + qs.per_page = limit; + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/cdn/stories`, {}, qs); + responseData = responseData.stories; + } + } + } + } + if (source === 'managementApi') { + if (resource === 'story') { + // if (operation === 'create') { + // const space = this.getNodeParameter('space', i) as string; + // const name = this.getNodeParameter('name', i) as string; + // const slug = this.getNodeParameter('slug', i) as string; + // const jsonParameters = this.getNodeParameter('jsonParameters', i) as boolean; + // const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject; + // const body: IDataObject = { + // name, + // slug, + // }; + + // if (jsonParameters) { + // if (additionalFields.contentJson) { + // const json = validateJSON(additionalFields.contentJson as string); + // body.content = json; + // } + // } else { + // if (additionalFields.contentUi) { + // const contentValue = (additionalFields.contentUi as IDataObject).contentValue as IDataObject; + // const content: { component: string, body: IDataObject[] } = { component: '', body: [] }; + // if (contentValue) { + // content.component = contentValue.component as string; + // const elementValues = (contentValue.elementUi as IDataObject).elementValues as IDataObject[]; + // for (const elementValue of elementValues) { + // const body: IDataObject = {}; + // body._uid = uuidv4(); + // body.component = elementValue.component; + // if (elementValue.dataUi) { + // const dataValues = (elementValue.dataUi as IDataObject).dataValues as IDataObject[]; + // for (const dataValue of dataValues) { + // body[dataValue.key as string] = dataValue.value; + // } + // } + // content.body.push(body); + // } + // } + // body.content = content; + // } + // } + + // if (additionalFields.parentId) { + // body.parent_id = additionalFields.parentId as string; + // } + // if (additionalFields.path) { + // body.path = additionalFields.path as string; + // } + // if (additionalFields.isStartpage) { + // body.is_startpage = additionalFields.isStartpage as string; + // } + // if (additionalFields.firstPublishedAt) { + // body.first_published_at = additionalFields.firstPublishedAt as string; + // } + + // responseData = await storyblokApiRequest.call(this, 'POST', `/v1/spaces/${space}/stories`, { story: body }); + // responseData = responseData.story; + // } + if (operation === 'delete') { + const space = this.getNodeParameter('space', i) as string; + const storyId = this.getNodeParameter('storyId', i) as string; + + responseData = await storyblokApiRequest.call(this, 'DELETE', `/v1/spaces/${space}/stories/${storyId}`); + responseData = responseData.story; + } + if (operation === 'get') { + const space = this.getNodeParameter('space', i) as string; + const storyId = this.getNodeParameter('storyId', i) as string; + + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/spaces/${space}/stories/${storyId}`); + responseData = responseData.story; + } + if (operation === 'getAll') { + const space = this.getNodeParameter('space', i) as string; + const filters = this.getNodeParameter('filters', i) as string; + const returnAll = this.getNodeParameter('returnAll', i) as boolean; + Object.assign(qs, filters); + + if (returnAll) { + responseData = await storyblokApiRequestAllItems.call(this, 'stories', 'GET', `/v1/spaces/${space}/stories`, {}, qs); + } else { + const limit = this.getNodeParameter('limit', i) as number; + qs.per_page = limit; + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/spaces/${space}/stories`, {}, qs); + responseData = responseData.stories; + } + } + if (operation === 'publish') { + const space = this.getNodeParameter('space', i) as string; + const storyId = this.getNodeParameter('storyId', i) as string; + const options = this.getNodeParameter('options', i) as IDataObject; + const query: IDataObject = {}; + // Not sure if these two options work + if (options.releaseId) { + query.release_id = options.releaseId as string; + } + if (options.language) { + query.lang = options.language as string; + } + + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/spaces/${space}/stories/${storyId}/publish`, {}, query); + responseData = responseData.story; + } + if (operation === 'unpublish') { + const space = this.getNodeParameter('space', i) as string; + const storyId = this.getNodeParameter('storyId', i) as string; + + responseData = await storyblokApiRequest.call(this, 'GET', `/v1/spaces/${space}/stories/${storyId}/unpublish`); + responseData = responseData.story; + } + } + } + if (Array.isArray(responseData)) { + returnData.push.apply(returnData, responseData as IDataObject[]); + } else { + returnData.push(responseData as IDataObject); + } + } + return [this.helpers.returnJsonArray(returnData)]; + } +} diff --git a/packages/nodes-base/nodes/Storyblok/storyblok.svg b/packages/nodes-base/nodes/Storyblok/storyblok.svg new file mode 100644 index 0000000000..b5c3a9f264 --- /dev/null +++ b/packages/nodes-base/nodes/Storyblok/storyblok.svg @@ -0,0 +1,17 @@ +<?xml version="1.0" encoding="UTF-8"?> +<svg width="87px" height="103px" viewBox="0 0 87 103" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> + <!-- Generator: Sketch 51.3 (57544) - http://www.bohemiancoding.com/sketch --> + <title>colored-standalone</title> + <desc>Created with Sketch.</desc> + <defs></defs> + <g id="colored-standalone" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"> + <g id="logo" fill="#09B3AF" fill-rule="nonzero"> + <path d="M51.8,49 L31,49 L31,59 L51.3,59 C52.5,59 53.6,58.5 54.5,57.7 C55.3,56.9 55.8,55.7 55.8,54.2 C55.8290035,52.9130708 55.4433501,51.6509323 54.7,50.6 C53.9,49.6 53,49 51.8,49 Z M52.5,36.9 C53.4,36.3 53.8,35 53.8,33.3 C53.8,31.8 53.4,30.7 52.7,30 C52,29.4 51.1,29 50.1,29 L31,29 L31,38 L49.7,38 C50.7,38 51.7,37.5 52.5,36.9 Z" id="Shape"></path> + <path d="M83,0 L4.5,0 C2,0 0,2 0,4.4 L0,83 C0,85.4 2,86.9 4.4,86.9 L16,86.9 L16,102.6 L30.4,87 L83,87 C85.4,87 86.9,85.5 86.9,83 L86.9,4.5 C86.9,2.1 85.4,0.1 82.9,0.1 L83,0 Z M69.8,63.7 C68.8,65.5 67.3,67 65.5,68.1 C63.6,69.3 61.5,70.4 59.1,70.9 C56.7,71.5 54.1,72 51.4,72 L16,72 L16,16 L56.2,16 C58.2,16 59.9,16.4 61.5,17.3 C63,18.1 64.4,19.2 65.5,20.5 C67.7403434,23.2320077 68.9444137,26.6671496 68.9,30.2 C68.9,32.8 68.2,35.3 66.9,37.7 C65.5522265,40.1140117 63.4421536,42.0130773 60.9,43.1 C64.1,44 66.6,45.6 68.5,47.9 C70.3,50.3 71.2,53.4 71.2,57.3 C71.2,59.8 70.7,61.9 69.7,63.7 L69.8,63.7 Z" id="Shape"></path> + </g> + <g id="logo" fill="#09B3AF" fill-rule="nonzero"> + <path d="M51.8,49 L31,49 L31,59 L51.3,59 C52.5,59 53.6,58.5 54.5,57.7 C55.3,56.9 55.8,55.7 55.8,54.2 C55.8290035,52.9130708 55.4433501,51.6509323 54.7,50.6 C53.9,49.6 53,49 51.8,49 Z M52.5,36.9 C53.4,36.3 53.8,35 53.8,33.3 C53.8,31.8 53.4,30.7 52.7,30 C52,29.4 51.1,29 50.1,29 L31,29 L31,38 L49.7,38 C50.7,38 51.7,37.5 52.5,36.9 Z" id="Shape"></path> + <path d="M83,0 L4.5,0 C2,0 0,2 0,4.4 L0,83 C0,85.4 2,86.9 4.4,86.9 L16,86.9 L16,102.6 L30.4,87 L83,87 C85.4,87 86.9,85.5 86.9,83 L86.9,4.5 C86.9,2.1 85.4,0.1 82.9,0.1 L83,0 Z M69.8,63.7 C68.8,65.5 67.3,67 65.5,68.1 C63.6,69.3 61.5,70.4 59.1,70.9 C56.7,71.5 54.1,72 51.4,72 L16,72 L16,16 L56.2,16 C58.2,16 59.9,16.4 61.5,17.3 C63,18.1 64.4,19.2 65.5,20.5 C67.7403434,23.2320077 68.9444137,26.6671496 68.9,30.2 C68.9,32.8 68.2,35.3 66.9,37.7 C65.5522265,40.1140117 63.4421536,42.0130773 60.9,43.1 C64.1,44 66.6,45.6 68.5,47.9 C70.3,50.3 71.2,53.4 71.2,57.3 C71.2,59.8 70.7,61.9 69.7,63.7 L69.8,63.7 Z" id="Shape"></path> + </g> + </g> +</svg> \ No newline at end of file diff --git a/packages/nodes-base/package.json b/packages/nodes-base/package.json index 1b3e482664..9e3f116fc4 100644 --- a/packages/nodes-base/package.json +++ b/packages/nodes-base/package.json @@ -66,6 +66,7 @@ "dist/credentials/EventbriteApi.credentials.js", "dist/credentials/EventbriteOAuth2Api.credentials.js", "dist/credentials/FacebookGraphApi.credentials.js", + "dist/credentials/FacebookGraphAppApi.credentials.js", "dist/credentials/FreshdeskApi.credentials.js", "dist/credentials/FileMaker.credentials.js", "dist/credentials/FlowApi.credentials.js", @@ -171,6 +172,10 @@ "dist/credentials/StravaOAuth2Api.credentials.js", "dist/credentials/StripeApi.credentials.js", "dist/credentials/Sftp.credentials.js", + "dist/credentials/Signl4Api.credentials.js", + "dist/credentials/SpotifyOAuth2Api.credentials.js", + "dist/credentials/StoryblokContentApi.credentials.js", + "dist/credentials/StoryblokManagementApi.credentials.js", "dist/credentials/SurveyMonkeyApi.credentials.js", "dist/credentials/SurveyMonkeyOAuth2Api.credentials.js", "dist/credentials/TaigaCloudApi.credentials.js", @@ -263,6 +268,7 @@ "dist/nodes/ExecuteCommand.node.js", "dist/nodes/ExecuteWorkflow.node.js", "dist/nodes/Facebook/FacebookGraphApi.node.js", + "dist/nodes/Facebook/FacebookTrigger.node.js", "dist/nodes/FileMaker/FileMaker.node.js", "dist/nodes/Ftp.node.js", "dist/nodes/Freshdesk/Freshdesk.node.js", @@ -373,6 +379,7 @@ "dist/nodes/SpreadsheetFile.node.js", "dist/nodes/SseTrigger.node.js", "dist/nodes/Start.node.js", + "dist/nodes/Storyblok/Storyblok.node.js", "dist/nodes/Strava/Strava.node.js", "dist/nodes/Strava/StravaTrigger.node.js", "dist/nodes/Stripe/StripeTrigger.node.js",