From 68c14979d3ce73d9aea26165a349bf4884a785be Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Iv=C3=A1n=20Ovejero?= Date: Wed, 24 Mar 2021 17:59:45 +0100 Subject: [PATCH] :sparkles: Add Mattermost reactions and ephemeral message (#1552) * Replace PNG icon with SVG icon * Add reaction resource and operations * :zap: Finalize reactions and add postEphemeral * :zap: Improvements Co-authored-by: ricardo --- .../nodes/Mattermost/Mattermost.node.ts | 366 +++++++++++++++++- .../nodes/Mattermost/mattermost.png | Bin 814 -> 0 bytes .../nodes/Mattermost/mattermost.svg | 1 + 3 files changed, 366 insertions(+), 1 deletion(-) delete mode 100644 packages/nodes-base/nodes/Mattermost/mattermost.png create mode 100644 packages/nodes-base/nodes/Mattermost/mattermost.svg diff --git a/packages/nodes-base/nodes/Mattermost/Mattermost.node.ts b/packages/nodes-base/nodes/Mattermost/Mattermost.node.ts index e4618f5919..5d6476a83b 100644 --- a/packages/nodes-base/nodes/Mattermost/Mattermost.node.ts +++ b/packages/nodes-base/nodes/Mattermost/Mattermost.node.ts @@ -25,7 +25,7 @@ export class Mattermost implements INodeType { description: INodeTypeDescription = { displayName: 'Mattermost', name: 'mattermost', - icon: 'file:mattermost.png', + icon: 'file:mattermost.svg', group: ['output'], version: 1, subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}', @@ -56,6 +56,10 @@ export class Mattermost implements INodeType { name: 'Message', value: 'message', }, + { + name: 'Reaction', + value: 'reaction', + }, { name: 'User', value: 'user', @@ -138,10 +142,46 @@ export class Mattermost implements INodeType { value: 'post', description: 'Post a message into a channel', }, + { + name: 'Post Ephemeral', + value: 'postEphemeral', + description: 'Post an ephemeral message into a channel', + }, ], default: 'post', description: 'The operation to perform', }, + { + displayName: 'Operation', + name: 'operation', + type: 'options', + displayOptions: { + show: { + resource: [ + 'reaction', + ], + }, + }, + options: [ + { + name: 'Create', + value: 'create', + description: 'Add a reaction to a post.', + }, + { + name: 'Delete', + value: 'delete', + description: 'Remove a reaction from a post', + }, + { + name: 'Get All', + value: 'getAll', + description: 'Get all the reactions to one or more posts', + }, + ], + default: 'create', + description: 'The operation to perform', + }, @@ -910,6 +950,73 @@ export class Mattermost implements INodeType { }, ], }, + + // ---------------------------------- + // message:post (ephemeral) + // ---------------------------------- + { + displayName: 'User ID', + name: 'userId', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getUsers', + }, + options: [], + default: '', + required: true, + displayOptions: { + show: { + operation: [ + 'postEphemeral', + ], + resource: [ + 'message', + ], + }, + }, + description: 'ID of the user to send the ephemeral message to.', + }, + { + displayName: 'Channel ID', + name: 'channelId', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getChannels', + }, + default: '', + required: true, + displayOptions: { + show: { + operation: [ + 'postEphemeral', + ], + resource: [ + 'message', + ], + }, + }, + description: 'ID of the channel to send the ephemeral message in.', + }, + { + displayName: 'Message', + name: 'message', + type: 'string', + typeOptions: { + alwaysOpenEditWindow: true, + }, + default: '', + displayOptions: { + show: { + operation: [ + 'postEphemeral', + ], + resource: [ + 'message', + ], + }, + }, + description: 'Text to send in the ephemeral message.', + }, { displayName: 'Other Options', name: 'otherOptions', @@ -937,6 +1044,188 @@ export class Mattermost implements INodeType { }, ], }, + + // ---------------------------------- + // reaction + // ---------------------------------- + { + displayName: 'User ID', + name: 'userId', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getUsers', + }, + options: [], + default: '', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'create', + ], + }, + }, + description: 'ID of the user sending the reaction.', + }, + { + displayName: 'Post ID', + name: 'postId', + type: 'string', + default: '', + placeholder: '3moacfqxmbdw38r38fjprh6zsr', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'create', + ], + }, + }, + description: 'ID of the post to react to.
Obtainable from the post link:
https://mattermost.internal.n8n.io/[server]/pl/[postId]', + }, + { + displayName: 'Emoji Name', + name: 'emojiName', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'create', + ], + }, + }, + description: 'Emoji to use for this reaction.', + }, + { + displayName: 'User ID', + name: 'userId', + type: 'options', + typeOptions: { + loadOptionsMethod: 'getUsers', + }, + options: [], + default: '', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'delete', + ], + }, + }, + description: 'ID of the user whose reaction to delete.', + }, + { + displayName: 'Post ID', + name: 'postId', + type: 'string', + default: '', + placeholder: '3moacfqxmbdw38r38fjprh6zsr', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'delete', + ], + }, + }, + description: 'ID of the post whose reaction to delete.
Obtainable from the post link:
https://mattermost.internal.n8n.io/[server]/pl/[postId]', + }, + { + displayName: 'Emoji Name', + name: 'emojiName', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'delete', + ], + }, + }, + description: 'Name of the emoji to delete.', + }, + { + displayName: 'Post ID', + name: 'postId', + type: 'string', + default: '', + required: true, + displayOptions: { + show: { + resource: [ + 'reaction', + ], + operation: [ + 'getAll', + ], + }, + }, + description: 'One or more (comma-separated) posts to retrieve reactions from.', + }, + { + displayName: 'Return All', + name: 'returnAll', + type: 'boolean', + displayOptions: { + show: { + operation: [ + 'getAll', + ], + resource: [ + 'reaction', + ], + }, + }, + default: true, + description: 'If all results should be returned or only up to a given limit.', + }, + { + displayName: 'Limit', + name: 'limit', + type: 'number', + displayOptions: { + show: { + operation: [ + 'getAll', + ], + resource: [ + 'reaction', + ], + returnAll: [ + false, + ], + }, + }, + typeOptions: { + minValue: 1, + maxValue: 100, + }, + default: 100, + description: 'How many results to return.', + }, + // ---------------------------------- // user // ---------------------------------- @@ -1830,7 +2119,79 @@ export class Mattermost implements INodeType { // Add all the other options to the request const otherOptions = this.getNodeParameter('otherOptions', i) as IDataObject; Object.assign(body, otherOptions); + + } else if (operation === 'postEphemeral') { + + // ---------------------------------- + // message:post (ephemeral) + // ---------------------------------- + + // https://api.mattermost.com/#tag/posts/paths/~1posts~1ephemeral/post + + body = { + user_id: this.getNodeParameter('userId', i), + post: { + channel_id: this.getNodeParameter('channelId', i), + message: this.getNodeParameter('message', i), + }, + } as IDataObject; + + requestMethod = 'POST'; + endpoint = 'posts/ephemeral'; + } + + } else if (resource === 'reaction') { + + // ---------------------------------- + // reaction:create + // ---------------------------------- + + // https://api.mattermost.com/#tag/reactions/paths/~1reactions/post + + if (operation === 'create') { + + body = { + user_id: this.getNodeParameter('userId', i), + post_id: this.getNodeParameter('postId', i), + emoji_name: (this.getNodeParameter('emojiName', i) as string).replace(/:/g, ''), + create_at: Date.now(), + } as { user_id: string; post_id: string; emoji_name: string; create_at: number }; + + requestMethod = 'POST'; + endpoint = 'reactions'; + + } else if (operation === 'delete') { + + // ---------------------------------- + // reaction:delete + // ---------------------------------- + + // https://api.mattermost.com/#tag/reactions/paths/~1users~1{user_id}~1posts~1{post_id}~1reactions~1{emoji_name}/delete + + const userId = this.getNodeParameter('userId', i) as string; + const postId = this.getNodeParameter('postId', i) as string; + const emojiName = (this.getNodeParameter('emojiName', i) as string).replace(/:/g, ''); + + requestMethod = 'DELETE'; + endpoint = `users/${userId}/posts/${postId}/reactions/${emojiName}`; + + } else if (operation === 'getAll') { + + // ---------------------------------- + // reaction:getAll + // ---------------------------------- + + // https://api.mattermost.com/#tag/reactions/paths/~1posts~1ids~1reactions/post + + const postId = this.getNodeParameter('postId', i) as string; + + requestMethod = 'GET'; + endpoint = `posts/${postId}/reactions`; + + qs.limit = this.getNodeParameter('limit', 0, 0) as number; + } + } else if (resource === 'user') { if (operation === 'create') { @@ -2006,6 +2367,9 @@ export class Mattermost implements INodeType { responseData = await apiRequestAllItems.call(this, requestMethod, endpoint, body, qs); } else { responseData = await apiRequest.call(this, requestMethod, endpoint, body, qs); + if (qs.limit) { + responseData = responseData.slice(0, qs.limit); + } if (resource === 'channel' && operation === 'members') { const resolveData = this.getNodeParameter('resolveData', i) as boolean; if (resolveData) { diff --git a/packages/nodes-base/nodes/Mattermost/mattermost.png b/packages/nodes-base/nodes/Mattermost/mattermost.png deleted file mode 100644 index 9a2fe2a49c44fb12294ec752b7774481c68265e9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 814 zcmeAS@N?(olHy`uVBq!ia0vp^HXzKw3=&b&bO2H_1AIbUfwa<~1o=N|(twULDhcun zW_ZW*_4;ow*8Q(%-JW2muw!bxt*SKp!*kuGg%Lp(;%isVYjb{fbYoprMw+(JVy;s) zCm5L)t1vJy33<9WhIkx*JN0JLA_I}O!&)5*?^ZN$N4*ZR{=EPHVP3gqmT5jw$FleD zcG;(s-udWg_j1emyBekcRUMK!Z*O~rC0y#(+;3Ssc^zxwW-Yywm~B44sh~R9@yo;Q zA=b7=bGQ7k%5HP(6Sv=4Hvi69k8?b)68(%W$=_ixy*4SGi~WGZ6&1N%_vg+0Husp# z(t}HvA~>IQw5?UR$5d^;m{x3gSISlTG_Yg;lJd;gueG_&wM-*gd@iam?b+-Xw^(h<(;o~c z-3qooj18B!ex+==uSI8ls>l!F)~4?Y)7})Z39)Ywns#MN%C6wYzLxXWuhl!Jp*qzw z^{-n51`mFjq#(ZAfvb+$l1zF#= z`8oMAeqVaaA&c|IY2OR>JuI6Yd!O0P`~0`3NB7DH>(2sr8x1xuoqcpxsI0*5OE0D= z9=#&=tDtyW&)ViIKg0^AN-hq{T&(zcmSoI27UxYJp)3EYEA}~^J~46IZ0@N_4$djk z`bSxQwwBA*Roqc#v#P$cF+(kG z&B}{^&#vqI7`47?rf$@RIhl77G;SG6zPYMloxkq~|D($KEK`@&k3IHOe3YqY$h_RQ V@sJMPC2_ diff --git a/packages/nodes-base/nodes/Mattermost/mattermost.svg b/packages/nodes-base/nodes/Mattermost/mattermost.svg new file mode 100644 index 0000000000..76d5d58a41 --- /dev/null +++ b/packages/nodes-base/nodes/Mattermost/mattermost.svg @@ -0,0 +1 @@ + \ No newline at end of file