@@ -31,14 +31,13 @@ import {
31
31
32
32
// The maximum time between any two messages after which a thread is considered
33
33
// async.
34
- const MAX_HEURISTIC_SYNC_THREAD_DURATION = 12 * HOUR // 60 * MINUTE
34
+ const MAX_HEURISTIC_SYNC_THREAD_DURATION = 60 * MINUTE // 60 * MINUTE
35
35
// How frequently threads are checked for archive requirements.
36
36
const THREAD_CHECK_CADENCE = 12 * HOUR // 12 * HOUR
37
37
// Use a ThreadAutoArchiveDuration as we'll still lean on Discord to
38
38
// auto-archive after issuing the warning, so we want the value to be
39
39
// one that we can update auto-archiving to.
40
- const AUTO_ARCHIVE_WARNING_LEAD_MINUTES : ThreadAutoArchiveDuration =
41
- ThreadAutoArchiveDuration . OneDay
40
+ // const AUTO_ARCHIVE_WARNING_LEAD_MINUTES: ThreadAutoArchiveDuration = ThreadAutoArchiveDuration.OneDay
42
41
43
42
/**
44
43
* A helper to request follow-up action on a thread based on the id of the user
@@ -264,7 +263,13 @@ async function updateThreadStatusFromMessage(
264
263
return
265
264
}
266
265
267
- robot . logger . info ( "New thread being monitored" )
266
+ const { autoArchiveDuration, id : threadId } = thread
267
+
268
+ robot . logger . info (
269
+ `New thread being monitored. ID: ${ threadId } , AutoArchiveDuration: ${
270
+ autoArchiveDuration ?? "Unknown"
271
+ } `,
272
+ )
268
273
269
274
const channelMetadata = getThreadMetadata ( robot . brain , thread ) ?? {
270
275
sync : true ,
@@ -275,7 +280,7 @@ async function updateThreadStatusFromMessage(
275
280
messageTimestamp - ( thread . createdTimestamp ?? 0 ) >
276
281
MAX_HEURISTIC_SYNC_THREAD_DURATION
277
282
) {
278
- robot . logger . info ( "Marking thread" , thread . id , "as async" )
283
+ robot . logger . info ( "Marking thread" , threadId , "as async" )
279
284
channelMetadata . sync = false
280
285
updateThreadMetadata ( robot . brain , thread , channelMetadata )
281
286
}
@@ -340,15 +345,15 @@ async function checkThreadStatus(
340
345
discordClient : Client ,
341
346
) : Promise < void > {
342
347
const threadMetadataByThreadId = getAllThreadMetadata ( robot . brain )
348
+
343
349
Object . entries ( threadMetadataByThreadId )
344
350
. filter ( ( [ , metadata ] ) => metadata ?. sync === false )
345
351
. forEach ( async ( [ threadId ] ) => {
346
352
const thread = discordClient . channels . cache . get ( threadId )
347
353
348
- if ( thread === undefined || ! thread . isThread ( ) ) {
354
+ if ( ! thread ? .isThread ( ) ) {
349
355
robot . logger . error (
350
- `Error looking up thread with id ${ threadId } in the client cache; ` +
351
- "skipping archive status check." ,
356
+ `Error looking up thread with id ${ threadId } in the client cache; skipping archive status check.` ,
352
357
)
353
358
return
354
359
}
@@ -358,21 +363,77 @@ async function checkThreadStatus(
358
363
( thread . lastMessageId !== null
359
364
? await thread . messages . fetch ( thread . lastMessageId )
360
365
: undefined )
361
- const firstActiveTimestamp = thread . createdTimestamp ?? 0
362
- const lastActiveTimestamp =
363
- lastMessage ?. createdTimestamp ?? firstActiveTimestamp
364
-
365
366
// About a day before the thread auto-archives, issue a warning that it will
366
367
// be archived and ask for follow up, then set the thread to auto-archive
367
368
// after a day.
369
+ // Calculate the last activity timestamp (use thread creation as fallback)
370
+ const lastActivityTimestamp =
371
+ lastMessage ?. createdTimestamp ?? thread . createdTimestamp ?? 0
372
+
373
+ const autoArchiveDuration = thread . autoArchiveDuration as
374
+ | ThreadAutoArchiveDuration
375
+ | undefined
376
+
377
+ if ( ! autoArchiveDuration ) {
378
+ robot . logger . info (
379
+ `Thread ${ threadId } has no valid autoArchiveDuration; skipping archive check.` ,
380
+ )
381
+ return
382
+ }
383
+
384
+ // Let's be sure to calculate the exact archive time based on last activity
385
+ const autoArchiveTime =
386
+ lastActivityTimestamp + autoArchiveDuration * MINUTE
387
+
388
+ const currentTime = Date . now ( )
389
+
390
+ // We can then archive the thread if the expiry time has been reached or passed
391
+ if ( autoArchiveTime - currentTime <= 0 ) {
392
+ const warningKey = `thread-warning:${ threadId } `
393
+ const warningMessageId = robot . brain . get ( warningKey )
394
+
395
+ if ( warningMessageId ) {
396
+ try {
397
+ const warningMessage = await thread . messages . fetch ( warningMessageId )
398
+ // this will edit the original message to indicate the thread has been archived
399
+ await warningMessage . edit ( {
400
+ content :
401
+ "This thread is now archived as the auto-archive duration has been reached." ,
402
+ components : [ ] ,
403
+ } )
404
+ } catch ( error ) {
405
+ robot . logger . error (
406
+ `Failed to edit the warning message for thread ${ threadId } : ${ error } ` ,
407
+ )
408
+ }
409
+ }
410
+
411
+ await thread . setArchived ( true )
412
+ robot . logger . info (
413
+ `Archived thread ${ threadId } as the auto-archive time has been reached.` ,
414
+ )
415
+ return
416
+ }
417
+
418
+ // Then check if the thread is within the warning window
368
419
if (
369
- lastActiveTimestamp - ( firstActiveTimestamp ?? 0 ) >
370
- ( thread . autoArchiveDuration ?? 0 ) * MINUTE -
371
- AUTO_ARCHIVE_WARNING_LEAD_MINUTES * MINUTE
420
+ autoArchiveTime - currentTime <= 24 * HOUR &&
421
+ autoArchiveTime - currentTime > 0
372
422
) {
373
- await thread . send ( {
374
- content :
375
- "This thread will be auto-archived in 24 hours without further updates; what's next?" ,
423
+ const autoArchiveTimestamp = Math . floor ( autoArchiveTime / 1000 )
424
+
425
+ // + thread metadata for an existing warning message ID
426
+ const warningKey = `thread-warning:${ threadId } `
427
+ if ( robot . brain . get ( warningKey ) ) {
428
+ robot . logger . info (
429
+ `Thread ${ threadId } already has a warning message. Skipping warning.` ,
430
+ )
431
+ return
432
+ }
433
+
434
+ // Send warning message to the thread with actions
435
+ const warningMessage = await thread . send ( {
436
+ content : `This thread will be auto-archived on <t:${ autoArchiveTimestamp } :F> (<t:${ autoArchiveTimestamp } :R>) without further updates; what's next?` ,
376
437
components : [
377
438
{
378
439
type : ComponentType . ActionRow ,
@@ -389,9 +450,23 @@ async function checkThreadStatus(
389
450
] ,
390
451
} )
391
452
392
- // Set to auto-archive to the lead time so Discord handles
393
- // auto-archiving for us.
394
- await thread . setAutoArchiveDuration ( AUTO_ARCHIVE_WARNING_LEAD_MINUTES )
453
+ // Use robot brain to store the warning event data
454
+ robot . brain . set ( warningKey , warningMessage . id )
455
+ robot . logger . info (
456
+ `Sent auto-archive warning for thread ${ threadId } . Message ID: ${
457
+ warningMessage . id
458
+ } , Auto-archive time: ${ new Date ( autoArchiveTime ) . toISOString ( ) } `,
459
+ )
460
+ } else {
461
+ robot . logger . info (
462
+ `Thread ${ threadId } is not within the warning window. Current time: ${ new Date (
463
+ currentTime ,
464
+ ) . toISOString ( ) } , Auto-archive time: ${ new Date (
465
+ autoArchiveTime ,
466
+ ) . toISOString ( ) } , Time remaining: ${
467
+ ( autoArchiveTime - currentTime ) / ( 60 * 60 * 1000 )
468
+ } hours`,
469
+ )
395
470
}
396
471
// FIXME Force thread archiving once we hit the auto-archive threshold,
397
472
// FIXME as Discord no longer _actually_ auto-archives, instead
0 commit comments