Root causes: 1. Multiple plugin subsystems each called startModeratorPresence, creating competing WebSocket connections to the same bot token. Discord only allows one connection per bot → 4008 rate limit → infinite reconnect loop (1000+ connects → token reset by Discord) 2. Invalid session (op 9) handler called scheduleReconnect, but the new connection would also get kicked → cascading reconnects Fixes: - Singleton guard: startModeratorPresence is a no-op if already started - cleanup() nullifies old ws handlers before creating new connection - Stale ws check: all callbacks verify they belong to current ws - Exponential backoff with cap (max 60s) instead of fixed 2-5s delay - heartbeat ACK tracking: detect zombie connections - Non-recoverable codes (4004) properly stop all reconnection
WhisperGate Plugin
Hook strategy
message:receivedcaches a per-session decision from deterministic rules.before_model_resolveappliesproviderOverride + modelOverridewhen decision says no-reply.before_prompt_buildprepends instruction你的这次发言必须以🔚作为结尾。when decision is:bypass_senderend_symbol:*
Rules (in order)
- non-discord -> skip
- bypass sender -> skip
- end symbol matched -> skip
- else -> no-reply override
Config
See docs/CONFIG.example.json.
Required:
noReplyProvidernoReplyModel
Optional:
enabled(default true)discordOnly(default true)listMode(human-list|agent-list, defaulthuman-list)humanList(default [])agentList(default [])channelPoliciesFile(per-channel overrides in a standalone JSON file)enableWhispergatePolicyTool(default true)
Unified optional tool:
whispergateway_tools- Discord actions:
channel-private-create,channel-private-update,member-list - Policy actions:
policy-get,policy-set-channel,policy-delete-channel
- Discord actions:
bypassUserIds(deprecated alias ofhumanList)endSymbols(default ["🔚"])enableDiscordControlTool(default true)discordControlApiBaseUrl(defaulthttp://127.0.0.1:8790)discordControlApiTokendiscordControlCallerIdenableDebugLogs(default false)debugLogChannelIds(default [], empty = all channels when debug enabled)
Per-channel policy file example: docs/channel-policies.example.json.
Policy file behavior:
- loaded once on startup into memory
- runtime decisions read memory state only
- direct file edits do NOT affect memory state
whispergateway_toolspolicy actions update memory first, then persist to file (atomic write)
Optional tool: whispergateway_tools
This plugin registers one unified optional tool: whispergateway_tools.
To use it, add tool allowlist entry for either:
- tool name:
whispergateway_tools - plugin id:
whispergate
Supported actions:
- Discord:
channel-private-create,channel-private-update,member-list - Policy:
policy-get,policy-set-channel,policy-delete-channel
Debug logging:
- set
enableDebugLogs: trueto emit detailed hook diagnostics - optionally set
debugLogChannelIdsto only log selected channel IDs - logs include key ctx fields + decision status at
message_received,before_model_resolve,before_prompt_build