@@ -11,6 +11,10 @@ import { LinearClient } from "@linear/sdk"
11
11
12
12
const { LINEAR_API_TOKEN } = process . env
13
13
14
+ // Add channelIDs which should ignore the embed processing entirely
15
+ // #mezo-standup
16
+ const IGNORED_CHANNELS = new Set ( [ "1187783048427741296" ] )
17
+
14
18
// track processed message to avoid duplicates if original message is edited
15
19
const processedMessages = new Map <
16
20
string ,
@@ -27,20 +31,20 @@ const processedMessages = new Map<
27
31
> ( )
28
32
29
33
// 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 > > ( )
31
35
32
36
let issueTagRegex : RegExp | null = null
33
37
34
38
function initializeIssueTagRegex ( ) {
35
39
issueTagRegex =
36
- / (?< ! h t t p s : \/ \/ l i n e a r \. a p p \/ [ a - z A - Z 0 - 9 - ] + \/ i s s u e \/ ) [ A - Z ] { 3 , } - \d + \b / gi
40
+ / (?< ! h t t p s : \/ \/ l i n e a r \. a p p \/ [ a - z A - Z 0 - 9 - ] + \/ i s s u e \/ ) (?< ! [ < [ ] ) [ A - Z ] { 3 , } - \d + (? ! [ > \] ] ) \b / gi
37
41
}
38
42
39
43
const projectRegex =
40
44
/ h t t p s : \/ \/ l i n e a r \. a p p \/ ( [ a - z A - Z 0 - 9 - ] + ) \/ p r o j e c t \/ ( [ a - z A - Z 0 - 9 - ] + ) (?: # p r o j e c t U p d a t e - ( [ a - z A - Z 0 - 9 ] + ) ) ? / g
41
45
42
46
const issueUrlRegex =
43
- / h t t p s : \/ \/ l i n e a r \. a p p \/ ( [ a - z A - Z 0 - 9 - ] + ) \/ i s s u e \/ ( [ a - z A - Z 0 - 9 - ] + ) (?: .* # c o m m e n t - ( [ a - z A - Z 0 - 9 ] + ) ) ? / g
47
+ / (?< ! [ < [ ] ] ) h t t p s : \/ \/ l i n e a r \. a p p \/ ( [ a - z A - Z 0 - 9 - ] + ) \/ i s s u e \/ ( [ a - z A - Z 0 - 9 - ] + ) (?: .* # c o m m e n t - ( [ a - z A - Z 0 - 9 ] + ) ) ? (? ! [ > \] ] ) / g
44
48
45
49
function truncateToWords (
46
50
content : string | undefined ,
@@ -195,6 +199,18 @@ async function processLinearEmbeds(
195
199
logger : Log ,
196
200
linearClient : LinearClient ,
197
201
) {
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
+
198
214
if ( ! issueTagRegex ) {
199
215
logger . error ( "IssueTagRegex is not initialized." )
200
216
return
@@ -286,19 +302,31 @@ async function processLinearEmbeds(
286
302
result . embed !== null ,
287
303
)
288
304
. 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 {
290
320
channel
291
321
. send ( { embeds : [ embed ] } )
292
322
. then ( ( sentMessage ) => {
293
- sentEmbeds . set ( messageId , sentMessage )
323
+ messageEmbeds . set ( issueId , sentMessage )
294
324
} )
295
325
. catch ( ( error ) =>
296
326
logger . error (
297
327
`Failed to send embed for issue ID: ${ issueId } : ${ error } ` ,
298
328
) ,
299
329
)
300
- } else {
301
- logger . error ( `Failed to create embed for issue ID: ${ issueId } ` )
302
330
}
303
331
} )
304
332
}
@@ -337,85 +365,65 @@ export default async function linearEmbeds(
337
365
! newMessage . content ||
338
366
! (
339
367
newMessage . channel instanceof TextChannel ||
340
- newMessage . channel instanceof ThreadChannel ||
341
- newMessage . channel instanceof VoiceChannel
368
+ newMessage . channel instanceof ThreadChannel
342
369
) ||
343
370
newMessage . author ?. bot
344
371
) {
345
372
return
346
373
}
347
374
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 > ( )
356
378
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 ] ) )
371
384
}
372
385
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 )
399
395
}
400
- } else {
396
+ } )
397
+
398
+ const addedIssues = [ ...newIssues ] . filter (
399
+ ( issueId ) => ! oldIssues . has ( issueId ) && ! messageEmbeds . has ( issueId ) ,
400
+ )
401
+
402
+ if ( addedIssues . length > 0 ) {
401
403
await processLinearEmbeds (
402
404
newMessage . content ,
403
405
newMessage . id ,
404
- newMessage . channel as TextChannel | ThreadChannel ,
406
+ newMessage . channel ,
405
407
robot . logger ,
406
408
linearClient ,
407
409
)
408
410
}
409
411
} )
410
412
411
413
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
+ )
419
427
sentEmbeds . delete ( message . id )
420
428
}
421
429
} )
0 commit comments