Skip to content

Commit d55b232

Browse files
authored
Change thread decisions time checks (#322)
### Notes This changes the flow of the `checkThreadStatus` so that we take the existing `thread.autoArchiveDuration` and calculate the exact archival time based off the last message. That timestamp is added to the warning message in order to verify the exact archival time that discord will archive automatically. - Also prevents multiple warning messages to be sent (optional and whether we want this) - Changes archival behavior to always follow auto archive duration of thread. - Edits original message when thread is archived (timer hits zero), sets `thread.setArchived: true`
2 parents f1a38bf + 5504df8 commit d55b232

File tree

1 file changed

+96
-21
lines changed

1 file changed

+96
-21
lines changed

discord-scripts/thread-management/check-thread-archiving.ts

+96-21
Original file line numberDiff line numberDiff line change
@@ -31,14 +31,13 @@ import {
3131

3232
// The maximum time between any two messages after which a thread is considered
3333
// async.
34-
const MAX_HEURISTIC_SYNC_THREAD_DURATION = 12 * HOUR // 60 * MINUTE
34+
const MAX_HEURISTIC_SYNC_THREAD_DURATION = 60 * MINUTE // 60 * MINUTE
3535
// How frequently threads are checked for archive requirements.
3636
const THREAD_CHECK_CADENCE = 12 * HOUR // 12 * HOUR
3737
// Use a ThreadAutoArchiveDuration as we'll still lean on Discord to
3838
// auto-archive after issuing the warning, so we want the value to be
3939
// 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
4241

4342
/**
4443
* A helper to request follow-up action on a thread based on the id of the user
@@ -264,7 +263,13 @@ async function updateThreadStatusFromMessage(
264263
return
265264
}
266265

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+
)
268273

269274
const channelMetadata = getThreadMetadata(robot.brain, thread) ?? {
270275
sync: true,
@@ -275,7 +280,7 @@ async function updateThreadStatusFromMessage(
275280
messageTimestamp - (thread.createdTimestamp ?? 0) >
276281
MAX_HEURISTIC_SYNC_THREAD_DURATION
277282
) {
278-
robot.logger.info("Marking thread", thread.id, "as async")
283+
robot.logger.info("Marking thread", threadId, "as async")
279284
channelMetadata.sync = false
280285
updateThreadMetadata(robot.brain, thread, channelMetadata)
281286
}
@@ -340,15 +345,15 @@ async function checkThreadStatus(
340345
discordClient: Client,
341346
): Promise<void> {
342347
const threadMetadataByThreadId = getAllThreadMetadata(robot.brain)
348+
343349
Object.entries(threadMetadataByThreadId)
344350
.filter(([, metadata]) => metadata?.sync === false)
345351
.forEach(async ([threadId]) => {
346352
const thread = discordClient.channels.cache.get(threadId)
347353

348-
if (thread === undefined || !thread.isThread()) {
354+
if (!thread?.isThread()) {
349355
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.`,
352357
)
353358
return
354359
}
@@ -358,21 +363,77 @@ async function checkThreadStatus(
358363
(thread.lastMessageId !== null
359364
? await thread.messages.fetch(thread.lastMessageId)
360365
: undefined)
361-
const firstActiveTimestamp = thread.createdTimestamp ?? 0
362-
const lastActiveTimestamp =
363-
lastMessage?.createdTimestamp ?? firstActiveTimestamp
364-
365366
// About a day before the thread auto-archives, issue a warning that it will
366367
// be archived and ask for follow up, then set the thread to auto-archive
367368
// 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
368419
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
372422
) {
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?`,
376437
components: [
377438
{
378439
type: ComponentType.ActionRow,
@@ -389,9 +450,23 @@ async function checkThreadStatus(
389450
],
390451
})
391452

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+
)
395470
}
396471
// FIXME Force thread archiving once we hit the auto-archive threshold,
397472
// FIXME as Discord no longer _actually_ auto-archives, instead

0 commit comments

Comments
 (0)