Skip to content

Commit c4285f0

Browse files
authored
Fixes to linear embeds (#341)
### Notes This PR resolves a number of issues that were happening with Linear embeds into Discord such as: - Fix duplicate issues, we now properly delete embeds if the message is updated with `messageUpdate` - Add a `IGNORED_CHANNEL` const so that specific channelIDs can have embeds disabled. - Add a `<no_embeds>` text event so that if it's present in the message, ignore embeds - Refactored the regex so that if a ISE-123 is wrapped in `[*]` or `<*>`, we ignore the embeds. In some cases this was causing duplicate issues to be embedded as issue tags were being linked. - Refactored to get rid of un-necessary duplicate calls
2 parents ecd9a6b + 4aad984 commit c4285f0

File tree

1 file changed

+74
-66
lines changed

1 file changed

+74
-66
lines changed

discord-scripts/fix-linear-embed.ts

+74-66
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,10 @@ import { LinearClient } from "@linear/sdk"
1111

1212
const { LINEAR_API_TOKEN } = process.env
1313

14+
// Add channelIDs which should ignore the embed processing entirely
15+
// #mezo-standup
16+
const IGNORED_CHANNELS = new Set(["1187783048427741296"])
17+
1418
// track processed message to avoid duplicates if original message is edited
1519
const processedMessages = new Map<
1620
string,
@@ -27,20 +31,20 @@ const processedMessages = new Map<
2731
>()
2832

2933
// let us also track sent embeds to delete them if the original message is deleted or edited WIP
30-
const sentEmbeds = new Map<string, Message>()
34+
const sentEmbeds = new Map<string, Map<string, Message>>()
3135

3236
let issueTagRegex: RegExp | null = null
3337

3438
function initializeIssueTagRegex() {
3539
issueTagRegex =
36-
/(?<!https:\/\/linear\.app\/[a-zA-Z0-9-]+\/issue\/)[A-Z]{3,}-\d+\b/gi
40+
/(?<!https:\/\/linear\.app\/[a-zA-Z0-9-]+\/issue\/)(?<![<[])[A-Z]{3,}-\d+(?![>\]])\b/gi
3741
}
3842

3943
const projectRegex =
4044
/https:\/\/linear\.app\/([a-zA-Z0-9-]+)\/project\/([a-zA-Z0-9-]+)(?:#projectUpdate-([a-zA-Z0-9]+))?/g
4145

4246
const issueUrlRegex =
43-
/https:\/\/linear\.app\/([a-zA-Z0-9-]+)\/issue\/([a-zA-Z0-9-]+)(?:.*#comment-([a-zA-Z0-9]+))?/g
47+
/(?<![<[]])https:\/\/linear\.app\/([a-zA-Z0-9-]+)\/issue\/([a-zA-Z0-9-]+)(?:.*#comment-([a-zA-Z0-9]+))?(?![>\]])/g
4448

4549
function truncateToWords(
4650
content: string | undefined,
@@ -195,6 +199,18 @@ async function processLinearEmbeds(
195199
logger: Log,
196200
linearClient: LinearClient,
197201
) {
202+
if (IGNORED_CHANNELS.has(channel.id)) {
203+
logger.debug(`Ignoring embeds in channel: ${channel.id}`)
204+
return
205+
}
206+
// Let's include a text call if no embeds are to be used in a message
207+
if (message.includes("<no_embeds>")) {
208+
logger.debug(
209+
`Skipping embeds for message: ${messageId} (contains <no_embeds>)`,
210+
)
211+
return
212+
}
213+
198214
if (!issueTagRegex) {
199215
logger.error("IssueTagRegex is not initialized.")
200216
return
@@ -286,19 +302,31 @@ async function processLinearEmbeds(
286302
result.embed !== null,
287303
)
288304
.forEach(({ embed, issueId }) => {
289-
if (embed) {
305+
if (!sentEmbeds.has(messageId)) {
306+
sentEmbeds.set(messageId, new Map())
307+
}
308+
const messageEmbeds = sentEmbeds.get(messageId)!
309+
310+
if (messageEmbeds.has(issueId)) {
311+
messageEmbeds
312+
.get(issueId)!
313+
.edit({ embeds: [embed] })
314+
.catch((error) =>
315+
logger.error(
316+
`Failed to edit embed for issue ID: ${issueId}: ${error}`,
317+
),
318+
)
319+
} else {
290320
channel
291321
.send({ embeds: [embed] })
292322
.then((sentMessage) => {
293-
sentEmbeds.set(messageId, sentMessage)
323+
messageEmbeds.set(issueId, sentMessage)
294324
})
295325
.catch((error) =>
296326
logger.error(
297327
`Failed to send embed for issue ID: ${issueId}: ${error}`,
298328
),
299329
)
300-
} else {
301-
logger.error(`Failed to create embed for issue ID: ${issueId}`)
302330
}
303331
})
304332
}
@@ -337,85 +365,65 @@ export default async function linearEmbeds(
337365
!newMessage.content ||
338366
!(
339367
newMessage.channel instanceof TextChannel ||
340-
newMessage.channel instanceof ThreadChannel ||
341-
newMessage.channel instanceof VoiceChannel
368+
newMessage.channel instanceof ThreadChannel
342369
) ||
343370
newMessage.author?.bot
344371
) {
345372
return
346373
}
347374

348-
const embedMessage = sentEmbeds.get(newMessage.id)
349-
const urlMatches = Array.from(newMessage.content.matchAll(issueUrlRegex))
350-
const issueMatches = issueTagRegex
351-
? Array.from(newMessage.content.matchAll(issueTagRegex))
352-
: []
353-
const projectMatches = projectRegex
354-
? Array.from(newMessage.content.matchAll(projectRegex))
355-
: []
375+
const messageEmbeds = sentEmbeds.get(newMessage.id) ?? new Map()
376+
const oldIssues = new Set<string>()
377+
const newIssues = new Set<string>()
356378

357-
if (
358-
urlMatches.length === 0 &&
359-
issueMatches.length === 0 &&
360-
projectMatches.length === 0
361-
) {
362-
if (embedMessage) {
363-
await embedMessage.delete().catch((error) => {
364-
robot.logger.error(
365-
`Failed to delete embed for message ID: ${newMessage.id}: ${error}`,
366-
)
367-
})
368-
sentEmbeds.delete(newMessage.id)
369-
}
370-
return
379+
if (oldMessage.content) {
380+
;[
381+
...oldMessage.content.matchAll(issueUrlRegex),
382+
...oldMessage.content.matchAll(issueTagRegex ?? /$^/),
383+
].forEach((match) => oldIssues.add(match[2] ?? match[0]))
371384
}
372385

373-
const match = urlMatches[0] || issueMatches[0] || projectMatches[0]
374-
const teamName = match[1] || undefined
375-
const issueId = match[2] || match[0]
376-
const commentId = urlMatches.length > 0 ? match[3] || undefined : undefined
377-
378-
if (embedMessage) {
379-
// we will then update the existing embed
380-
try {
381-
const embed = await createLinearEmbed(
382-
linearClient,
383-
issueId,
384-
commentId,
385-
teamName,
386-
)
387-
if (embed) {
388-
await embedMessage.edit({ embeds: [embed] })
389-
robot.logger.debug(`Updated embed for message ID: ${newMessage.id}`)
390-
} else {
391-
robot.logger.error(
392-
`Failed to create embed for updated message ID: ${newMessage.id}`,
393-
)
394-
}
395-
} catch (error) {
396-
robot.logger.error(
397-
`Failed to edit embed for message ID: ${newMessage.id}: ${error}`,
398-
)
386+
;[
387+
...newMessage.content.matchAll(issueUrlRegex),
388+
...newMessage.content.matchAll(issueTagRegex ?? /$^/),
389+
].forEach((match) => newIssues.add(match[2] ?? match[0]))
390+
391+
oldIssues.forEach((issueId) => {
392+
if (!newIssues.has(issueId) && messageEmbeds.has(issueId)) {
393+
messageEmbeds.get(issueId)?.delete().catch(console.error)
394+
messageEmbeds.delete(issueId)
399395
}
400-
} else {
396+
})
397+
398+
const addedIssues = [...newIssues].filter(
399+
(issueId) => !oldIssues.has(issueId) && !messageEmbeds.has(issueId),
400+
)
401+
402+
if (addedIssues.length > 0) {
401403
await processLinearEmbeds(
402404
newMessage.content,
403405
newMessage.id,
404-
newMessage.channel as TextChannel | ThreadChannel,
406+
newMessage.channel,
405407
robot.logger,
406408
linearClient,
407409
)
408410
}
409411
})
410412

411413
discordClient.on("messageDelete", async (message) => {
412-
const embedMessage = sentEmbeds.get(message.id)
413-
if (embedMessage) {
414-
await embedMessage.delete().catch((error) => {
415-
robot.logger.error(
416-
`Failed to delete embed for message ID: ${message.id}: ${error}`,
417-
)
418-
})
414+
const embedMessages = sentEmbeds.get(message.id)
415+
if (embedMessages) {
416+
await Promise.all(
417+
Array.from(embedMessages.values()).map((embedMessage) =>
418+
embedMessage.delete().catch((error: unknown) => {
419+
if (error instanceof Error) {
420+
robot.logger.error(`Failed to delete embed: ${error.message}`)
421+
} else {
422+
robot.logger.error(`Unknown error: ${error}`)
423+
}
424+
}),
425+
),
426+
)
419427
sentEmbeds.delete(message.id)
420428
}
421429
})

0 commit comments

Comments
 (0)