2026-01-23 23:04:09 +00:00
|
|
|
import { resolveHumanDelayConfig } from "../../../agents/identity.js";
|
2026-01-23 22:51:37 +00:00
|
|
|
import { dispatchInboundMessage } from "../../../auto-reply/dispatch.js";
|
2026-01-23 22:36:43 +00:00
|
|
|
import { clearHistoryEntriesIfEnabled } from "../../../auto-reply/reply/history.js";
|
2026-02-01 10:03:47 +09:00
|
|
|
import { createReplyDispatcherWithTyping } from "../../../auto-reply/reply/reply-dispatcher.js";
|
2026-01-23 22:29:47 +00:00
|
|
|
import { removeAckReactionAfterReply } from "../../../channels/ack-reactions.js";
|
2026-01-23 23:20:07 +00:00
|
|
|
import { logAckFailure, logTypingFailure } from "../../../channels/logging.js";
|
feat: per-channel responsePrefix override (#9001)
* feat: per-channel responsePrefix override
Add responsePrefix field to all channel config types and Zod schemas,
enabling per-channel and per-account outbound response prefix overrides.
Resolution cascade (most specific wins):
L1: channels.<ch>.accounts.<id>.responsePrefix
L2: channels.<ch>.responsePrefix
L3: (reserved for channels.defaults)
L4: messages.responsePrefix (existing global)
Semantics:
- undefined -> inherit from parent level
- empty string -> explicitly no prefix (stops cascade)
- "auto" -> derive [identity.name] from routed agent
Changes:
- Core logic: resolveResponsePrefix() in identity.ts accepts
optional channel/accountId and walks the cascade
- resolveEffectiveMessagesConfig() passes channel context through
- Types: responsePrefix added to WhatsApp, Telegram, Discord, Slack,
Signal, iMessage, Google Chat, MS Teams, Feishu, BlueBubbles configs
- Zod schemas: responsePrefix added for config validation
- All channel handlers wired: telegram, discord, slack, signal,
imessage, line, heartbeat runner, route-reply, native commands
- 23 new tests covering backward compat, channel/account levels,
full cascade, auto keyword, empty string stops, unknown fallthrough
Fully backward compatible - no existing config is affected.
Fixes #8857
* fix: address CI lint + review feedback
- Replace Record<string, any> with proper typed helpers (no-explicit-any)
- Add curly braces to single-line if returns (eslint curly)
- Fix JSDoc: 'Per-channel' → 'channel/account' on shared config types
- Extract getChannelConfig() helper for type-safe dynamic key access
* fix: finish responsePrefix overrides (#9001) (thanks @mudrii)
* fix: normalize prefix wiring and types (#9001) (thanks @mudrii)
---------
Co-authored-by: Gustavo Madeira Santana <gumadeiras@gmail.com>
2026-02-05 05:16:34 +08:00
|
|
|
import { createReplyPrefixOptions } from "../../../channels/reply-prefix.js";
|
2026-01-23 22:55:41 +00:00
|
|
|
import { createTypingCallbacks } from "../../../channels/typing.js";
|
2026-01-23 23:04:09 +00:00
|
|
|
import { resolveStorePath, updateLastRoute } from "../../../config/sessions.js";
|
2026-01-14 01:08:15 +00:00
|
|
|
import { danger, logVerbose, shouldLogVerbose } from "../../../globals.js";
|
|
|
|
|
import { removeSlackReaction } from "../../actions.js";
|
2026-02-16 15:43:29 -05:00
|
|
|
import { createSlackDraftStream } from "../../draft-stream.js";
|
2026-02-16 16:56:41 -05:00
|
|
|
import {
|
|
|
|
|
applyAppendOnlyStreamUpdate,
|
|
|
|
|
buildStatusFinalPreviewText,
|
|
|
|
|
resolveSlackStreamMode,
|
|
|
|
|
} from "../../stream-mode.js";
|
2026-01-14 01:08:15 +00:00
|
|
|
import { resolveSlackThreadTargets } from "../../threading.js";
|
|
|
|
|
import { createSlackReplyDeliveryPlan, deliverReplies } from "../replies.js";
|
2026-02-17 13:36:48 +09:00
|
|
|
import type { PreparedSlackMessage } from "./types.js";
|
2026-01-14 01:08:15 +00:00
|
|
|
|
2026-01-14 14:31:43 +00:00
|
|
|
export async function dispatchPreparedSlackMessage(prepared: PreparedSlackMessage) {
|
2026-01-14 01:08:15 +00:00
|
|
|
const { ctx, account, message, route } = prepared;
|
|
|
|
|
const cfg = ctx.cfg;
|
|
|
|
|
const runtime = ctx.runtime;
|
|
|
|
|
|
2026-01-23 23:04:09 +00:00
|
|
|
if (prepared.isDirectMessage) {
|
|
|
|
|
const sessionCfg = cfg.session;
|
|
|
|
|
const storePath = resolveStorePath(sessionCfg?.store, {
|
|
|
|
|
agentId: route.agentId,
|
|
|
|
|
});
|
|
|
|
|
await updateLastRoute({
|
|
|
|
|
storePath,
|
|
|
|
|
sessionKey: route.mainSessionKey,
|
|
|
|
|
deliveryContext: {
|
|
|
|
|
channel: "slack",
|
|
|
|
|
to: `user:${message.user}`,
|
|
|
|
|
accountId: route.accountId,
|
|
|
|
|
},
|
|
|
|
|
ctx: prepared.ctxPayload,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-14 01:08:15 +00:00
|
|
|
const { statusThreadTs } = resolveSlackThreadTargets({
|
|
|
|
|
message,
|
|
|
|
|
replyToMode: ctx.replyToMode,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const messageTs = message.ts ?? message.event_ts;
|
|
|
|
|
const incomingThreadTs = message.thread_ts;
|
|
|
|
|
let didSetStatus = false;
|
|
|
|
|
|
|
|
|
|
// Shared mutable ref for "replyToMode=first". Both tool + auto-reply flows
|
|
|
|
|
// mark this to ensure only the first reply is threaded.
|
|
|
|
|
const hasRepliedRef = { value: false };
|
|
|
|
|
const replyPlan = createSlackReplyDeliveryPlan({
|
|
|
|
|
replyToMode: ctx.replyToMode,
|
|
|
|
|
incomingThreadTs,
|
|
|
|
|
messageTs,
|
|
|
|
|
hasRepliedRef,
|
|
|
|
|
});
|
|
|
|
|
|
2026-01-23 23:20:07 +00:00
|
|
|
const typingTarget = statusThreadTs ? `${message.channel}/${statusThreadTs}` : message.channel;
|
2026-01-23 22:55:41 +00:00
|
|
|
const typingCallbacks = createTypingCallbacks({
|
|
|
|
|
start: async () => {
|
|
|
|
|
didSetStatus = true;
|
|
|
|
|
await ctx.setSlackThreadStatus({
|
|
|
|
|
channelId: message.channel,
|
|
|
|
|
threadTs: statusThreadTs,
|
|
|
|
|
status: "is typing...",
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
stop: async () => {
|
2026-01-31 16:19:20 +09:00
|
|
|
if (!didSetStatus) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2026-01-23 23:30:52 +00:00
|
|
|
didSetStatus = false;
|
2026-01-23 22:55:41 +00:00
|
|
|
await ctx.setSlackThreadStatus({
|
|
|
|
|
channelId: message.channel,
|
|
|
|
|
threadTs: statusThreadTs,
|
|
|
|
|
status: "",
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
onStartError: (err) => {
|
2026-01-23 23:20:07 +00:00
|
|
|
logTypingFailure({
|
|
|
|
|
log: (message) => runtime.error?.(danger(message)),
|
|
|
|
|
channel: "slack",
|
|
|
|
|
action: "start",
|
|
|
|
|
target: typingTarget,
|
|
|
|
|
error: err,
|
|
|
|
|
});
|
2026-01-23 22:55:41 +00:00
|
|
|
},
|
|
|
|
|
onStopError: (err) => {
|
2026-01-23 23:20:07 +00:00
|
|
|
logTypingFailure({
|
|
|
|
|
log: (message) => runtime.error?.(danger(message)),
|
|
|
|
|
channel: "slack",
|
|
|
|
|
action: "stop",
|
|
|
|
|
target: typingTarget,
|
|
|
|
|
error: err,
|
|
|
|
|
});
|
2026-01-23 22:55:41 +00:00
|
|
|
},
|
|
|
|
|
});
|
2026-01-14 01:08:15 +00:00
|
|
|
|
feat: per-channel responsePrefix override (#9001)
* feat: per-channel responsePrefix override
Add responsePrefix field to all channel config types and Zod schemas,
enabling per-channel and per-account outbound response prefix overrides.
Resolution cascade (most specific wins):
L1: channels.<ch>.accounts.<id>.responsePrefix
L2: channels.<ch>.responsePrefix
L3: (reserved for channels.defaults)
L4: messages.responsePrefix (existing global)
Semantics:
- undefined -> inherit from parent level
- empty string -> explicitly no prefix (stops cascade)
- "auto" -> derive [identity.name] from routed agent
Changes:
- Core logic: resolveResponsePrefix() in identity.ts accepts
optional channel/accountId and walks the cascade
- resolveEffectiveMessagesConfig() passes channel context through
- Types: responsePrefix added to WhatsApp, Telegram, Discord, Slack,
Signal, iMessage, Google Chat, MS Teams, Feishu, BlueBubbles configs
- Zod schemas: responsePrefix added for config validation
- All channel handlers wired: telegram, discord, slack, signal,
imessage, line, heartbeat runner, route-reply, native commands
- 23 new tests covering backward compat, channel/account levels,
full cascade, auto keyword, empty string stops, unknown fallthrough
Fully backward compatible - no existing config is affected.
Fixes #8857
* fix: address CI lint + review feedback
- Replace Record<string, any> with proper typed helpers (no-explicit-any)
- Add curly braces to single-line if returns (eslint curly)
- Fix JSDoc: 'Per-channel' → 'channel/account' on shared config types
- Extract getChannelConfig() helper for type-safe dynamic key access
* fix: finish responsePrefix overrides (#9001) (thanks @mudrii)
* fix: normalize prefix wiring and types (#9001) (thanks @mudrii)
---------
Co-authored-by: Gustavo Madeira Santana <gumadeiras@gmail.com>
2026-02-05 05:16:34 +08:00
|
|
|
const { onModelSelected, ...prefixOptions } = createReplyPrefixOptions({
|
|
|
|
|
cfg,
|
|
|
|
|
agentId: route.agentId,
|
|
|
|
|
channel: "slack",
|
|
|
|
|
accountId: route.accountId,
|
|
|
|
|
});
|
feat: add dynamic template variables to messages.responsePrefix (#923)
Adds support for template variables in `messages.responsePrefix` that
resolve dynamically at runtime with the actual model used (including
after fallback).
Supported variables (case-insensitive):
- {model} - short model name (e.g., "claude-opus-4-5", "gpt-4o")
- {modelFull} - full model identifier (e.g., "anthropic/claude-opus-4-5")
- {provider} - provider name (e.g., "anthropic", "openai")
- {thinkingLevel} or {think} - thinking level ("high", "low", "off")
- {identity.name} or {identityName} - agent identity name
Example: "[{model} | think:{thinkingLevel}]" → "[claude-opus-4-5 | think:high]"
Variables show the actual model used after fallback, not the intended
model. Unresolved variables remain as literal text.
Implementation:
- New module: src/auto-reply/reply/response-prefix-template.ts
- Template interpolation in normalize-reply.ts via context provider
- onModelSelected callback in agent-runner-execution.ts
- Updated all 6 provider message handlers (web, signal, discord,
telegram, slack, imessage)
- 27 unit tests covering all variables and edge cases
- Documentation in docs/gateway/configuration.md and JSDoc
Fixes #923
2026-01-14 23:05:08 -05:00
|
|
|
|
2026-01-14 14:31:43 +00:00
|
|
|
const { dispatcher, replyOptions, markDispatchIdle } = createReplyDispatcherWithTyping({
|
feat: per-channel responsePrefix override (#9001)
* feat: per-channel responsePrefix override
Add responsePrefix field to all channel config types and Zod schemas,
enabling per-channel and per-account outbound response prefix overrides.
Resolution cascade (most specific wins):
L1: channels.<ch>.accounts.<id>.responsePrefix
L2: channels.<ch>.responsePrefix
L3: (reserved for channels.defaults)
L4: messages.responsePrefix (existing global)
Semantics:
- undefined -> inherit from parent level
- empty string -> explicitly no prefix (stops cascade)
- "auto" -> derive [identity.name] from routed agent
Changes:
- Core logic: resolveResponsePrefix() in identity.ts accepts
optional channel/accountId and walks the cascade
- resolveEffectiveMessagesConfig() passes channel context through
- Types: responsePrefix added to WhatsApp, Telegram, Discord, Slack,
Signal, iMessage, Google Chat, MS Teams, Feishu, BlueBubbles configs
- Zod schemas: responsePrefix added for config validation
- All channel handlers wired: telegram, discord, slack, signal,
imessage, line, heartbeat runner, route-reply, native commands
- 23 new tests covering backward compat, channel/account levels,
full cascade, auto keyword, empty string stops, unknown fallthrough
Fully backward compatible - no existing config is affected.
Fixes #8857
* fix: address CI lint + review feedback
- Replace Record<string, any> with proper typed helpers (no-explicit-any)
- Add curly braces to single-line if returns (eslint curly)
- Fix JSDoc: 'Per-channel' → 'channel/account' on shared config types
- Extract getChannelConfig() helper for type-safe dynamic key access
* fix: finish responsePrefix overrides (#9001) (thanks @mudrii)
* fix: normalize prefix wiring and types (#9001) (thanks @mudrii)
---------
Co-authored-by: Gustavo Madeira Santana <gumadeiras@gmail.com>
2026-02-05 05:16:34 +08:00
|
|
|
...prefixOptions,
|
2026-01-14 14:31:43 +00:00
|
|
|
humanDelay: resolveHumanDelayConfig(cfg, route.agentId),
|
|
|
|
|
deliver: async (payload) => {
|
2026-02-16 15:43:29 -05:00
|
|
|
const mediaCount = payload.mediaUrls?.length ?? (payload.mediaUrl ? 1 : 0);
|
|
|
|
|
const draftMessageId = draftStream?.messageId();
|
|
|
|
|
const draftChannelId = draftStream?.channelId();
|
|
|
|
|
const finalText = payload.text;
|
|
|
|
|
const canFinalizeViaPreviewEdit =
|
2026-02-16 16:56:41 -05:00
|
|
|
streamMode !== "status_final" &&
|
2026-02-16 15:43:29 -05:00
|
|
|
mediaCount === 0 &&
|
|
|
|
|
!payload.isError &&
|
|
|
|
|
typeof finalText === "string" &&
|
|
|
|
|
finalText.trim().length > 0 &&
|
|
|
|
|
typeof draftMessageId === "string" &&
|
|
|
|
|
typeof draftChannelId === "string";
|
|
|
|
|
|
|
|
|
|
if (canFinalizeViaPreviewEdit) {
|
|
|
|
|
draftStream?.stop();
|
|
|
|
|
try {
|
|
|
|
|
await ctx.app.client.chat.update({
|
2026-02-16 15:50:44 -05:00
|
|
|
token: ctx.botToken,
|
2026-02-16 15:43:29 -05:00
|
|
|
channel: draftChannelId,
|
|
|
|
|
ts: draftMessageId,
|
|
|
|
|
text: finalText.trim(),
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
} catch (err) {
|
|
|
|
|
logVerbose(
|
|
|
|
|
`slack: preview final edit failed; falling back to standard send (${String(err)})`,
|
|
|
|
|
);
|
|
|
|
|
}
|
2026-02-16 16:56:41 -05:00
|
|
|
} else if (streamMode === "status_final" && hasStreamedMessage) {
|
|
|
|
|
try {
|
|
|
|
|
const statusChannelId = draftStream?.channelId();
|
|
|
|
|
const statusMessageId = draftStream?.messageId();
|
|
|
|
|
if (statusChannelId && statusMessageId) {
|
|
|
|
|
await ctx.app.client.chat.update({
|
|
|
|
|
token: ctx.botToken,
|
|
|
|
|
channel: statusChannelId,
|
|
|
|
|
ts: statusMessageId,
|
|
|
|
|
text: "Status: complete. Final answer posted below.",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
logVerbose(`slack: status_final completion update failed (${String(err)})`);
|
|
|
|
|
}
|
2026-02-16 15:43:29 -05:00
|
|
|
} else if (mediaCount > 0) {
|
2026-02-16 16:07:00 -05:00
|
|
|
await draftStream?.clear();
|
|
|
|
|
hasStreamedMessage = false;
|
2026-02-16 15:43:29 -05:00
|
|
|
}
|
|
|
|
|
|
2026-01-14 14:31:43 +00:00
|
|
|
const replyThreadTs = replyPlan.nextThreadTs();
|
|
|
|
|
await deliverReplies({
|
|
|
|
|
replies: [payload],
|
|
|
|
|
target: prepared.replyTarget,
|
|
|
|
|
token: ctx.botToken,
|
|
|
|
|
accountId: account.accountId,
|
|
|
|
|
runtime,
|
|
|
|
|
textLimit: ctx.textLimit,
|
|
|
|
|
replyThreadTs,
|
|
|
|
|
});
|
|
|
|
|
replyPlan.markSent();
|
|
|
|
|
},
|
|
|
|
|
onError: (err, info) => {
|
|
|
|
|
runtime.error?.(danger(`slack ${info.kind} reply failed: ${String(err)}`));
|
2026-01-23 22:55:41 +00:00
|
|
|
typingCallbacks.onIdle?.();
|
2026-01-14 14:31:43 +00:00
|
|
|
},
|
2026-01-23 22:55:41 +00:00
|
|
|
onReplyStart: typingCallbacks.onReplyStart,
|
|
|
|
|
onIdle: typingCallbacks.onIdle,
|
2026-01-14 14:31:43 +00:00
|
|
|
});
|
2026-01-14 01:08:15 +00:00
|
|
|
|
2026-02-16 15:43:29 -05:00
|
|
|
const draftStream = createSlackDraftStream({
|
|
|
|
|
target: prepared.replyTarget,
|
|
|
|
|
token: ctx.botToken,
|
|
|
|
|
accountId: account.accountId,
|
|
|
|
|
maxChars: Math.min(ctx.textLimit, 4000),
|
|
|
|
|
resolveThreadTs: () => replyPlan.nextThreadTs(),
|
|
|
|
|
onMessageSent: () => replyPlan.markSent(),
|
|
|
|
|
log: logVerbose,
|
|
|
|
|
warn: logVerbose,
|
|
|
|
|
});
|
|
|
|
|
let hasStreamedMessage = false;
|
2026-02-16 16:56:41 -05:00
|
|
|
const streamMode = resolveSlackStreamMode(account.config.streamMode);
|
|
|
|
|
let appendRenderedText = "";
|
|
|
|
|
let appendSourceText = "";
|
|
|
|
|
let statusUpdateCount = 0;
|
2026-02-16 15:43:29 -05:00
|
|
|
const updateDraftFromPartial = (text?: string) => {
|
|
|
|
|
const trimmed = text?.trimEnd();
|
|
|
|
|
if (!trimmed) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2026-02-16 16:56:41 -05:00
|
|
|
|
|
|
|
|
if (streamMode === "append") {
|
|
|
|
|
const next = applyAppendOnlyStreamUpdate({
|
|
|
|
|
incoming: trimmed,
|
|
|
|
|
rendered: appendRenderedText,
|
|
|
|
|
source: appendSourceText,
|
|
|
|
|
});
|
|
|
|
|
appendRenderedText = next.rendered;
|
|
|
|
|
appendSourceText = next.source;
|
|
|
|
|
if (!next.changed) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
draftStream.update(next.rendered);
|
|
|
|
|
hasStreamedMessage = true;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (streamMode === "status_final") {
|
|
|
|
|
statusUpdateCount += 1;
|
|
|
|
|
if (statusUpdateCount > 1 && statusUpdateCount % 4 !== 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
draftStream.update(buildStatusFinalPreviewText(statusUpdateCount));
|
|
|
|
|
hasStreamedMessage = true;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-16 15:43:29 -05:00
|
|
|
draftStream.update(trimmed);
|
|
|
|
|
hasStreamedMessage = true;
|
|
|
|
|
};
|
|
|
|
|
|
2026-01-23 22:51:37 +00:00
|
|
|
const { queuedFinal, counts } = await dispatchInboundMessage({
|
2026-01-14 01:08:15 +00:00
|
|
|
ctx: prepared.ctxPayload,
|
|
|
|
|
cfg,
|
|
|
|
|
dispatcher,
|
|
|
|
|
replyOptions: {
|
|
|
|
|
...replyOptions,
|
|
|
|
|
skillFilter: prepared.channelConfig?.skills,
|
|
|
|
|
hasRepliedRef,
|
|
|
|
|
disableBlockStreaming:
|
|
|
|
|
typeof account.config.blockStreaming === "boolean"
|
|
|
|
|
? !account.config.blockStreaming
|
|
|
|
|
: undefined,
|
feat: per-channel responsePrefix override (#9001)
* feat: per-channel responsePrefix override
Add responsePrefix field to all channel config types and Zod schemas,
enabling per-channel and per-account outbound response prefix overrides.
Resolution cascade (most specific wins):
L1: channels.<ch>.accounts.<id>.responsePrefix
L2: channels.<ch>.responsePrefix
L3: (reserved for channels.defaults)
L4: messages.responsePrefix (existing global)
Semantics:
- undefined -> inherit from parent level
- empty string -> explicitly no prefix (stops cascade)
- "auto" -> derive [identity.name] from routed agent
Changes:
- Core logic: resolveResponsePrefix() in identity.ts accepts
optional channel/accountId and walks the cascade
- resolveEffectiveMessagesConfig() passes channel context through
- Types: responsePrefix added to WhatsApp, Telegram, Discord, Slack,
Signal, iMessage, Google Chat, MS Teams, Feishu, BlueBubbles configs
- Zod schemas: responsePrefix added for config validation
- All channel handlers wired: telegram, discord, slack, signal,
imessage, line, heartbeat runner, route-reply, native commands
- 23 new tests covering backward compat, channel/account levels,
full cascade, auto keyword, empty string stops, unknown fallthrough
Fully backward compatible - no existing config is affected.
Fixes #8857
* fix: address CI lint + review feedback
- Replace Record<string, any> with proper typed helpers (no-explicit-any)
- Add curly braces to single-line if returns (eslint curly)
- Fix JSDoc: 'Per-channel' → 'channel/account' on shared config types
- Extract getChannelConfig() helper for type-safe dynamic key access
* fix: finish responsePrefix overrides (#9001) (thanks @mudrii)
* fix: normalize prefix wiring and types (#9001) (thanks @mudrii)
---------
Co-authored-by: Gustavo Madeira Santana <gumadeiras@gmail.com>
2026-02-05 05:16:34 +08:00
|
|
|
onModelSelected,
|
2026-02-16 15:43:29 -05:00
|
|
|
onPartialReply: async (payload) => {
|
|
|
|
|
updateDraftFromPartial(payload.text);
|
|
|
|
|
},
|
|
|
|
|
onAssistantMessageStart: async () => {
|
|
|
|
|
if (hasStreamedMessage) {
|
|
|
|
|
draftStream.forceNewMessage();
|
|
|
|
|
hasStreamedMessage = false;
|
2026-02-16 16:56:41 -05:00
|
|
|
appendRenderedText = "";
|
|
|
|
|
appendSourceText = "";
|
|
|
|
|
statusUpdateCount = 0;
|
2026-02-16 15:43:29 -05:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
onReasoningEnd: async () => {
|
|
|
|
|
if (hasStreamedMessage) {
|
|
|
|
|
draftStream.forceNewMessage();
|
|
|
|
|
hasStreamedMessage = false;
|
2026-02-16 16:56:41 -05:00
|
|
|
appendRenderedText = "";
|
|
|
|
|
appendSourceText = "";
|
|
|
|
|
statusUpdateCount = 0;
|
2026-02-16 15:43:29 -05:00
|
|
|
}
|
|
|
|
|
},
|
2026-01-14 01:08:15 +00:00
|
|
|
},
|
|
|
|
|
});
|
2026-02-16 15:43:29 -05:00
|
|
|
await draftStream.flush();
|
|
|
|
|
draftStream.stop();
|
2026-01-14 01:08:15 +00:00
|
|
|
markDispatchIdle();
|
|
|
|
|
|
2026-01-25 21:28:46 -06:00
|
|
|
const anyReplyDelivered = queuedFinal || (counts.block ?? 0) > 0 || (counts.final ?? 0) > 0;
|
|
|
|
|
|
|
|
|
|
if (!anyReplyDelivered) {
|
2026-02-16 16:07:00 -05:00
|
|
|
await draftStream.clear();
|
2026-01-23 22:36:43 +00:00
|
|
|
if (prepared.isRoomish) {
|
|
|
|
|
clearHistoryEntriesIfEnabled({
|
2026-01-14 01:08:15 +00:00
|
|
|
historyMap: ctx.channelHistories,
|
|
|
|
|
historyKey: prepared.historyKey,
|
2026-01-23 22:36:43 +00:00
|
|
|
limit: ctx.historyLimit,
|
2026-01-14 01:08:15 +00:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (shouldLogVerbose()) {
|
|
|
|
|
const finalCount = counts.final;
|
|
|
|
|
logVerbose(
|
|
|
|
|
`slack: delivered ${finalCount} reply${finalCount === 1 ? "" : "ies"} to ${prepared.replyTarget}`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-23 22:29:47 +00:00
|
|
|
removeAckReactionAfterReply({
|
|
|
|
|
removeAfterReply: ctx.removeAckAfterReply,
|
|
|
|
|
ackReactionPromise: prepared.ackReactionPromise,
|
|
|
|
|
ackReactionValue: prepared.ackReactionValue,
|
|
|
|
|
remove: () =>
|
|
|
|
|
removeSlackReaction(
|
|
|
|
|
message.channel,
|
|
|
|
|
prepared.ackReactionMessageTs ?? "",
|
|
|
|
|
prepared.ackReactionValue,
|
|
|
|
|
{
|
|
|
|
|
token: ctx.botToken,
|
|
|
|
|
client: ctx.app.client,
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
onError: (err) => {
|
2026-01-23 23:20:07 +00:00
|
|
|
logAckFailure({
|
|
|
|
|
log: logVerbose,
|
|
|
|
|
channel: "slack",
|
|
|
|
|
target: `${message.channel}/${message.ts}`,
|
|
|
|
|
error: err,
|
|
|
|
|
});
|
2026-01-23 22:29:47 +00:00
|
|
|
},
|
|
|
|
|
});
|
2026-01-14 01:08:15 +00:00
|
|
|
|
2026-01-23 22:36:43 +00:00
|
|
|
if (prepared.isRoomish) {
|
|
|
|
|
clearHistoryEntriesIfEnabled({
|
2026-01-14 01:08:15 +00:00
|
|
|
historyMap: ctx.channelHistories,
|
|
|
|
|
historyKey: prepared.historyKey,
|
2026-01-23 22:36:43 +00:00
|
|
|
limit: ctx.historyLimit,
|
2026-01-14 01:08:15 +00:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|