Skip to content

Commit 98b60b9

Browse files
ayoubqrtpotb
authored andcommitted
feat: allow deletion of fxtwitter bot link (#86)
1 parent 746e15f commit 98b60b9

File tree

4 files changed

+178
-48
lines changed

4 files changed

+178
-48
lines changed

.env.example

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,4 +8,4 @@ REDIS_URL=
88
COOL_LINKS_MANAGEMENT_CHANNEL_ID=
99
COOL_LINKS_MANAGEMENT_PAGE_SUMMARIZER_BASE_URL=
1010

11-
PATTERN_REPLACE_EXCLUDED_CHANNEL_ID=
11+
FIX_EMBED_TWITTER_VIDEO_EXCLUDED_CHANNEL_ID=
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,175 @@
1+
import { ActionRowBuilder, ButtonBuilder, ButtonStyle, MessageType } from 'discord.js';
2+
import { z } from 'zod';
3+
4+
import { createModule } from '../../core/createModule';
5+
import { resolveCatch } from '../../helpers/resolveCatch.helper';
6+
7+
const FXTwitterResponseSchema = z.object({
8+
code: z.number(),
9+
message: z.string(),
10+
tweet: z.object({
11+
media: z
12+
.object({
13+
videos: z
14+
.array(
15+
z
16+
.object({
17+
type: z.enum(['gif', 'video']),
18+
url: z.string(),
19+
})
20+
.optional(),
21+
)
22+
.optional(),
23+
})
24+
.optional(),
25+
}),
26+
});
27+
28+
type URLMapping = {
29+
pattern: RegExp;
30+
replacement: string;
31+
};
32+
33+
const modulePrefixButtonId = 'fixEmbedTwitterVideo-';
34+
const deleteBotAnswerPrefixButtonId = modulePrefixButtonId + 'deleteBotAnswer-';
35+
const deleteBotAnswerButtonId = deleteBotAnswerPrefixButtonId + '$authorIdMessage';
36+
const ignoreConfirmationButtonId = modulePrefixButtonId + 'ignoreBotButtons';
37+
38+
const twitterUrlMappings: URLMapping[] = [
39+
{
40+
pattern: /https?:\/\/(mobile\.)?(twitter|x)\.com\/(\S+)\/status\/(\d+)/g,
41+
replacement: 'https://fxtwitter.com/$3/status/$4',
42+
},
43+
];
44+
45+
const FXTwitterUrlMappings: URLMapping[] = [
46+
{
47+
pattern: /https?:\/\/fxtwitter\.com\/(\S+)\/status\/(\d+)/g,
48+
replacement: 'https://api.fxtwitter.com/$1/status/$2',
49+
},
50+
];
51+
52+
const matchAndReplaceTweetLink = (message: string, urlMappings: URLMapping[]) => {
53+
let tweetLink = '';
54+
55+
for (const urlMapping of urlMappings) {
56+
const twitterLinks = message.match(urlMapping.pattern);
57+
58+
if (twitterLinks && twitterLinks.length > 0) {
59+
tweetLink = twitterLinks[0].replace(urlMapping.pattern, urlMapping.replacement);
60+
61+
break;
62+
}
63+
}
64+
65+
return tweetLink;
66+
};
67+
68+
const isTwitterVideo = async (tweetURL: string): Promise<boolean> => {
69+
const apiFxTweetURL = matchAndReplaceTweetLink(tweetURL, FXTwitterUrlMappings);
70+
71+
const [tweetInfoResponseError, tweetInfoResponse] = await resolveCatch(
72+
fetch(apiFxTweetURL, { method: 'GET' }),
73+
);
74+
if (tweetInfoResponseError) return false;
75+
76+
const [tweetInfoJsonError, tweetInfoJson] = await resolveCatch(tweetInfoResponse.json());
77+
if (tweetInfoJsonError) return false;
78+
79+
const tweetInfo = FXTwitterResponseSchema.safeParse(tweetInfoJson);
80+
if (!tweetInfo.success) return false;
81+
82+
if (tweetInfo.data.code !== 200) return false;
83+
84+
const video = tweetInfo.data.tweet.media?.videos?.at(0);
85+
86+
return video?.type === 'video';
87+
};
88+
89+
export const fixEmbedTwitterVideo = createModule({
90+
env: {
91+
EXCLUDED_CHANNEL_ID: z.string().nonempty(),
92+
},
93+
eventHandlers: ({ env }) => ({
94+
messageCreate: async (message) => {
95+
if (
96+
message.author.bot ||
97+
message.type !== MessageType.Default ||
98+
message.channelId === env.EXCLUDED_CHANNEL_ID
99+
) {
100+
return;
101+
}
102+
103+
const tweetLink = matchAndReplaceTweetLink(message.content, twitterUrlMappings);
104+
if (tweetLink === '') return;
105+
106+
const isTwitterVideoLink = await isTwitterVideo(tweetLink);
107+
if (!isTwitterVideoLink) return;
108+
109+
const cancel = new ButtonBuilder()
110+
.setCustomId(deleteBotAnswerButtonId.replace('$authorIdMessage', message.id))
111+
.setLabel('Remove bot answer')
112+
.setEmoji('🚮')
113+
.setStyle(ButtonStyle.Primary);
114+
115+
const ignore = new ButtonBuilder()
116+
.setCustomId(ignoreConfirmationButtonId)
117+
.setLabel('Ignore bot buttons')
118+
.setEmoji('💨')
119+
.setStyle(ButtonStyle.Primary);
120+
121+
const row = new ActionRowBuilder<ButtonBuilder>().addComponents(cancel, ignore);
122+
123+
await message.suppressEmbeds(true);
124+
await message.reply({
125+
content: tweetLink,
126+
components: [row],
127+
});
128+
},
129+
interactionCreate: async (interaction) => {
130+
if (!interaction.isButton()) return;
131+
if (!interaction.customId.startsWith(modulePrefixButtonId)) return;
132+
if (!interaction.message.author?.bot) return;
133+
134+
if (interaction.customId === ignoreConfirmationButtonId) {
135+
await interaction.update({ components: [] });
136+
137+
return;
138+
}
139+
140+
if (interaction.customId.startsWith(deleteBotAnswerPrefixButtonId))
141+
await interaction.message.delete();
142+
143+
const authorMessageId = interaction.customId.split('-')[2];
144+
if (!authorMessageId) return;
145+
146+
const authorMessage = await interaction.channel?.messages.fetch(authorMessageId);
147+
if (!authorMessage) return;
148+
149+
await authorMessage.suppressEmbeds(false);
150+
},
151+
// Added this handler in case if the user has ignored the bot buttons and still wants to delete the bot answer
152+
messageReactionAdd: async (reaction, user) => {
153+
if (user.bot) return;
154+
155+
if (
156+
reaction.message.author?.bot &&
157+
reaction.message.content?.includes('fxtwitter.com') &&
158+
reaction.emoji.name === '🚮' &&
159+
reaction.message.type === MessageType.Reply
160+
) {
161+
const referenceMessageId = reaction.message.reference?.messageId;
162+
if (!referenceMessageId) return;
163+
164+
const reference = await reaction.message.channel.messages.fetch(referenceMessageId);
165+
if (!reference) return;
166+
167+
if (reference.author.id !== user.id) return;
168+
169+
await reaction.message.delete();
170+
await reference.suppressEmbeds(false);
171+
}
172+
},
173+
}),
174+
intents: ['GuildMessages', 'MessageContent', 'GuildMessageReactions'],
175+
});

src/modules/modules.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { coolLinksManagement } from './coolLinksManagement/coolLinksManagement.module';
22
import { fart } from './fart/fart.module';
3-
import { patternReplace } from './patternReplace/patternReplace.module';
3+
import { fixEmbedTwitterVideo } from './fixEmbedTwitterVideo/fixEmbedTwitterVideo.module';
44
import { quoiFeur } from './quoiFeur/quoiFeur.module';
55
import { recurringMessage } from './recurringMessage/recurringMessage.module';
66
import { voiceOnDemand } from './voiceOnDemand/voiceOnDemand.module';
@@ -9,7 +9,7 @@ export const modules = {
99
fart,
1010
voiceOnDemand,
1111
coolLinksManagement,
12-
patternReplace,
1312
quoiFeur,
1413
recurringMessage,
14+
fixEmbedTwitterVideo,
1515
};

src/modules/patternReplace/patternReplace.module.ts

-45
This file was deleted.

0 commit comments

Comments
 (0)