From 619bd4ea32a5f4669dfd96daf86a0bb7ff2116ca Mon Sep 17 00:00:00 2001 From: Raymond Feng Date: Sat, 18 Feb 2023 07:48:04 -0800 Subject: [PATCH] feat: improve the hello-action UI Signed-off-by: Raymond Feng --- .../hello-action-ecdsa.acceptance.ts | 34 ++- .../hello-action-ed25519.acceptance.ts | 36 ++- src/actions/hello-action.controller.ts | 256 ++++++++++++------ 3 files changed, 238 insertions(+), 88 deletions(-) diff --git a/src/__tests__/acceptance/hello-action-ecdsa.acceptance.ts b/src/__tests__/acceptance/hello-action-ecdsa.acceptance.ts index 33e53ac..89fb03b 100644 --- a/src/__tests__/acceptance/hello-action-ecdsa.acceptance.ts +++ b/src/__tests__/acceptance/hello-action-ecdsa.acceptance.ts @@ -85,13 +85,45 @@ describe('HelloAction - ecdsa', () => { description: "Name of person we're greeting", type: 3, required: true, + autocomplete: true, }, ], }, ]); expect(result.response).to.eql({ type: 4, - data: {content: 'Hello, John!', flags: 64}, + data: { + content: 'Hello, John!', + embeds: [ + { + title: 'Hello Action', + color: 16106056, + author: { + name: 'Collab.Land', + url: 'https://collab.land', + icon_url: + 'https://cdn.discordapp.com/app-icons/715138531994894397/8a814f663844a69d22344dc8f4983de6.png', + }, + description: + 'This is demo Collab.Land action that adds `/hello-action` command to your Discord server. Please click the `Count down` button below to proceed.', + url: 'https://github.com/abridged/collabland-hello-action/', + }, + ], + components: [ + { + type: 1, + components: [ + { + type: 2, + label: 'Count down', + style: 1, + custom_id: 'hello-action:count-button', + }, + ], + }, + ], + flags: 64, + }, }); }); }); diff --git a/src/__tests__/acceptance/hello-action-ed25519.acceptance.ts b/src/__tests__/acceptance/hello-action-ed25519.acceptance.ts index 0f798bb..bfbf4b9 100644 --- a/src/__tests__/acceptance/hello-action-ed25519.acceptance.ts +++ b/src/__tests__/acceptance/hello-action-ed25519.acceptance.ts @@ -21,7 +21,7 @@ describe('HelloAction - ed25519', () => { await app.stop(); }); - it('invokes action with ecdsa signature', async () => { + it('invokes action with ed25519 signature', async () => { const result = await client( app.restServer.url + '/hello-action', 'ed25519:' + signingKey, @@ -42,13 +42,45 @@ describe('HelloAction - ed25519', () => { description: "Name of person we're greeting", type: 3, required: true, + autocomplete: true, }, ], }, ]); expect(result.response).to.eql({ type: 4, - data: {content: 'Hello, John!', flags: 64}, + data: { + content: 'Hello, John!', + embeds: [ + { + title: 'Hello Action', + color: 16106056, + author: { + name: 'Collab.Land', + url: 'https://collab.land', + icon_url: + 'https://cdn.discordapp.com/app-icons/715138531994894397/8a814f663844a69d22344dc8f4983de6.png', + }, + description: + 'This is demo Collab.Land action that adds `/hello-action` command to your Discord server. Please click the `Count down` button below to proceed.', + url: 'https://github.com/abridged/collabland-hello-action/', + }, + ], + components: [ + { + type: 1, + components: [ + { + type: 2, + label: 'Count down', + style: 1, + custom_id: 'hello-action:count-button', + }, + ], + }, + ], + flags: 64, + }, }); }); }); diff --git a/src/actions/hello-action.controller.ts b/src/actions/hello-action.controller.ts index 9f338df..3897786 100644 --- a/src/actions/hello-action.controller.ts +++ b/src/actions/hello-action.controller.ts @@ -3,9 +3,8 @@ // This file is licensed under the MIT License. // License text available at https://opensource.org/licenses/MIT -import {sleep, stringify} from '@collabland/common'; +import {debugFactory, sleep} from '@collabland/common'; import { - APIApplicationCommandInteraction, APIChatInputApplicationCommandInteraction, APIInteractionResponse, ApplicationCommandOptionType, @@ -19,13 +18,26 @@ import { InteractionType, MessageFlags, RESTPatchAPIWebhookWithTokenMessageJSONBody, - RESTPostAPIWebhookWithTokenJSONBody, buildSimpleResponse, getCommandOptionValue, } from '@collabland/discord'; import {MiniAppManifest} from '@collabland/models'; import {BindingScope, injectable} from '@loopback/core'; import {api} from '@loopback/rest'; +import { + APIApplicationCommandAutocompleteInteraction, + APIApplicationCommandAutocompleteResponse, + APIInteraction, + APIMessageComponentInteraction, + ActionRowBuilder, + ButtonBuilder, + ButtonStyle, + EmbedBuilder, + InteractionResponseType, + MessageActionRowComponentBuilder, +} from 'discord.js'; + +const debug = debugFactory('collabland:hello-action'); /** * HelloActionController is a LoopBack REST API controller that exposes endpoints @@ -35,9 +47,10 @@ import {api} from '@loopback/rest'; scope: BindingScope.SINGLETON, }) @api({basePath: '/hello-action'}) // Set the base path to `/hello-action` -export class HelloActionController extends BaseDiscordActionController { +export class HelloActionController extends BaseDiscordActionController { /** - * Expose metadata for the action + * Expose metadata for the action. The return value is used by Collab.Land `/test-flight` command + * or marketplace to list this action as a miniapp. * @returns */ async getMetadata(): Promise { @@ -70,94 +83,161 @@ export class HelloActionController extends BaseDiscordActionController, - ): Promise { - let response: APIInteractionResponse | undefined = undefined; - let message: string = 'Hello'; - if (interaction.data.type === ApplicationCommandType.ChatInput) { - // Handle `/hello-action` - /** - * Get the value of `your-name` argument for `/hello-action` - */ - const yourName = getCommandOptionValue( - interaction as APIChatInputApplicationCommandInteraction, - 'your-name', - ); - message = `Hello, ${yourName ?? interaction.user?.username ?? 'World'}!`; - /** - * Build a simple Discord message private to the user - */ - response = buildSimpleResponse(message, true); - } else if (interaction.data.type === ApplicationCommandType.Message) { - // Handle `Verify` message command - const discordMsg = - interaction.data.resolved.messages[interaction.data.target_id]; - const content = stringify({ - hello: discordMsg, - }); - message = `Hello, ${ - discordMsg.id ?? interaction.user?.username ?? 'World' - }!`; - response = buildSimpleResponse(content, true); - } else if (interaction.data.type === ApplicationCommandType.User) { - // Handle `Verify` user command - const discordUser = - interaction.data.resolved.users[interaction.data.target_id]; - const content = stringify({ - hello: discordUser, - }); - message = `Hello, ${ - discordUser.username ?? interaction.user?.username ?? 'World' - }!`; - response = buildSimpleResponse(content, true); + protected async handleApplicationCommand( + request: DiscordActionRequest, + ): Promise { + switch (request.data.name) { + case 'hello-action': { + /** + * Get the value of `your-name` argument for `/hello-action` + */ + const yourName = getCommandOptionValue(request, 'your-name'); + const message = `Hello, ${ + yourName ?? request.user?.username ?? 'World' + }!`; + + const appId = request.application_id; + const response: APIInteractionResponse = { + type: InteractionResponseType.ChannelMessageWithSource, + data: { + content: message, + embeds: [ + new EmbedBuilder() + .setTitle('Hello Action') + .setColor('#f5c248') + .setAuthor({ + name: 'Collab.Land', + url: 'https://collab.land', + iconURL: `https://cdn.discordapp.com/app-icons/${appId}/8a814f663844a69d22344dc8f4983de6.png`, + }) + .setDescription( + 'This is demo Collab.Land action that adds `/hello-action` ' + + 'command to your Discord server. Please click the `Count down` button below to proceed.', + ) + .setURL('https://github.com/abridged/collabland-hello-action/') + .toJSON(), + ], + components: [ + new ActionRowBuilder() + .addComponents( + new ButtonBuilder() + .setLabel(`Count down`) + .setStyle(ButtonStyle.Primary) + // Set the custom id to start with `hello-action:` + .setCustomId('hello-action:count-button'), + ) + .toJSON(), + ], + flags: MessageFlags.Ephemeral, + }, + }; + + // Return the 1st response to Discord + return response; + } + default: { + return buildSimpleResponse( + `Slash command ${request.data.name} is not implemented.`, + ); + } } - /** - * Allow advanced followup messages - */ - this.followup(interaction, message).catch(err => { - console.error( - 'Fail to send followup message to interaction %s: %O', - interaction.id, - err, - ); - }); - // Return the 1st response to Discord - return response; } - private async followup( - request: DiscordActionRequest, - message: string, + /** + * Handle the Discord message components including buttons + * @param interaction - Discord interaction with Collab.Land action context + * @returns - Discord interaction response + */ + protected async handleMessageComponent( + request: DiscordActionRequest, + ): Promise { + switch (request.data.custom_id) { + case 'hello-action:count-button': { + // Run count down in the background after 1 second + this.countDown(request).catch(err => { + console.error( + 'Fail to send followup message to interaction %s: %O', + request.id, + err, + ); + }); + } + } + // Instruct Discord that we'll edit the original message later on + return { + type: InteractionResponseType.DeferredMessageUpdate, + }; + } + + /** + * Run a countdown by updating the original message content + * @param request + */ + private async countDown( + request: DiscordActionRequest, ) { - const callback = request.actionContext?.callbackUrl; - if (callback != null) { - const followupMsg: RESTPostAPIWebhookWithTokenJSONBody = { - content: `Follow-up: **${message}**`, - flags: MessageFlags.Ephemeral, + await sleep(1000); + const message = request.message.content; + // 5 seconds count down + for (let i = 5; i > 0; i--) { + const updated: RESTPatchAPIWebhookWithTokenMessageJSONBody = { + content: `[${i}s]: **${message}**`, + components: [], // Remove the `Count down` button }; + await this.editMessage(request, updated, request.message.id); await sleep(1000); - let msg = await this.followupMessage(request, followupMsg); - await sleep(1000); - // 5 seconds count down - for (let i = 5; i > 0; i--) { - const updated: RESTPatchAPIWebhookWithTokenMessageJSONBody = { - content: `[${i}s]: **${message}**`, - }; - msg = await this.editMessage(request, updated, msg?.id); - await sleep(1000); - } - // Delete the follow-up message - await this.deleteMessage(request, msg?.id); + } + // Delete the follow-up message + await this.deleteMessage(request, request.message.id); + } + + protected async handleApplicationCommandAutoComplete( + interaction: DiscordActionRequest, + ): Promise { + debug('Autocomplete request: %O', interaction); + const option = interaction.data.options.find(o => { + return ( + o.name === 'your-name' && + o.type === ApplicationCommandOptionType.String && + o.focused + ); + }); + if (option?.type === ApplicationCommandOptionType.String) { + const candidates = [ + 'Ethereum', + 'Polygon', + 'Optimism', + 'Arbitrum', + 'Flow', + 'Solana', + 'Near', + 'Tezos', + 'Ronin', + 'Xrpl', + ]; + const prefix = option.value; + const choices = candidates + .filter(c => c.toLowerCase().startsWith(prefix.toLowerCase())) + .map(c => ({name: c, value: c})); + + const res: APIApplicationCommandAutocompleteResponse = { + type: InteractionResponseType.ApplicationCommandAutocompleteResult, + data: { + choices, + }, + }; + debug('Autocomplete response: %O', res); + return res; } } /** - * Build a list of supported Discord interactions + * Build a list of supported Discord interactions. The return value is used as filter so that + * Collab.Land can route the corresponding interactions to this action. * @returns */ private getSupportedInteractions(): DiscordInteractionPattern[] { @@ -168,10 +248,15 @@ export class HelloActionController extends BaseDiscordActionController