@openclaw/discord 2026.5.2 → 2026.5.3-beta.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/access-B9ujuUtS.js +89 -0
- package/dist/account-inspect-C2UEUhbG.js +81 -0
- package/dist/account-inspect-api.js +10 -0
- package/dist/accounts-BKnkNaoA.js +128 -0
- package/dist/action-runtime-api.js +2 -0
- package/dist/agent-components.runtime-DUhLr9hy.js +4 -0
- package/dist/allow-list-ek-1hMKN.js +336 -0
- package/dist/api-DzNBVTto.js +130 -0
- package/dist/api.js +24 -0
- package/dist/approval-handler.runtime-v8nzQHlT.js +426 -0
- package/dist/approval-native-DqWGp0bM.js +153 -0
- package/dist/approval-shared-DKnwwjZM.js +93 -0
- package/dist/audit-CJ92YD6J.js +102 -0
- package/dist/channel-B3aTtBj1.js +745 -0
- package/dist/channel-access-ewDxhd9q.js +62 -0
- package/dist/channel-actions-TNih7k3w.js +140 -0
- package/dist/channel-actions.runtime-CaPytiY4.js +236 -0
- package/dist/channel-api-CTSWMrnD.js +21 -0
- package/dist/channel-config-api.js +2 -0
- package/dist/channel-plugin-api.js +2 -0
- package/dist/channel.setup-Dt4tIDrl.js +336 -0
- package/dist/components-BapWDmDM.js +760 -0
- package/dist/config-api-CFZtoMaS.js +2 -0
- package/dist/config-schema-DwFkL904.js +252 -0
- package/dist/configured-state.js +6 -0
- package/dist/contract-api.js +8 -0
- package/dist/conversation-identity-BN9wSmxJ.js +31 -0
- package/dist/directory-cache-D93eSrpB.js +62 -0
- package/dist/directory-config-LyMP0sdv.js +49 -0
- package/dist/directory-contract-api.js +2 -0
- package/dist/directory-live-BQapdpkZ.js +101 -0
- package/dist/discord-D1kDh0X_.js +2751 -0
- package/dist/doctor-B2G7WqO0.js +244 -0
- package/dist/doctor-contract-D3pSutkb.js +383 -0
- package/dist/doctor-contract-api.js +2 -0
- package/dist/doctor-shared-DU8RcnF5.js +4 -0
- package/dist/format-D8TsaXxW.js +24 -0
- package/dist/gateway-registry-BKG4KIVC.js +74 -0
- package/dist/handle-action.guild-admin-BuqsSVXu.js +283 -0
- package/dist/inbound-context-e_oBBJtF.js +51 -0
- package/dist/index.js +26 -0
- package/dist/manager.runtime-M2aAa7qA.js +1019 -0
- package/dist/mentions-BPZUaFk7.js +88 -0
- package/dist/message-handler-D6JfFV3P.js +381 -0
- package/dist/message-handler.preflight-DqaF3vHm.js +1022 -0
- package/dist/message-handler.process-tl3Nwnhr.js +1124 -0
- package/dist/message-utils-Dmgu-7fC.js +512 -0
- package/dist/normalize-B-ktw-T_.js +275 -0
- package/dist/outbound-adapter-DJf9_sfH.js +451 -0
- package/dist/outbound-session-route-uHGLDP-Y.js +43 -0
- package/dist/pluralkit-voQvSN3g.js +22 -0
- package/dist/preflight-audio-BpYtUAT6.js +72 -0
- package/dist/preflight-audio.runtime-BAGmU6uO.js +7 -0
- package/dist/preview-streaming-C0O92Qqz.js +14 -0
- package/dist/probe-DcNEodPI.js +139 -0
- package/dist/probe.runtime-P-e4r1Hl.js +2 -0
- package/dist/provider-CMvXOp-3.js +8440 -0
- package/dist/provider-session.runtime-JFemrDZT.js +6 -0
- package/dist/provider.runtime-BO007oR2.js +2 -0
- package/dist/reply-delivery-D9So77a6.js +131 -0
- package/dist/resolve-allowlist-common-DqqFY_qa.js +34 -0
- package/dist/resolve-channels-CGPntufJ.js +265 -0
- package/dist/resolve-users-CDvSlW0V.js +120 -0
- package/dist/rolldown-runtime-C3SqQTfK.js +28 -0
- package/dist/route-resolution-BYiC-6Cc.js +236 -0
- package/dist/runtime-K9RT6Egn.js +8 -0
- package/dist/runtime-api.actions.js +3 -0
- package/dist/runtime-api.js +31 -0
- package/dist/runtime-api.lookup.js +7 -0
- package/dist/runtime-api.monitor-BC-XN0tY.js +6 -0
- package/dist/runtime-api.monitor.js +9 -0
- package/dist/runtime-api.send.js +6 -0
- package/dist/runtime-api.threads.js +6 -0
- package/dist/runtime-n5xZHW55.js +1001 -0
- package/dist/runtime-setter-api.js +2 -0
- package/dist/secret-config-contract-CoGryS5c.js +115 -0
- package/dist/secret-contract-api.js +2 -0
- package/dist/security-audit-Cdz2iq3m.js +120 -0
- package/dist/security-audit-contract-api.js +2 -0
- package/dist/security-audit.runtime-DBV1T1_N.js +2 -0
- package/dist/security-contract-api.js +2 -0
- package/dist/security-contract-ei3Mz8Sa.js +26 -0
- package/dist/security-doctor-CzTzpXV8.js +18 -0
- package/dist/send-B_frVn_Q.js +845 -0
- package/dist/send.components-B1EgHAds.js +468 -0
- package/dist/send.outbound-DlBAuW7y.js +211 -0
- package/dist/send.shared-Db0opnak.js +708 -0
- package/dist/sender-identity-BiSDAk2P.js +43 -0
- package/dist/session-contract-goJZckp2.js +6 -0
- package/dist/session-key-api.js +2 -0
- package/dist/session-key-normalization-Daag9II6.js +23 -0
- package/dist/setup-entry.js +11 -0
- package/dist/setup-plugin-api.js +2 -0
- package/dist/shared-CqlrJmSs.js +166 -0
- package/dist/shared-interactive-KgJjCqnB.js +79 -0
- package/dist/subagent-hooks-api.js +22 -0
- package/dist/subagent-hooks-mEK5ARfP.js +113 -0
- package/dist/system-events-Bu9jmO4W.js +34 -0
- package/dist/targets-kKlbZ4ai.js +3 -0
- package/dist/test-api.js +45 -0
- package/dist/thread-binding-api.js +4 -0
- package/dist/thread-bindings-Bj1R-6QH.js +256 -0
- package/dist/thread-bindings.discord-api-ClPMuIr8.js +184 -0
- package/dist/thread-bindings.manager-BaN0l4y8.js +535 -0
- package/dist/thread-bindings.session-updates-TTP020qQ.js +54 -0
- package/dist/thread-bindings.state-Dzu1gCE7.js +318 -0
- package/dist/threading-CWhdYHVx.js +475 -0
- package/dist/timeouts-C7jeTtGs.js +52 -0
- package/dist/timeouts.js +2 -0
- package/dist/token-D-w3Rigl.js +42 -0
- package/dist/typing-CJiowRTZ.js +15 -0
- package/package.json +14 -6
- package/account-inspect-api.ts +0 -6
- package/action-runtime-api.ts +0 -1
- package/api.ts +0 -132
- package/channel-config-api.ts +0 -1
- package/channel-plugin-api.ts +0 -3
- package/config-api.ts +0 -4
- package/configured-state.ts +0 -6
- package/contract-api.ts +0 -21
- package/directory-contract-api.ts +0 -4
- package/doctor-contract-api.ts +0 -1
- package/index.test.ts +0 -13
- package/index.ts +0 -24
- package/runtime-api.actions.ts +0 -15
- package/runtime-api.lookup.ts +0 -22
- package/runtime-api.monitor.ts +0 -50
- package/runtime-api.send.ts +0 -79
- package/runtime-api.threads.ts +0 -30
- package/runtime-api.ts +0 -180
- package/runtime-setter-api.ts +0 -3
- package/secret-contract-api.ts +0 -4
- package/security-audit-contract-api.ts +0 -1
- package/security-contract-api.ts +0 -4
- package/session-key-api.ts +0 -1
- package/setup-entry.ts +0 -9
- package/setup-plugin-api.ts +0 -3
- package/src/account-inspect.test.ts +0 -126
- package/src/account-inspect.ts +0 -132
- package/src/accounts.test.ts +0 -247
- package/src/accounts.ts +0 -196
- package/src/actions/handle-action.guild-admin.ts +0 -411
- package/src/actions/handle-action.test.ts +0 -306
- package/src/actions/handle-action.ts +0 -372
- package/src/actions/runtime.guild.ts +0 -446
- package/src/actions/runtime.messaging.messages.ts +0 -205
- package/src/actions/runtime.messaging.reactions.ts +0 -67
- package/src/actions/runtime.messaging.runtime.ts +0 -69
- package/src/actions/runtime.messaging.send.ts +0 -248
- package/src/actions/runtime.messaging.shared.ts +0 -97
- package/src/actions/runtime.messaging.ts +0 -37
- package/src/actions/runtime.moderation-shared.ts +0 -48
- package/src/actions/runtime.moderation.authz.test.ts +0 -151
- package/src/actions/runtime.moderation.ts +0 -116
- package/src/actions/runtime.presence.test.ts +0 -160
- package/src/actions/runtime.presence.ts +0 -117
- package/src/actions/runtime.shared.ts +0 -83
- package/src/actions/runtime.test.ts +0 -1087
- package/src/actions/runtime.ts +0 -87
- package/src/api-barrel.test.ts +0 -80
- package/src/api.test.ts +0 -130
- package/src/api.ts +0 -169
- package/src/approval-handler.runtime.test.ts +0 -41
- package/src/approval-handler.runtime.ts +0 -632
- package/src/approval-native.test.ts +0 -330
- package/src/approval-native.ts +0 -219
- package/src/approval-runtime.ts +0 -14
- package/src/approval-shared.ts +0 -53
- package/src/audit-core.ts +0 -141
- package/src/audit.test.ts +0 -145
- package/src/audit.ts +0 -32
- package/src/channel-actions.contract.test.ts +0 -45
- package/src/channel-actions.runtime.ts +0 -1
- package/src/channel-actions.test.ts +0 -275
- package/src/channel-actions.ts +0 -203
- package/src/channel-api.ts +0 -29
- package/src/channel.conversation.ts +0 -159
- package/src/channel.loaders.ts +0 -47
- package/src/channel.runtime.ts +0 -1
- package/src/channel.setup.ts +0 -12
- package/src/channel.test.ts +0 -571
- package/src/channel.ts +0 -629
- package/src/chunk.test.ts +0 -157
- package/src/chunk.ts +0 -321
- package/src/client.proxy.test.ts +0 -176
- package/src/client.test.ts +0 -76
- package/src/client.ts +0 -132
- package/src/component-custom-id.ts +0 -72
- package/src/components-registry.ts +0 -356
- package/src/components.builders.ts +0 -409
- package/src/components.modal.ts +0 -124
- package/src/components.parse.ts +0 -407
- package/src/components.test.ts +0 -312
- package/src/components.ts +0 -54
- package/src/components.types.ts +0 -187
- package/src/config-schema.test.ts +0 -325
- package/src/config-schema.ts +0 -6
- package/src/config-ui-hints.ts +0 -249
- package/src/conversation-identity.ts +0 -58
- package/src/delivery-retry.ts +0 -56
- package/src/directory-cache.ts +0 -116
- package/src/directory-config.ts +0 -58
- package/src/directory-contract.test.ts +0 -129
- package/src/directory-live.test.ts +0 -126
- package/src/directory-live.ts +0 -135
- package/src/doctor-contract.ts +0 -477
- package/src/doctor-shared.ts +0 -5
- package/src/doctor.test.ts +0 -405
- package/src/doctor.ts +0 -340
- package/src/draft-chunking.test.ts +0 -64
- package/src/draft-chunking.ts +0 -43
- package/src/draft-stream.test.ts +0 -159
- package/src/draft-stream.ts +0 -154
- package/src/error-body.ts +0 -38
- package/src/exec-approvals.test.ts +0 -88
- package/src/exec-approvals.ts +0 -110
- package/src/gateway-logging.test.ts +0 -98
- package/src/gateway-logging.ts +0 -67
- package/src/group-policy.ts +0 -113
- package/src/guilds.ts +0 -29
- package/src/inbound-context.contract.test.ts +0 -11
- package/src/interactive-dispatch.ts +0 -104
- package/src/internal/api.commands.ts +0 -51
- package/src/internal/api.guild.ts +0 -164
- package/src/internal/api.interactions.ts +0 -53
- package/src/internal/api.messages.ts +0 -113
- package/src/internal/api.reactions.ts +0 -38
- package/src/internal/api.test.ts +0 -262
- package/src/internal/api.ts +0 -61
- package/src/internal/api.users.ts +0 -19
- package/src/internal/api.webhooks.ts +0 -13
- package/src/internal/client.test.ts +0 -440
- package/src/internal/client.ts +0 -310
- package/src/internal/command-deploy.ts +0 -297
- package/src/internal/commands.ts +0 -188
- package/src/internal/components.base.ts +0 -65
- package/src/internal/components.message.ts +0 -279
- package/src/internal/components.modal.ts +0 -95
- package/src/internal/components.ts +0 -31
- package/src/internal/discord.ts +0 -11
- package/src/internal/embeds.ts +0 -35
- package/src/internal/entity-cache.ts +0 -98
- package/src/internal/event-queue.ts +0 -162
- package/src/internal/gateway-close-codes.ts +0 -25
- package/src/internal/gateway-dispatch.ts +0 -96
- package/src/internal/gateway-identify-limiter.ts +0 -26
- package/src/internal/gateway-lifecycle.ts +0 -61
- package/src/internal/gateway-rate-limit.ts +0 -104
- package/src/internal/gateway.test.ts +0 -603
- package/src/internal/gateway.ts +0 -476
- package/src/internal/interaction-dispatch.test.ts +0 -148
- package/src/internal/interaction-dispatch.ts +0 -162
- package/src/internal/interaction-options.ts +0 -98
- package/src/internal/interaction-response.ts +0 -53
- package/src/internal/interactions.test.ts +0 -325
- package/src/internal/interactions.ts +0 -378
- package/src/internal/listeners.ts +0 -85
- package/src/internal/live-smoke.live.test.ts +0 -26
- package/src/internal/modal-fields.ts +0 -95
- package/src/internal/payload.ts +0 -69
- package/src/internal/rest-body.ts +0 -115
- package/src/internal/rest-errors.ts +0 -88
- package/src/internal/rest-routes.ts +0 -50
- package/src/internal/rest-scheduler.ts +0 -557
- package/src/internal/rest.test.ts +0 -673
- package/src/internal/rest.ts +0 -322
- package/src/internal/schemas.ts +0 -36
- package/src/internal/structures.test.ts +0 -43
- package/src/internal/structures.ts +0 -280
- package/src/internal/test-builders.test-support.ts +0 -167
- package/src/internal/voice.ts +0 -49
- package/src/media-detection.ts +0 -28
- package/src/mentions.test.ts +0 -111
- package/src/mentions.ts +0 -147
- package/src/monitor/access-groups.ts +0 -55
- package/src/monitor/ack-reactions.ts +0 -70
- package/src/monitor/acp-bind-here.integration.test.ts +0 -211
- package/src/monitor/agent-components-auth.ts +0 -7
- package/src/monitor/agent-components-context.ts +0 -154
- package/src/monitor/agent-components-data.ts +0 -224
- package/src/monitor/agent-components-dm-auth.ts +0 -221
- package/src/monitor/agent-components-guild-auth.ts +0 -322
- package/src/monitor/agent-components-helpers.runtime.ts +0 -5
- package/src/monitor/agent-components-helpers.ts +0 -34
- package/src/monitor/agent-components-reply.ts +0 -10
- package/src/monitor/agent-components.deps.runtime.ts +0 -2
- package/src/monitor/agent-components.dispatch.ts +0 -366
- package/src/monitor/agent-components.handlers.ts +0 -303
- package/src/monitor/agent-components.modal.ts +0 -160
- package/src/monitor/agent-components.plugin-interactive.ts +0 -187
- package/src/monitor/agent-components.runtime.ts +0 -14
- package/src/monitor/agent-components.system-controls.ts +0 -211
- package/src/monitor/agent-components.ts +0 -70
- package/src/monitor/agent-components.types.ts +0 -58
- package/src/monitor/agent-components.wildcard-controls.ts +0 -168
- package/src/monitor/agent-components.wildcard.test.ts +0 -71
- package/src/monitor/allow-list.test.ts +0 -14
- package/src/monitor/allow-list.ts +0 -633
- package/src/monitor/auto-presence.test.ts +0 -156
- package/src/monitor/auto-presence.ts +0 -356
- package/src/monitor/channel-access.test.ts +0 -99
- package/src/monitor/channel-access.ts +0 -102
- package/src/monitor/commands.test.ts +0 -24
- package/src/monitor/commands.ts +0 -9
- package/src/monitor/dm-command-auth.test.ts +0 -197
- package/src/monitor/dm-command-auth.ts +0 -158
- package/src/monitor/dm-command-decision.test.ts +0 -113
- package/src/monitor/dm-command-decision.ts +0 -49
- package/src/monitor/exec-approvals.test.ts +0 -226
- package/src/monitor/exec-approvals.ts +0 -158
- package/src/monitor/format.ts +0 -45
- package/src/monitor/gateway-handle.ts +0 -34
- package/src/monitor/gateway-metadata.test.ts +0 -29
- package/src/monitor/gateway-metadata.ts +0 -298
- package/src/monitor/gateway-plugin.test.ts +0 -297
- package/src/monitor/gateway-plugin.ts +0 -294
- package/src/monitor/gateway-registry.ts +0 -37
- package/src/monitor/gateway-supervisor.test.ts +0 -150
- package/src/monitor/gateway-supervisor.ts +0 -206
- package/src/monitor/inbound-context.test-helpers.ts +0 -37
- package/src/monitor/inbound-context.test.ts +0 -106
- package/src/monitor/inbound-context.ts +0 -103
- package/src/monitor/inbound-dedupe.ts +0 -79
- package/src/monitor/inbound-job.test.ts +0 -203
- package/src/monitor/inbound-job.ts +0 -118
- package/src/monitor/listeners.queue.ts +0 -91
- package/src/monitor/listeners.reactions.ts +0 -610
- package/src/monitor/listeners.test.ts +0 -200
- package/src/monitor/listeners.ts +0 -150
- package/src/monitor/message-channel-info.ts +0 -96
- package/src/monitor/message-forwarded.ts +0 -107
- package/src/monitor/message-handler.batch-gate.test.ts +0 -22
- package/src/monitor/message-handler.batch-gate.ts +0 -19
- package/src/monitor/message-handler.bot-self-filter.test.ts +0 -68
- package/src/monitor/message-handler.context.ts +0 -406
- package/src/monitor/message-handler.dm-preflight.ts +0 -123
- package/src/monitor/message-handler.draft-preview.ts +0 -246
- package/src/monitor/message-handler.hydration.test.ts +0 -80
- package/src/monitor/message-handler.hydration.ts +0 -198
- package/src/monitor/message-handler.inbound-context.test.ts +0 -59
- package/src/monitor/message-handler.module-test-helpers.ts +0 -31
- package/src/monitor/message-handler.preflight-channel-access.ts +0 -86
- package/src/monitor/message-handler.preflight-channel-context.test.ts +0 -18
- package/src/monitor/message-handler.preflight-channel-context.ts +0 -58
- package/src/monitor/message-handler.preflight-context.ts +0 -54
- package/src/monitor/message-handler.preflight-helpers.ts +0 -164
- package/src/monitor/message-handler.preflight-history.ts +0 -23
- package/src/monitor/message-handler.preflight-logging.ts +0 -36
- package/src/monitor/message-handler.preflight-pluralkit.ts +0 -26
- package/src/monitor/message-handler.preflight-runtime.ts +0 -28
- package/src/monitor/message-handler.preflight-thread.ts +0 -49
- package/src/monitor/message-handler.preflight.acp-bindings.test.ts +0 -369
- package/src/monitor/message-handler.preflight.test-helpers.ts +0 -111
- package/src/monitor/message-handler.preflight.test.ts +0 -1623
- package/src/monitor/message-handler.preflight.ts +0 -679
- package/src/monitor/message-handler.preflight.types.ts +0 -110
- package/src/monitor/message-handler.process.test.ts +0 -1369
- package/src/monitor/message-handler.process.ts +0 -686
- package/src/monitor/message-handler.queue.test.ts +0 -496
- package/src/monitor/message-handler.routing-preflight.ts +0 -112
- package/src/monitor/message-handler.test-harness.ts +0 -99
- package/src/monitor/message-handler.test-helpers.ts +0 -75
- package/src/monitor/message-handler.ts +0 -274
- package/src/monitor/message-media.ts +0 -509
- package/src/monitor/message-run-queue.ts +0 -101
- package/src/monitor/message-text.ts +0 -171
- package/src/monitor/message-utils.test.ts +0 -1157
- package/src/monitor/message-utils.ts +0 -32
- package/src/monitor/model-picker-preferences.test.ts +0 -67
- package/src/monitor/model-picker-preferences.ts +0 -184
- package/src/monitor/model-picker.state.ts +0 -364
- package/src/monitor/model-picker.test-utils.ts +0 -26
- package/src/monitor/model-picker.test.ts +0 -794
- package/src/monitor/model-picker.ts +0 -38
- package/src/monitor/model-picker.view.ts +0 -695
- package/src/monitor/monitor.agent-components.test.ts +0 -375
- package/src/monitor/monitor.test.ts +0 -849
- package/src/monitor/monitor.threading-utils.test.ts +0 -598
- package/src/monitor/native-command-agent-reply.ts +0 -125
- package/src/monitor/native-command-arg-ui.ts +0 -233
- package/src/monitor/native-command-auth.ts +0 -308
- package/src/monitor/native-command-bypass.ts +0 -13
- package/src/monitor/native-command-context.test.ts +0 -98
- package/src/monitor/native-command-context.ts +0 -103
- package/src/monitor/native-command-dispatch.ts +0 -35
- package/src/monitor/native-command-model-picker-apply.ts +0 -177
- package/src/monitor/native-command-model-picker-interaction.ts +0 -461
- package/src/monitor/native-command-model-picker-ui.ts +0 -368
- package/src/monitor/native-command-reply.test.ts +0 -68
- package/src/monitor/native-command-reply.ts +0 -185
- package/src/monitor/native-command-route.ts +0 -91
- package/src/monitor/native-command-status.ts +0 -76
- package/src/monitor/native-command-ui.ts +0 -26
- package/src/monitor/native-command-ui.types.ts +0 -20
- package/src/monitor/native-command.args.ts +0 -45
- package/src/monitor/native-command.command-arg.test.ts +0 -99
- package/src/monitor/native-command.commands-allowfrom.test.ts +0 -490
- package/src/monitor/native-command.model-picker.test.ts +0 -767
- package/src/monitor/native-command.options.test.ts +0 -369
- package/src/monitor/native-command.options.ts +0 -153
- package/src/monitor/native-command.plugin-dispatch.test.ts +0 -961
- package/src/monitor/native-command.runtime.ts +0 -50
- package/src/monitor/native-command.status-direct.test.ts +0 -272
- package/src/monitor/native-command.test-helpers.ts +0 -64
- package/src/monitor/native-command.think-autocomplete.test.ts +0 -416
- package/src/monitor/native-command.ts +0 -700
- package/src/monitor/native-command.types.ts +0 -9
- package/src/monitor/native-interaction-channel-context.ts +0 -50
- package/src/monitor/preflight-audio.runtime.ts +0 -9
- package/src/monitor/preflight-audio.test.ts +0 -157
- package/src/monitor/preflight-audio.ts +0 -130
- package/src/monitor/presence-cache.ts +0 -61
- package/src/monitor/presence.test.ts +0 -44
- package/src/monitor/presence.ts +0 -50
- package/src/monitor/provider-session.runtime.ts +0 -12
- package/src/monitor/provider.acp.ts +0 -89
- package/src/monitor/provider.allowlist.test.ts +0 -149
- package/src/monitor/provider.allowlist.ts +0 -394
- package/src/monitor/provider.cleanup.ts +0 -41
- package/src/monitor/provider.commands.ts +0 -129
- package/src/monitor/provider.config-log.ts +0 -45
- package/src/monitor/provider.deploy-errors.ts +0 -362
- package/src/monitor/provider.deploy.ts +0 -221
- package/src/monitor/provider.interactions.ts +0 -160
- package/src/monitor/provider.lifecycle.test.ts +0 -713
- package/src/monitor/provider.lifecycle.ts +0 -552
- package/src/monitor/provider.proxy.test.ts +0 -745
- package/src/monitor/provider.rest-proxy.test.ts +0 -121
- package/src/monitor/provider.runtime.ts +0 -1
- package/src/monitor/provider.skill-dedupe.test.ts +0 -42
- package/src/monitor/provider.startup-log.ts +0 -32
- package/src/monitor/provider.startup.test.ts +0 -426
- package/src/monitor/provider.startup.ts +0 -330
- package/src/monitor/provider.test.ts +0 -1111
- package/src/monitor/provider.ts +0 -713
- package/src/monitor/reply-context.ts +0 -64
- package/src/monitor/reply-delivery.test.ts +0 -244
- package/src/monitor/reply-delivery.ts +0 -203
- package/src/monitor/rest-fetch.ts +0 -43
- package/src/monitor/route-resolution.test.ts +0 -204
- package/src/monitor/route-resolution.ts +0 -140
- package/src/monitor/sender-identity.ts +0 -81
- package/src/monitor/startup-status.test.ts +0 -30
- package/src/monitor/startup-status.ts +0 -10
- package/src/monitor/status.ts +0 -22
- package/src/monitor/system-events.ts +0 -55
- package/src/monitor/thread-bindings.config.ts +0 -35
- package/src/monitor/thread-bindings.discord-api.test.ts +0 -229
- package/src/monitor/thread-bindings.discord-api.ts +0 -310
- package/src/monitor/thread-bindings.lifecycle.test.ts +0 -1871
- package/src/monitor/thread-bindings.lifecycle.ts +0 -354
- package/src/monitor/thread-bindings.manager.ts +0 -553
- package/src/monitor/thread-bindings.messages.ts +0 -6
- package/src/monitor/thread-bindings.persona.test.ts +0 -34
- package/src/monitor/thread-bindings.persona.ts +0 -25
- package/src/monitor/thread-bindings.session-adapter.ts +0 -229
- package/src/monitor/thread-bindings.session-shared.ts +0 -59
- package/src/monitor/thread-bindings.session-updates.ts +0 -35
- package/src/monitor/thread-bindings.shared-state.test.ts +0 -36
- package/src/monitor/thread-bindings.state.ts +0 -540
- package/src/monitor/thread-bindings.ts +0 -48
- package/src/monitor/thread-bindings.types.ts +0 -83
- package/src/monitor/thread-channel-context.ts +0 -112
- package/src/monitor/thread-session-close.test.ts +0 -180
- package/src/monitor/thread-session-close.ts +0 -63
- package/src/monitor/thread-title.generate.test.ts +0 -197
- package/src/monitor/thread-title.test.ts +0 -31
- package/src/monitor/thread-title.ts +0 -181
- package/src/monitor/threading.auto-thread.test.ts +0 -327
- package/src/monitor/threading.auto-thread.ts +0 -287
- package/src/monitor/threading.cache.ts +0 -45
- package/src/monitor/threading.parent-info.test.ts +0 -156
- package/src/monitor/threading.starter.test.ts +0 -260
- package/src/monitor/threading.starter.ts +0 -287
- package/src/monitor/threading.ts +0 -20
- package/src/monitor/threading.types.ts +0 -102
- package/src/monitor/timeouts.ts +0 -84
- package/src/monitor/typing.test.ts +0 -42
- package/src/monitor/typing.ts +0 -17
- package/src/monitor.gateway.test.ts +0 -187
- package/src/monitor.gateway.ts +0 -75
- package/src/monitor.test.ts +0 -1397
- package/src/monitor.ts +0 -28
- package/src/normalize.test.ts +0 -56
- package/src/normalize.ts +0 -86
- package/src/outbound-adapter.interactive-order.test.ts +0 -64
- package/src/outbound-adapter.test-harness.ts +0 -207
- package/src/outbound-adapter.test.ts +0 -696
- package/src/outbound-adapter.ts +0 -291
- package/src/outbound-approval.ts +0 -29
- package/src/outbound-components.ts +0 -81
- package/src/outbound-payload.contract.test.ts +0 -38
- package/src/outbound-payload.ts +0 -134
- package/src/outbound-send-context.ts +0 -92
- package/src/outbound-session-route.test.ts +0 -34
- package/src/outbound-session-route.ts +0 -72
- package/src/pluralkit.test.ts +0 -67
- package/src/pluralkit.ts +0 -58
- package/src/preview-streaming.ts +0 -32
- package/src/probe.intents.test.ts +0 -94
- package/src/probe.parse-token.test.ts +0 -43
- package/src/probe.runtime.ts +0 -1
- package/src/probe.ts +0 -237
- package/src/proxy-fetch.ts +0 -92
- package/src/proxy-request-client.test.ts +0 -78
- package/src/proxy-request-client.ts +0 -21
- package/src/recipient-resolution.ts +0 -39
- package/src/resolve-allowlist-common.test.ts +0 -36
- package/src/resolve-allowlist-common.ts +0 -39
- package/src/resolve-channels.test.ts +0 -340
- package/src/resolve-channels.ts +0 -369
- package/src/resolve-users.test.ts +0 -222
- package/src/resolve-users.ts +0 -184
- package/src/retry.test.ts +0 -83
- package/src/retry.ts +0 -98
- package/src/runtime-api.ts +0 -64
- package/src/runtime.ts +0 -23
- package/src/secret-config-contract.ts +0 -140
- package/src/security-audit.runtime.ts +0 -1
- package/src/security-audit.test.ts +0 -246
- package/src/security-audit.ts +0 -208
- package/src/security-contract.ts +0 -47
- package/src/security-doctor.test.ts +0 -25
- package/src/security-doctor.ts +0 -20
- package/src/security.ts +0 -60
- package/src/send-target-parsing.ts +0 -14
- package/src/send.channels.ts +0 -139
- package/src/send.components.test.ts +0 -275
- package/src/send.components.ts +0 -381
- package/src/send.creates-thread.test.ts +0 -643
- package/src/send.emojis-stickers.ts +0 -57
- package/src/send.guild.ts +0 -170
- package/src/send.message-request.ts +0 -97
- package/src/send.messages.test.ts +0 -53
- package/src/send.messages.ts +0 -225
- package/src/send.outbound.ts +0 -413
- package/src/send.permissions.authz.test.ts +0 -188
- package/src/send.permissions.ts +0 -283
- package/src/send.reactions.ts +0 -155
- package/src/send.sends-basic-channel-messages.test.ts +0 -941
- package/src/send.shared.ts +0 -447
- package/src/send.test-harness.ts +0 -56
- package/src/send.ts +0 -82
- package/src/send.types.ts +0 -188
- package/src/send.typing.test.ts +0 -41
- package/src/send.typing.ts +0 -9
- package/src/send.voice.ts +0 -134
- package/src/send.webhook-activity.test.ts +0 -105
- package/src/send.webhook.proxy.test.ts +0 -191
- package/src/send.webhook.ts +0 -133
- package/src/session-contract.ts +0 -3
- package/src/session-key-normalization.test.ts +0 -44
- package/src/session-key-normalization.ts +0 -47
- package/src/setup-account-state.test.ts +0 -91
- package/src/setup-account-state.ts +0 -144
- package/src/setup-adapter.ts +0 -12
- package/src/setup-core.ts +0 -212
- package/src/setup-runtime-helpers.ts +0 -10
- package/src/setup-surface.test.ts +0 -137
- package/src/setup-surface.ts +0 -129
- package/src/shared-interactive.test.ts +0 -153
- package/src/shared-interactive.ts +0 -124
- package/src/shared.test.ts +0 -165
- package/src/shared.ts +0 -190
- package/src/status-issues.test.ts +0 -70
- package/src/status-issues.ts +0 -169
- package/src/subagent-hooks.test.ts +0 -432
- package/src/subagent-hooks.ts +0 -214
- package/src/target-parsing.ts +0 -53
- package/src/target-resolver.ts +0 -129
- package/src/targets.test.ts +0 -367
- package/src/targets.ts +0 -12
- package/src/test-http-helpers.ts +0 -10
- package/src/test-support/component-runtime.ts +0 -190
- package/src/test-support/config.ts +0 -7
- package/src/test-support/configured-binding-runtime.ts +0 -29
- package/src/test-support/partial-channel.ts +0 -26
- package/src/test-support/provider.test-support.ts +0 -545
- package/src/token.test.ts +0 -107
- package/src/token.ts +0 -60
- package/src/ui-colors.ts +0 -27
- package/src/ui.ts +0 -20
- package/src/voice/access.test.ts +0 -217
- package/src/voice/access.ts +0 -124
- package/src/voice/audio.ts +0 -173
- package/src/voice/capture-state.test.ts +0 -48
- package/src/voice/capture-state.ts +0 -120
- package/src/voice/command.test.ts +0 -164
- package/src/voice/command.ts +0 -283
- package/src/voice/config.ts +0 -8
- package/src/voice/manager.e2e.test.ts +0 -928
- package/src/voice/manager.ready-listener.test.ts +0 -37
- package/src/voice/manager.runtime.ts +0 -11
- package/src/voice/manager.ts +0 -691
- package/src/voice/prompt.test.ts +0 -16
- package/src/voice/prompt.ts +0 -17
- package/src/voice/receive-recovery.test.ts +0 -79
- package/src/voice/receive-recovery.ts +0 -159
- package/src/voice/sanitize.test.ts +0 -34
- package/src/voice/sanitize.ts +0 -32
- package/src/voice/sdk-runtime.ts +0 -14
- package/src/voice/segment.ts +0 -156
- package/src/voice/session.ts +0 -50
- package/src/voice/speaker-context.ts +0 -127
- package/src/voice/tts.ts +0 -125
- package/src/voice-message.test.ts +0 -234
- package/src/voice-message.ts +0 -444
- package/subagent-hooks-api.ts +0 -27
- package/test-api.ts +0 -4
- package/thread-binding-api.ts +0 -1
- package/timeouts.ts +0 -6
- package/tsconfig.json +0 -16
|
@@ -0,0 +1,845 @@
|
|
|
1
|
+
import { t as __exportAll } from "./rolldown-runtime-C3SqQTfK.js";
|
|
2
|
+
import { s as resolveDiscordAccount } from "./accounts-BKnkNaoA.js";
|
|
3
|
+
import { $ as createChannelMessage, At as putChannelPermission, Ct as getGuildVoiceState, Dt as listGuildRoles, Et as listGuildEmojis, Mt as removeGuildMemberRole, Nt as timeoutGuildMember, Ot as listGuildScheduledEvents, Q as listMessageReactionUsers, St as getGuildMember, Tt as listGuildChannels, X as createOwnMessageReaction, Z as deleteOwnMessageReaction, _ as readDiscordMessage, _t as createGuildEmoji, at as getChannel, bt as deleteChannelPermission, ct as listChannelMessages, dt as searchGuildMessages, et as createThread, ft as sendChannelTyping, g as readDiscordCode, gt as createGuildChannel, h as RateLimitError, ht as createGuildBan, it as editChannelMessage, jt as removeGuildMember, kt as moveGuildChannels, lt as listChannelPins, m as DiscordError, mt as addGuildMemberRole, nt as deleteChannelMessage, ot as getChannelMessage, pt as unpinChannelMessage, rt as editChannel, st as listChannelArchivedThreads, tt as deleteChannel, ut as pinChannelMessage, v as readRetryAfter, vt as createGuildScheduledEvent, wt as listGuildActiveThreads, yt as createGuildSticker } from "./discord-D1kDh0X_.js";
|
|
4
|
+
import { C as DiscordSendError, D as hasAllGuildPermissionsDiscord, E as fetchMemberGuildPermissionsDiscord, F as resolveDiscordClientAccountContext, I as resolveDiscordRest, M as createDiscordClient, O as hasAnyGuildPermissionDiscord, S as DISCORD_MAX_STICKER_BYTES, T as fetchChannelPermissionsDiscord, b as DISCORD_MAX_EMOJI_BYTES, i as formatReactionEmoji, k as parseAndResolveRecipient, l as resolveChannelId, o as normalizeEmojiName, r as buildReactionIdentifier, s as normalizeReactionEmoji, t as buildDiscordSendError, w as canViewDiscordGuildChannel, x as DISCORD_MAX_EVENT_COVER_BYTES } from "./send.shared-Db0opnak.js";
|
|
5
|
+
import { n as rewriteDiscordKnownMentions } from "./mentions-BPZUaFk7.js";
|
|
6
|
+
import { n as sendPollDiscord, r as sendStickerDiscord, t as sendMessageDiscord } from "./send.outbound-DlBAuW7y.js";
|
|
7
|
+
import { normalizeLowercaseStringOrEmpty, normalizeOptionalLowercaseString, normalizeOptionalString } from "openclaw/plugin-sdk/text-runtime";
|
|
8
|
+
import { ChannelType } from "discord-api-types/v10";
|
|
9
|
+
import crypto from "node:crypto";
|
|
10
|
+
import fs from "node:fs/promises";
|
|
11
|
+
import path from "node:path";
|
|
12
|
+
import { MEDIA_FFMPEG_MAX_AUDIO_DURATION_SECS, extensionForMime, maxBytesForKind, parseFfprobeCodecAndSampleRate, runFfmpeg, runFfprobe, unlinkIfExists } from "openclaw/plugin-sdk/media-runtime";
|
|
13
|
+
import { requireRuntimeConfig } from "openclaw/plugin-sdk/plugin-config-runtime";
|
|
14
|
+
import { loadWebMediaRaw } from "openclaw/plugin-sdk/web-media";
|
|
15
|
+
import { formatErrorMessage } from "openclaw/plugin-sdk/error-runtime";
|
|
16
|
+
import { recordChannelActivity } from "openclaw/plugin-sdk/channel-activity-runtime";
|
|
17
|
+
import { resolvePreferredOpenClawTmpDir } from "openclaw/plugin-sdk/temp-path";
|
|
18
|
+
import { fetchWithSsrFGuard } from "openclaw/plugin-sdk/ssrf-runtime";
|
|
19
|
+
//#region extensions/discord/src/send.channels.ts
|
|
20
|
+
async function createChannelDiscord(payload, opts) {
|
|
21
|
+
const rest = resolveDiscordRest(opts);
|
|
22
|
+
const body = { name: payload.name };
|
|
23
|
+
if (payload.type !== void 0) body.type = payload.type;
|
|
24
|
+
if (payload.parentId) body.parent_id = payload.parentId;
|
|
25
|
+
if (payload.topic) body.topic = payload.topic;
|
|
26
|
+
if (payload.position !== void 0) body.position = payload.position;
|
|
27
|
+
if (payload.nsfw !== void 0) body.nsfw = payload.nsfw;
|
|
28
|
+
return await createGuildChannel(rest, payload.guildId, { body });
|
|
29
|
+
}
|
|
30
|
+
async function editChannelDiscord(payload, opts) {
|
|
31
|
+
const rest = resolveDiscordRest(opts);
|
|
32
|
+
const body = {};
|
|
33
|
+
if (payload.name !== void 0) body.name = payload.name;
|
|
34
|
+
if (payload.topic !== void 0) body.topic = payload.topic;
|
|
35
|
+
if (payload.position !== void 0) body.position = payload.position;
|
|
36
|
+
if (payload.parentId !== void 0) body.parent_id = payload.parentId;
|
|
37
|
+
if (payload.nsfw !== void 0) body.nsfw = payload.nsfw;
|
|
38
|
+
if (payload.rateLimitPerUser !== void 0) body.rate_limit_per_user = payload.rateLimitPerUser;
|
|
39
|
+
if (payload.archived !== void 0) body.archived = payload.archived;
|
|
40
|
+
if (payload.locked !== void 0) body.locked = payload.locked;
|
|
41
|
+
if (payload.autoArchiveDuration !== void 0) body.auto_archive_duration = payload.autoArchiveDuration;
|
|
42
|
+
if (payload.availableTags !== void 0) body.available_tags = payload.availableTags.map((t) => ({
|
|
43
|
+
...t.id !== void 0 && { id: t.id },
|
|
44
|
+
name: t.name,
|
|
45
|
+
...t.moderated !== void 0 && { moderated: t.moderated },
|
|
46
|
+
...t.emoji_id !== void 0 && { emoji_id: t.emoji_id },
|
|
47
|
+
...t.emoji_name !== void 0 && { emoji_name: t.emoji_name }
|
|
48
|
+
}));
|
|
49
|
+
return await editChannel(rest, payload.channelId, { body });
|
|
50
|
+
}
|
|
51
|
+
async function deleteChannelDiscord(channelId, opts) {
|
|
52
|
+
await deleteChannel(resolveDiscordRest(opts), channelId);
|
|
53
|
+
return {
|
|
54
|
+
ok: true,
|
|
55
|
+
channelId
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
async function moveChannelDiscord(payload, opts) {
|
|
59
|
+
const rest = resolveDiscordRest(opts);
|
|
60
|
+
const body = [{
|
|
61
|
+
id: payload.channelId,
|
|
62
|
+
...payload.parentId !== void 0 && { parent_id: payload.parentId },
|
|
63
|
+
...payload.position !== void 0 && { position: payload.position }
|
|
64
|
+
}];
|
|
65
|
+
await moveGuildChannels(rest, payload.guildId, { body });
|
|
66
|
+
return { ok: true };
|
|
67
|
+
}
|
|
68
|
+
async function setChannelPermissionDiscord(payload, opts) {
|
|
69
|
+
const rest = resolveDiscordRest(opts);
|
|
70
|
+
const body = { type: payload.targetType };
|
|
71
|
+
if (payload.allow !== void 0) body.allow = payload.allow;
|
|
72
|
+
if (payload.deny !== void 0) body.deny = payload.deny;
|
|
73
|
+
await putChannelPermission(rest, payload.channelId, payload.targetId, { body });
|
|
74
|
+
return { ok: true };
|
|
75
|
+
}
|
|
76
|
+
async function removeChannelPermissionDiscord(channelId, targetId, opts) {
|
|
77
|
+
await deleteChannelPermission(resolveDiscordRest(opts), channelId, targetId);
|
|
78
|
+
return { ok: true };
|
|
79
|
+
}
|
|
80
|
+
//#endregion
|
|
81
|
+
//#region extensions/discord/src/send.emojis-stickers.ts
|
|
82
|
+
async function listGuildEmojisDiscord(guildId, opts) {
|
|
83
|
+
return await listGuildEmojis(resolveDiscordRest(opts), guildId);
|
|
84
|
+
}
|
|
85
|
+
async function uploadEmojiDiscord(payload, opts) {
|
|
86
|
+
const rest = resolveDiscordRest(opts);
|
|
87
|
+
const media = await loadWebMediaRaw(payload.mediaUrl, DISCORD_MAX_EMOJI_BYTES);
|
|
88
|
+
const contentType = normalizeOptionalLowercaseString(media.contentType);
|
|
89
|
+
if (!contentType || ![
|
|
90
|
+
"image/png",
|
|
91
|
+
"image/jpeg",
|
|
92
|
+
"image/jpg",
|
|
93
|
+
"image/gif"
|
|
94
|
+
].includes(contentType)) throw new Error("Discord emoji uploads require a PNG, JPG, or GIF image");
|
|
95
|
+
const image = `data:${contentType};base64,${media.buffer.toString("base64")}`;
|
|
96
|
+
const roleIds = (payload.roleIds ?? []).map((id) => id.trim()).filter(Boolean);
|
|
97
|
+
return await createGuildEmoji(rest, payload.guildId, { body: {
|
|
98
|
+
name: normalizeEmojiName(payload.name, "Emoji name"),
|
|
99
|
+
image,
|
|
100
|
+
roles: roleIds.length ? roleIds : void 0
|
|
101
|
+
} });
|
|
102
|
+
}
|
|
103
|
+
async function uploadStickerDiscord(payload, opts) {
|
|
104
|
+
const rest = resolveDiscordRest(opts);
|
|
105
|
+
const media = await loadWebMediaRaw(payload.mediaUrl, DISCORD_MAX_STICKER_BYTES);
|
|
106
|
+
const contentType = normalizeOptionalLowercaseString(media.contentType);
|
|
107
|
+
if (!contentType || ![
|
|
108
|
+
"image/png",
|
|
109
|
+
"image/apng",
|
|
110
|
+
"application/json"
|
|
111
|
+
].includes(contentType)) throw new Error("Discord sticker uploads require a PNG, APNG, or Lottie JSON file");
|
|
112
|
+
return await createGuildSticker(rest, payload.guildId, {
|
|
113
|
+
multipartStyle: "form",
|
|
114
|
+
body: {
|
|
115
|
+
name: normalizeEmojiName(payload.name, "Sticker name"),
|
|
116
|
+
description: normalizeEmojiName(payload.description, "Sticker description"),
|
|
117
|
+
tags: normalizeEmojiName(payload.tags, "Sticker tags"),
|
|
118
|
+
files: [{
|
|
119
|
+
data: media.buffer,
|
|
120
|
+
fieldName: "file",
|
|
121
|
+
name: media.fileName ?? "sticker",
|
|
122
|
+
contentType
|
|
123
|
+
}]
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
//#endregion
|
|
128
|
+
//#region extensions/discord/src/send.guild.ts
|
|
129
|
+
async function fetchMemberInfoDiscord(guildId, userId, opts) {
|
|
130
|
+
return await getGuildMember(resolveDiscordRest(opts), guildId, userId);
|
|
131
|
+
}
|
|
132
|
+
async function fetchRoleInfoDiscord(guildId, opts) {
|
|
133
|
+
return await listGuildRoles(resolveDiscordRest(opts), guildId);
|
|
134
|
+
}
|
|
135
|
+
async function addRoleDiscord(payload, opts) {
|
|
136
|
+
await addGuildMemberRole(resolveDiscordRest(opts), payload.guildId, payload.userId, payload.roleId);
|
|
137
|
+
return { ok: true };
|
|
138
|
+
}
|
|
139
|
+
async function removeRoleDiscord(payload, opts) {
|
|
140
|
+
await removeGuildMemberRole(resolveDiscordRest(opts), payload.guildId, payload.userId, payload.roleId);
|
|
141
|
+
return { ok: true };
|
|
142
|
+
}
|
|
143
|
+
async function fetchChannelInfoDiscord(channelId, opts) {
|
|
144
|
+
return await getChannel(resolveDiscordRest(opts), channelId);
|
|
145
|
+
}
|
|
146
|
+
async function listGuildChannelsDiscord(guildId, opts) {
|
|
147
|
+
return await listGuildChannels(resolveDiscordRest(opts), guildId);
|
|
148
|
+
}
|
|
149
|
+
async function fetchVoiceStatusDiscord(guildId, userId, opts) {
|
|
150
|
+
return await getGuildVoiceState(resolveDiscordRest(opts), guildId, userId);
|
|
151
|
+
}
|
|
152
|
+
async function listScheduledEventsDiscord(guildId, opts) {
|
|
153
|
+
return await listGuildScheduledEvents(resolveDiscordRest(opts), guildId);
|
|
154
|
+
}
|
|
155
|
+
const ALLOWED_EVENT_COVER_TYPES = new Set([
|
|
156
|
+
"image/png",
|
|
157
|
+
"image/jpeg",
|
|
158
|
+
"image/jpg",
|
|
159
|
+
"image/gif"
|
|
160
|
+
]);
|
|
161
|
+
async function resolveEventCoverImage(imageUrl, opts) {
|
|
162
|
+
const media = await loadWebMediaRaw(imageUrl, DISCORD_MAX_EVENT_COVER_BYTES, { localRoots: opts?.localRoots });
|
|
163
|
+
const contentType = normalizeOptionalLowercaseString(media.contentType);
|
|
164
|
+
if (!contentType || !ALLOWED_EVENT_COVER_TYPES.has(contentType)) throw new Error(`Discord event cover images must be PNG, JPG, or GIF (got ${contentType ?? "unknown"})`);
|
|
165
|
+
return `data:${contentType};base64,${media.buffer.toString("base64")}`;
|
|
166
|
+
}
|
|
167
|
+
async function createScheduledEventDiscord(guildId, payload, opts) {
|
|
168
|
+
return await createGuildScheduledEvent(resolveDiscordRest(opts), guildId, payload);
|
|
169
|
+
}
|
|
170
|
+
async function timeoutMemberDiscord(payload, opts) {
|
|
171
|
+
const rest = resolveDiscordRest(opts);
|
|
172
|
+
let until = payload.until;
|
|
173
|
+
if (!until && payload.durationMinutes) {
|
|
174
|
+
const ms = payload.durationMinutes * 60 * 1e3;
|
|
175
|
+
until = new Date(Date.now() + ms).toISOString();
|
|
176
|
+
}
|
|
177
|
+
return await timeoutGuildMember(rest, payload.guildId, payload.userId, {
|
|
178
|
+
body: { communication_disabled_until: until ?? null },
|
|
179
|
+
headers: payload.reason ? { "X-Audit-Log-Reason": encodeURIComponent(payload.reason) } : void 0
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
async function kickMemberDiscord(payload, opts) {
|
|
183
|
+
await removeGuildMember(resolveDiscordRest(opts), payload.guildId, payload.userId, { headers: payload.reason ? { "X-Audit-Log-Reason": encodeURIComponent(payload.reason) } : void 0 });
|
|
184
|
+
return { ok: true };
|
|
185
|
+
}
|
|
186
|
+
async function banMemberDiscord(payload, opts) {
|
|
187
|
+
const rest = resolveDiscordRest(opts);
|
|
188
|
+
const deleteMessageDays = typeof payload.deleteMessageDays === "number" && Number.isFinite(payload.deleteMessageDays) ? Math.min(Math.max(Math.floor(payload.deleteMessageDays), 0), 7) : void 0;
|
|
189
|
+
await createGuildBan(rest, payload.guildId, payload.userId, {
|
|
190
|
+
body: deleteMessageDays !== void 0 ? { delete_message_days: deleteMessageDays } : void 0,
|
|
191
|
+
headers: payload.reason ? { "X-Audit-Log-Reason": encodeURIComponent(payload.reason) } : void 0
|
|
192
|
+
});
|
|
193
|
+
return { ok: true };
|
|
194
|
+
}
|
|
195
|
+
//#endregion
|
|
196
|
+
//#region extensions/discord/src/send.messages.ts
|
|
197
|
+
function formatDiscordThreadInitialMessageError(error) {
|
|
198
|
+
return error instanceof Error ? error.message : String(error);
|
|
199
|
+
}
|
|
200
|
+
var DiscordThreadInitialMessageError = class extends Error {
|
|
201
|
+
constructor(thread, error) {
|
|
202
|
+
const initialMessageError = formatDiscordThreadInitialMessageError(error);
|
|
203
|
+
super(`Discord thread was created, but sending the initial message failed: ${initialMessageError}`);
|
|
204
|
+
this.name = "DiscordThreadInitialMessageError";
|
|
205
|
+
this.initialMessageError = initialMessageError;
|
|
206
|
+
this.thread = thread;
|
|
207
|
+
}
|
|
208
|
+
};
|
|
209
|
+
async function readMessagesDiscord(channelId, query = {}, opts) {
|
|
210
|
+
const rest = resolveDiscordRest(opts);
|
|
211
|
+
const limit = typeof query.limit === "number" && Number.isFinite(query.limit) ? Math.min(Math.max(Math.floor(query.limit), 1), 100) : void 0;
|
|
212
|
+
const params = {};
|
|
213
|
+
if (limit) params.limit = limit;
|
|
214
|
+
if (query.before) params.before = query.before;
|
|
215
|
+
if (query.after) params.after = query.after;
|
|
216
|
+
if (query.around) params.around = query.around;
|
|
217
|
+
return await listChannelMessages(rest, channelId, params);
|
|
218
|
+
}
|
|
219
|
+
async function fetchMessageDiscord(channelId, messageId, opts) {
|
|
220
|
+
return await getChannelMessage(resolveDiscordRest(opts), channelId, messageId);
|
|
221
|
+
}
|
|
222
|
+
async function editMessageDiscord(channelId, messageId, payload, opts) {
|
|
223
|
+
return await editChannelMessage(resolveDiscordRest(opts), channelId, messageId, { body: { content: payload.content } });
|
|
224
|
+
}
|
|
225
|
+
async function deleteMessageDiscord(channelId, messageId, opts) {
|
|
226
|
+
await deleteChannelMessage(resolveDiscordRest(opts), channelId, messageId);
|
|
227
|
+
return { ok: true };
|
|
228
|
+
}
|
|
229
|
+
async function pinMessageDiscord(channelId, messageId, opts) {
|
|
230
|
+
await pinChannelMessage(resolveDiscordRest(opts), channelId, messageId);
|
|
231
|
+
return { ok: true };
|
|
232
|
+
}
|
|
233
|
+
async function unpinMessageDiscord(channelId, messageId, opts) {
|
|
234
|
+
await unpinChannelMessage(resolveDiscordRest(opts), channelId, messageId);
|
|
235
|
+
return { ok: true };
|
|
236
|
+
}
|
|
237
|
+
async function listPinsDiscord(channelId, opts) {
|
|
238
|
+
return await listChannelPins(resolveDiscordRest(opts), channelId);
|
|
239
|
+
}
|
|
240
|
+
async function createThreadDiscord(channelId, payload, opts) {
|
|
241
|
+
const rest = resolveDiscordRest(opts);
|
|
242
|
+
const body = { name: payload.name };
|
|
243
|
+
if (payload.autoArchiveMinutes) body.auto_archive_duration = payload.autoArchiveMinutes;
|
|
244
|
+
if (!payload.messageId && payload.type !== void 0) body.type = payload.type;
|
|
245
|
+
let channelType;
|
|
246
|
+
if (!payload.messageId) try {
|
|
247
|
+
channelType = (await getChannel(rest, channelId))?.type;
|
|
248
|
+
} catch {
|
|
249
|
+
channelType = void 0;
|
|
250
|
+
}
|
|
251
|
+
const isForumLike = channelType === ChannelType.GuildForum || channelType === ChannelType.GuildMedia;
|
|
252
|
+
if (isForumLike) {
|
|
253
|
+
body.message = { content: payload.content?.trim() ? payload.content : payload.name };
|
|
254
|
+
if (payload.appliedTags?.length) body.applied_tags = payload.appliedTags;
|
|
255
|
+
}
|
|
256
|
+
if (!payload.messageId && !isForumLike && body.type === void 0) body.type = ChannelType.PublicThread;
|
|
257
|
+
const thread = await createThread(rest, channelId, { body }, payload.messageId);
|
|
258
|
+
if (!isForumLike && payload.content?.trim() && "id" in thread) try {
|
|
259
|
+
await createChannelMessage(rest, thread.id, { body: { content: payload.content } });
|
|
260
|
+
} catch (error) {
|
|
261
|
+
throw new DiscordThreadInitialMessageError(thread, error);
|
|
262
|
+
}
|
|
263
|
+
return thread;
|
|
264
|
+
}
|
|
265
|
+
async function listThreadsDiscord(payload, opts) {
|
|
266
|
+
const rest = resolveDiscordRest(opts);
|
|
267
|
+
if (payload.includeArchived) {
|
|
268
|
+
if (!payload.channelId) throw new Error("channelId required to list archived threads");
|
|
269
|
+
const params = {};
|
|
270
|
+
if (payload.before) params.before = payload.before;
|
|
271
|
+
if (payload.limit) params.limit = payload.limit;
|
|
272
|
+
return await listChannelArchivedThreads(rest, payload.channelId, params);
|
|
273
|
+
}
|
|
274
|
+
return await listGuildActiveThreads(rest, payload.guildId);
|
|
275
|
+
}
|
|
276
|
+
async function searchMessagesDiscord(query, opts) {
|
|
277
|
+
const rest = resolveDiscordRest(opts);
|
|
278
|
+
const params = new URLSearchParams();
|
|
279
|
+
params.set("content", query.content);
|
|
280
|
+
if (query.channelIds?.length) for (const channelId of query.channelIds) params.append("channel_id", channelId);
|
|
281
|
+
if (query.authorIds?.length) for (const authorId of query.authorIds) params.append("author_id", authorId);
|
|
282
|
+
if (query.limit) {
|
|
283
|
+
const limit = Math.min(Math.max(Math.floor(query.limit), 1), 25);
|
|
284
|
+
params.set("limit", String(limit));
|
|
285
|
+
}
|
|
286
|
+
return await searchGuildMessages(rest, query.guildId, params);
|
|
287
|
+
}
|
|
288
|
+
//#endregion
|
|
289
|
+
//#region extensions/discord/src/send.webhook.ts
|
|
290
|
+
function resolveWebhookExecutionUrl(params) {
|
|
291
|
+
const baseUrl = new URL(`https://discord.com/api/v10/webhooks/${encodeURIComponent(params.webhookId)}/${encodeURIComponent(params.webhookToken)}`);
|
|
292
|
+
baseUrl.searchParams.set("wait", params.wait === false ? "false" : "true");
|
|
293
|
+
if (params.threadId !== void 0 && params.threadId !== null && params.threadId !== "") baseUrl.searchParams.set("thread_id", String(params.threadId));
|
|
294
|
+
return baseUrl.toString();
|
|
295
|
+
}
|
|
296
|
+
function coerceWebhookErrorBody(raw) {
|
|
297
|
+
if (!raw) return;
|
|
298
|
+
try {
|
|
299
|
+
return JSON.parse(raw);
|
|
300
|
+
} catch {
|
|
301
|
+
return { message: raw.slice(0, 200) };
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
async function throwWebhookResponseError(response) {
|
|
305
|
+
const parsed = coerceWebhookErrorBody(await response.text().catch(() => ""));
|
|
306
|
+
if (response.status === 429) throw new RateLimitError(response, {
|
|
307
|
+
message: readDiscordMessage(parsed, "Rate limited"),
|
|
308
|
+
retry_after: readRetryAfter(parsed, response, 1),
|
|
309
|
+
code: readDiscordCode(parsed),
|
|
310
|
+
global: parsed && typeof parsed === "object" && "global" in parsed ? Boolean(parsed.global) : false
|
|
311
|
+
});
|
|
312
|
+
throw new DiscordError(response, parsed);
|
|
313
|
+
}
|
|
314
|
+
async function sendWebhookMessageDiscord(text, opts) {
|
|
315
|
+
const webhookId = normalizeOptionalString(opts.webhookId) ?? "";
|
|
316
|
+
const webhookToken = normalizeOptionalString(opts.webhookToken) ?? "";
|
|
317
|
+
if (!webhookId || !webhookToken) throw new Error("Discord webhook id/token are required");
|
|
318
|
+
const replyTo = normalizeOptionalString(opts.replyTo) ?? "";
|
|
319
|
+
const messageReference = replyTo ? {
|
|
320
|
+
message_id: replyTo,
|
|
321
|
+
fail_if_not_exists: false
|
|
322
|
+
} : void 0;
|
|
323
|
+
const { account, proxyFetch } = resolveDiscordClientAccountContext({
|
|
324
|
+
cfg: opts.cfg,
|
|
325
|
+
accountId: opts.accountId
|
|
326
|
+
});
|
|
327
|
+
const rewrittenText = rewriteDiscordKnownMentions(text, {
|
|
328
|
+
accountId: account.accountId,
|
|
329
|
+
mentionAliases: account.config.mentionAliases
|
|
330
|
+
});
|
|
331
|
+
const response = await (proxyFetch ?? fetch)(resolveWebhookExecutionUrl({
|
|
332
|
+
webhookId,
|
|
333
|
+
webhookToken,
|
|
334
|
+
threadId: opts.threadId,
|
|
335
|
+
wait: opts.wait
|
|
336
|
+
}), {
|
|
337
|
+
method: "POST",
|
|
338
|
+
headers: { "content-type": "application/json" },
|
|
339
|
+
body: JSON.stringify({
|
|
340
|
+
content: rewrittenText,
|
|
341
|
+
username: normalizeOptionalString(opts.username),
|
|
342
|
+
avatar_url: normalizeOptionalString(opts.avatarUrl),
|
|
343
|
+
...messageReference ? { message_reference: messageReference } : {}
|
|
344
|
+
})
|
|
345
|
+
});
|
|
346
|
+
if (!response.ok) await throwWebhookResponseError(response);
|
|
347
|
+
const payload = await response.json().catch(() => ({}));
|
|
348
|
+
try {
|
|
349
|
+
recordChannelActivity({
|
|
350
|
+
channel: "discord",
|
|
351
|
+
accountId: account.accountId,
|
|
352
|
+
direction: "outbound"
|
|
353
|
+
});
|
|
354
|
+
} catch {}
|
|
355
|
+
return {
|
|
356
|
+
messageId: payload.id || "unknown",
|
|
357
|
+
channelId: payload.channel_id ? payload.channel_id : opts.threadId ? String(opts.threadId) : ""
|
|
358
|
+
};
|
|
359
|
+
}
|
|
360
|
+
//#endregion
|
|
361
|
+
//#region extensions/discord/src/voice-message.ts
|
|
362
|
+
/**
|
|
363
|
+
* Discord Voice Message Support
|
|
364
|
+
*
|
|
365
|
+
* Implements sending voice messages via Discord's API.
|
|
366
|
+
* Voice messages require:
|
|
367
|
+
* - OGG/Opus format audio
|
|
368
|
+
* - Waveform data (base64 encoded, up to 256 samples, 0-255 values)
|
|
369
|
+
* - Duration in seconds
|
|
370
|
+
* - Message flag 8192 (IS_VOICE_MESSAGE)
|
|
371
|
+
* - No other content (text, embeds, etc.)
|
|
372
|
+
*/
|
|
373
|
+
const DISCORD_VOICE_MESSAGE_FLAG = 8192;
|
|
374
|
+
const SUPPRESS_NOTIFICATIONS_FLAG = 4096;
|
|
375
|
+
const WAVEFORM_SAMPLES = 256;
|
|
376
|
+
const DISCORD_OPUS_SAMPLE_RATE_HZ = 48e3;
|
|
377
|
+
function createRateLimitError(response, body, request) {
|
|
378
|
+
return new RateLimitError(response, body, request ?? new Request("https://discord.com/api/v10/channels/voice/messages", { method: "POST" }));
|
|
379
|
+
}
|
|
380
|
+
/**
|
|
381
|
+
* Get audio duration using ffprobe
|
|
382
|
+
*/
|
|
383
|
+
async function getAudioDuration(filePath) {
|
|
384
|
+
try {
|
|
385
|
+
const stdout = await runFfprobe([
|
|
386
|
+
"-v",
|
|
387
|
+
"error",
|
|
388
|
+
"-show_entries",
|
|
389
|
+
"format=duration",
|
|
390
|
+
"-of",
|
|
391
|
+
"csv=p=0",
|
|
392
|
+
filePath
|
|
393
|
+
]);
|
|
394
|
+
const duration = Number.parseFloat(stdout.trim());
|
|
395
|
+
if (Number.isNaN(duration)) throw new Error("Could not parse duration");
|
|
396
|
+
return Math.round(duration * 100) / 100;
|
|
397
|
+
} catch (err) {
|
|
398
|
+
const errMessage = formatErrorMessage(err);
|
|
399
|
+
throw new Error(`Failed to get audio duration: ${errMessage}`, { cause: err });
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
/**
|
|
403
|
+
* Generate waveform data from audio file using ffmpeg
|
|
404
|
+
* Returns base64 encoded byte array of amplitude samples (0-255)
|
|
405
|
+
*/
|
|
406
|
+
async function generateWaveform(filePath) {
|
|
407
|
+
try {
|
|
408
|
+
return await generateWaveformFromPcm(filePath);
|
|
409
|
+
} catch {
|
|
410
|
+
return generatePlaceholderWaveform();
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
/**
|
|
414
|
+
* Generate waveform by extracting raw PCM data and sampling amplitudes
|
|
415
|
+
*/
|
|
416
|
+
async function generateWaveformFromPcm(filePath) {
|
|
417
|
+
const tempDir = resolvePreferredOpenClawTmpDir();
|
|
418
|
+
const tempPcm = path.join(tempDir, `waveform-${crypto.randomUUID()}.raw`);
|
|
419
|
+
try {
|
|
420
|
+
await runFfmpeg([
|
|
421
|
+
"-y",
|
|
422
|
+
"-i",
|
|
423
|
+
filePath,
|
|
424
|
+
"-vn",
|
|
425
|
+
"-sn",
|
|
426
|
+
"-dn",
|
|
427
|
+
"-t",
|
|
428
|
+
String(MEDIA_FFMPEG_MAX_AUDIO_DURATION_SECS),
|
|
429
|
+
"-f",
|
|
430
|
+
"s16le",
|
|
431
|
+
"-acodec",
|
|
432
|
+
"pcm_s16le",
|
|
433
|
+
"-ac",
|
|
434
|
+
"1",
|
|
435
|
+
"-ar",
|
|
436
|
+
"8000",
|
|
437
|
+
tempPcm
|
|
438
|
+
]);
|
|
439
|
+
const pcmData = await fs.readFile(tempPcm);
|
|
440
|
+
const samples = new Int16Array(pcmData.buffer, pcmData.byteOffset, pcmData.byteLength / 2);
|
|
441
|
+
const step = Math.max(1, Math.floor(samples.length / WAVEFORM_SAMPLES));
|
|
442
|
+
const waveform = [];
|
|
443
|
+
for (let i = 0; i < WAVEFORM_SAMPLES && i * step < samples.length; i++) {
|
|
444
|
+
let sum = 0;
|
|
445
|
+
let count = 0;
|
|
446
|
+
for (let j = 0; j < step && i * step + j < samples.length; j++) {
|
|
447
|
+
sum += Math.abs(samples[i * step + j]);
|
|
448
|
+
count++;
|
|
449
|
+
}
|
|
450
|
+
const avg = count > 0 ? sum / count : 0;
|
|
451
|
+
const normalized = Math.min(255, Math.round(avg / 32767 * 255));
|
|
452
|
+
waveform.push(normalized);
|
|
453
|
+
}
|
|
454
|
+
while (waveform.length < WAVEFORM_SAMPLES) waveform.push(0);
|
|
455
|
+
return Buffer.from(waveform).toString("base64");
|
|
456
|
+
} finally {
|
|
457
|
+
await unlinkIfExists(tempPcm);
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
/**
|
|
461
|
+
* Generate a placeholder waveform (for when audio processing fails)
|
|
462
|
+
*/
|
|
463
|
+
function generatePlaceholderWaveform() {
|
|
464
|
+
const waveform = [];
|
|
465
|
+
for (let i = 0; i < WAVEFORM_SAMPLES; i++) {
|
|
466
|
+
const value = Math.round(128 + 64 * Math.sin(i / WAVEFORM_SAMPLES * Math.PI * 8));
|
|
467
|
+
waveform.push(Math.min(255, Math.max(0, value)));
|
|
468
|
+
}
|
|
469
|
+
return Buffer.from(waveform).toString("base64");
|
|
470
|
+
}
|
|
471
|
+
/**
|
|
472
|
+
* Convert audio file to OGG/Opus format if needed
|
|
473
|
+
* Returns path to the OGG file (may be same as input if already OGG/Opus)
|
|
474
|
+
*/
|
|
475
|
+
async function ensureOggOpus(filePath) {
|
|
476
|
+
const trimmed = filePath.trim();
|
|
477
|
+
if (/^[a-z][a-z0-9+.-]*:\/\//i.test(trimmed)) throw new Error(`Voice message conversion requires a local file path; received a URL/protocol source: ${trimmed}`);
|
|
478
|
+
if (normalizeLowercaseStringOrEmpty(path.extname(filePath)) === ".ogg") try {
|
|
479
|
+
const { codec, sampleRateHz } = parseFfprobeCodecAndSampleRate(await runFfprobe([
|
|
480
|
+
"-v",
|
|
481
|
+
"error",
|
|
482
|
+
"-select_streams",
|
|
483
|
+
"a:0",
|
|
484
|
+
"-show_entries",
|
|
485
|
+
"stream=codec_name,sample_rate",
|
|
486
|
+
"-of",
|
|
487
|
+
"csv=p=0",
|
|
488
|
+
filePath
|
|
489
|
+
]));
|
|
490
|
+
if (codec === "opus" && sampleRateHz === DISCORD_OPUS_SAMPLE_RATE_HZ) return {
|
|
491
|
+
path: filePath,
|
|
492
|
+
cleanup: false
|
|
493
|
+
};
|
|
494
|
+
} catch {}
|
|
495
|
+
const tempDir = resolvePreferredOpenClawTmpDir();
|
|
496
|
+
const outputPath = path.join(tempDir, `voice-${crypto.randomUUID()}.ogg`);
|
|
497
|
+
await runFfmpeg([
|
|
498
|
+
"-y",
|
|
499
|
+
"-i",
|
|
500
|
+
filePath,
|
|
501
|
+
"-vn",
|
|
502
|
+
"-sn",
|
|
503
|
+
"-dn",
|
|
504
|
+
"-t",
|
|
505
|
+
String(MEDIA_FFMPEG_MAX_AUDIO_DURATION_SECS),
|
|
506
|
+
"-ar",
|
|
507
|
+
String(DISCORD_OPUS_SAMPLE_RATE_HZ),
|
|
508
|
+
"-c:a",
|
|
509
|
+
"libopus",
|
|
510
|
+
"-b:a",
|
|
511
|
+
"64k",
|
|
512
|
+
outputPath
|
|
513
|
+
]);
|
|
514
|
+
return {
|
|
515
|
+
path: outputPath,
|
|
516
|
+
cleanup: true
|
|
517
|
+
};
|
|
518
|
+
}
|
|
519
|
+
/**
|
|
520
|
+
* Get voice message metadata (duration and waveform)
|
|
521
|
+
*/
|
|
522
|
+
async function getVoiceMessageMetadata(filePath) {
|
|
523
|
+
const [durationSecs, waveform] = await Promise.all([getAudioDuration(filePath), generateWaveform(filePath)]);
|
|
524
|
+
return {
|
|
525
|
+
durationSecs,
|
|
526
|
+
waveform
|
|
527
|
+
};
|
|
528
|
+
}
|
|
529
|
+
function coerceDiscordErrorBody(raw) {
|
|
530
|
+
if (!raw) return;
|
|
531
|
+
try {
|
|
532
|
+
return JSON.parse(raw);
|
|
533
|
+
} catch {
|
|
534
|
+
return { message: raw.slice(0, 200) };
|
|
535
|
+
}
|
|
536
|
+
}
|
|
537
|
+
async function createVoiceRequestError(response, fallbackMessage) {
|
|
538
|
+
const parsed = coerceDiscordErrorBody(await response.text().catch(() => ""));
|
|
539
|
+
if (response.status === 429) throw createRateLimitError(response, {
|
|
540
|
+
message: readDiscordMessage(parsed, "You are being rate limited."),
|
|
541
|
+
retry_after: readRetryAfter(parsed, response, 1),
|
|
542
|
+
global: parsed && typeof parsed === "object" && "global" in parsed ? Boolean(parsed.global) : false
|
|
543
|
+
});
|
|
544
|
+
return new DiscordError(response, parsed ?? { message: fallbackMessage });
|
|
545
|
+
}
|
|
546
|
+
async function requestVoiceUploadUrl(params) {
|
|
547
|
+
const { response: res, release } = await fetchWithSsrFGuard({
|
|
548
|
+
url: `${params.rest.options?.baseUrl ?? "https://discord.com/api"}/channels/${params.channelId}/attachments`,
|
|
549
|
+
init: {
|
|
550
|
+
method: "POST",
|
|
551
|
+
headers: {
|
|
552
|
+
Authorization: `Bot ${params.botToken}`,
|
|
553
|
+
"Content-Type": "application/json"
|
|
554
|
+
},
|
|
555
|
+
body: JSON.stringify({ files: [{
|
|
556
|
+
filename: params.filename,
|
|
557
|
+
file_size: params.fileSize,
|
|
558
|
+
id: "0"
|
|
559
|
+
}] })
|
|
560
|
+
},
|
|
561
|
+
auditContext: "discord.voice.upload-url"
|
|
562
|
+
});
|
|
563
|
+
try {
|
|
564
|
+
if (!res.ok) throw await createVoiceRequestError(res, "Upload URL request failed");
|
|
565
|
+
return await res.json();
|
|
566
|
+
} finally {
|
|
567
|
+
await release();
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
async function uploadVoiceAttachment(params) {
|
|
571
|
+
const { response: uploadResponse, release } = await fetchWithSsrFGuard({
|
|
572
|
+
url: params.uploadUrl,
|
|
573
|
+
init: {
|
|
574
|
+
method: "PUT",
|
|
575
|
+
headers: { "Content-Type": "audio/ogg" },
|
|
576
|
+
body: new Uint8Array(params.audioBuffer)
|
|
577
|
+
},
|
|
578
|
+
auditContext: "discord.voice.attachment-upload"
|
|
579
|
+
});
|
|
580
|
+
try {
|
|
581
|
+
if (!uploadResponse.ok) throw await createVoiceRequestError(uploadResponse, "Failed to upload voice message");
|
|
582
|
+
} finally {
|
|
583
|
+
await release();
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
/**
|
|
587
|
+
* Send a voice message to Discord
|
|
588
|
+
*
|
|
589
|
+
* This follows Discord's voice message protocol:
|
|
590
|
+
* 1. Request upload URL from Discord
|
|
591
|
+
* 2. Upload the OGG file to the provided URL
|
|
592
|
+
* 3. Send the message with flag 8192 and attachment metadata
|
|
593
|
+
*/
|
|
594
|
+
async function sendDiscordVoiceMessage(rest, channelId, audioBuffer, metadata, replyTo, request, silent, token) {
|
|
595
|
+
const filename = "voice-message.ogg";
|
|
596
|
+
const fileSize = audioBuffer.byteLength;
|
|
597
|
+
const botToken = token;
|
|
598
|
+
if (!botToken) throw new Error("Discord bot token is required for voice message upload");
|
|
599
|
+
const { upload_filename } = await request(async () => {
|
|
600
|
+
const uploadUrlResponse = await requestVoiceUploadUrl({
|
|
601
|
+
rest,
|
|
602
|
+
channelId,
|
|
603
|
+
botToken,
|
|
604
|
+
filename,
|
|
605
|
+
fileSize
|
|
606
|
+
});
|
|
607
|
+
if (!uploadUrlResponse.attachments?.[0]) throw new Error("Failed to get upload URL for voice message");
|
|
608
|
+
const attachment = uploadUrlResponse.attachments[0];
|
|
609
|
+
await uploadVoiceAttachment({
|
|
610
|
+
uploadUrl: attachment.upload_url,
|
|
611
|
+
audioBuffer
|
|
612
|
+
});
|
|
613
|
+
return attachment;
|
|
614
|
+
}, "voice-upload");
|
|
615
|
+
const messagePayload = {
|
|
616
|
+
flags: silent ? DISCORD_VOICE_MESSAGE_FLAG | SUPPRESS_NOTIFICATIONS_FLAG : DISCORD_VOICE_MESSAGE_FLAG,
|
|
617
|
+
attachments: [{
|
|
618
|
+
id: "0",
|
|
619
|
+
filename,
|
|
620
|
+
uploaded_filename: upload_filename,
|
|
621
|
+
duration_secs: metadata.durationSecs,
|
|
622
|
+
waveform: metadata.waveform
|
|
623
|
+
}]
|
|
624
|
+
};
|
|
625
|
+
if (replyTo) messagePayload.message_reference = {
|
|
626
|
+
message_id: replyTo,
|
|
627
|
+
fail_if_not_exists: false
|
|
628
|
+
};
|
|
629
|
+
return await request(() => rest.post(`/channels/${channelId}/messages`, { body: messagePayload }), "voice-message");
|
|
630
|
+
}
|
|
631
|
+
//#endregion
|
|
632
|
+
//#region extensions/discord/src/send.voice.ts
|
|
633
|
+
function toDiscordSendResult(result, fallbackChannelId) {
|
|
634
|
+
return {
|
|
635
|
+
messageId: result.id || "unknown",
|
|
636
|
+
channelId: result.channel_id ?? fallbackChannelId
|
|
637
|
+
};
|
|
638
|
+
}
|
|
639
|
+
async function materializeVoiceMessageInput(mediaUrl) {
|
|
640
|
+
const media = await loadWebMediaRaw(mediaUrl, maxBytesForKind("audio"));
|
|
641
|
+
const extFromName = media.fileName ? path.extname(media.fileName) : "";
|
|
642
|
+
const extFromMime = media.contentType ? extensionForMime(media.contentType) : "";
|
|
643
|
+
const ext = extFromName || extFromMime || ".bin";
|
|
644
|
+
const tempDir = resolvePreferredOpenClawTmpDir();
|
|
645
|
+
const filePath = path.join(tempDir, `voice-src-${crypto.randomUUID()}${ext}`);
|
|
646
|
+
await fs.writeFile(filePath, media.buffer, { mode: 384 });
|
|
647
|
+
return { filePath };
|
|
648
|
+
}
|
|
649
|
+
/**
|
|
650
|
+
* Send a voice message to Discord.
|
|
651
|
+
*
|
|
652
|
+
* Voice messages are a special Discord feature that displays audio with a waveform
|
|
653
|
+
* visualization. They require OGG/Opus format and cannot include text content.
|
|
654
|
+
*
|
|
655
|
+
* @param to - Recipient (user ID for DM or channel ID)
|
|
656
|
+
* @param audioPath - Path to local audio file (will be converted to OGG/Opus if needed)
|
|
657
|
+
* @param opts - Send options
|
|
658
|
+
*/
|
|
659
|
+
async function sendVoiceMessageDiscord(to, audioPath, opts) {
|
|
660
|
+
const { filePath: localInputPath } = await materializeVoiceMessageInput(audioPath);
|
|
661
|
+
let oggPath = null;
|
|
662
|
+
let oggCleanup = false;
|
|
663
|
+
let token;
|
|
664
|
+
let rest;
|
|
665
|
+
let channelId;
|
|
666
|
+
const cfg = requireRuntimeConfig(opts.cfg, "Discord voice send");
|
|
667
|
+
try {
|
|
668
|
+
const accountInfo = resolveDiscordAccount({
|
|
669
|
+
cfg,
|
|
670
|
+
accountId: opts.accountId
|
|
671
|
+
});
|
|
672
|
+
const client = createDiscordClient({
|
|
673
|
+
...opts,
|
|
674
|
+
cfg
|
|
675
|
+
});
|
|
676
|
+
token = client.token;
|
|
677
|
+
rest = client.rest;
|
|
678
|
+
const request = client.request;
|
|
679
|
+
const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);
|
|
680
|
+
channelId = (await resolveChannelId(rest, recipient, request)).channelId;
|
|
681
|
+
const ogg = await ensureOggOpus(localInputPath);
|
|
682
|
+
oggPath = ogg.path;
|
|
683
|
+
oggCleanup = ogg.cleanup;
|
|
684
|
+
const metadata = await getVoiceMessageMetadata(oggPath);
|
|
685
|
+
const audioBuffer = await fs.readFile(oggPath);
|
|
686
|
+
const result = await sendDiscordVoiceMessage(rest, channelId, audioBuffer, metadata, opts.replyTo, request, opts.silent, token);
|
|
687
|
+
recordChannelActivity({
|
|
688
|
+
channel: "discord",
|
|
689
|
+
accountId: accountInfo.accountId,
|
|
690
|
+
direction: "outbound"
|
|
691
|
+
});
|
|
692
|
+
return toDiscordSendResult(result, channelId);
|
|
693
|
+
} catch (err) {
|
|
694
|
+
if (channelId && rest && token) throw await buildDiscordSendError(err, {
|
|
695
|
+
channelId,
|
|
696
|
+
cfg,
|
|
697
|
+
rest,
|
|
698
|
+
token,
|
|
699
|
+
hasMedia: true
|
|
700
|
+
});
|
|
701
|
+
throw err;
|
|
702
|
+
} finally {
|
|
703
|
+
await unlinkIfExists(oggCleanup ? oggPath : null);
|
|
704
|
+
await unlinkIfExists(localInputPath);
|
|
705
|
+
}
|
|
706
|
+
}
|
|
707
|
+
//#endregion
|
|
708
|
+
//#region extensions/discord/src/send.typing.ts
|
|
709
|
+
async function sendTypingDiscord(channelId, opts) {
|
|
710
|
+
await sendChannelTyping(resolveDiscordRest(opts), channelId);
|
|
711
|
+
return {
|
|
712
|
+
ok: true,
|
|
713
|
+
channelId
|
|
714
|
+
};
|
|
715
|
+
}
|
|
716
|
+
//#endregion
|
|
717
|
+
//#region extensions/discord/src/send.reactions.ts
|
|
718
|
+
function createDiscordReactionRuntimeClient(opts) {
|
|
719
|
+
return createDiscordClient(opts);
|
|
720
|
+
}
|
|
721
|
+
function resolveDiscordReactionClient(opts) {
|
|
722
|
+
if (!opts.cfg) throw new Error("Discord reactions requires a resolved runtime config. Load and resolve config at the command or gateway boundary, then pass cfg through the runtime path.");
|
|
723
|
+
const cfg = requireRuntimeConfig(opts.cfg, "Discord reactions");
|
|
724
|
+
return createDiscordClient({
|
|
725
|
+
...opts,
|
|
726
|
+
cfg
|
|
727
|
+
});
|
|
728
|
+
}
|
|
729
|
+
function isDiscordReactionRuntimeContext(opts) {
|
|
730
|
+
return Boolean(opts.rest && opts.cfg && opts.accountId);
|
|
731
|
+
}
|
|
732
|
+
async function reactMessageDiscord(channelId, messageId, emoji, opts) {
|
|
733
|
+
const { rest, request } = isDiscordReactionRuntimeContext(opts) ? createDiscordReactionRuntimeClient(opts) : resolveDiscordReactionClient(opts);
|
|
734
|
+
const encoded = normalizeReactionEmoji(emoji);
|
|
735
|
+
await request(() => createOwnMessageReaction(rest, channelId, messageId, encoded), "react");
|
|
736
|
+
return { ok: true };
|
|
737
|
+
}
|
|
738
|
+
async function removeReactionDiscord(channelId, messageId, emoji, opts) {
|
|
739
|
+
const { rest } = isDiscordReactionRuntimeContext(opts) ? createDiscordReactionRuntimeClient(opts) : resolveDiscordReactionClient(opts);
|
|
740
|
+
await deleteOwnMessageReaction(rest, channelId, messageId, normalizeReactionEmoji(emoji));
|
|
741
|
+
return { ok: true };
|
|
742
|
+
}
|
|
743
|
+
async function removeOwnReactionsDiscord(channelId, messageId, opts) {
|
|
744
|
+
const { rest } = isDiscordReactionRuntimeContext(opts) ? createDiscordReactionRuntimeClient(opts) : resolveDiscordReactionClient(opts);
|
|
745
|
+
const message = await getChannelMessage(rest, channelId, messageId);
|
|
746
|
+
const identifiers = /* @__PURE__ */ new Set();
|
|
747
|
+
for (const reaction of message.reactions ?? []) {
|
|
748
|
+
const identifier = buildReactionIdentifier(reaction.emoji);
|
|
749
|
+
if (identifier) identifiers.add(identifier);
|
|
750
|
+
}
|
|
751
|
+
if (identifiers.size === 0) return {
|
|
752
|
+
ok: true,
|
|
753
|
+
removed: []
|
|
754
|
+
};
|
|
755
|
+
const removed = [];
|
|
756
|
+
await Promise.allSettled(Array.from(identifiers, (identifier) => {
|
|
757
|
+
removed.push(identifier);
|
|
758
|
+
return deleteOwnMessageReaction(rest, channelId, messageId, normalizeReactionEmoji(identifier));
|
|
759
|
+
}));
|
|
760
|
+
return {
|
|
761
|
+
ok: true,
|
|
762
|
+
removed
|
|
763
|
+
};
|
|
764
|
+
}
|
|
765
|
+
async function fetchReactionsDiscord(channelId, messageId, opts) {
|
|
766
|
+
const { rest } = isDiscordReactionRuntimeContext(opts) ? createDiscordReactionRuntimeClient(opts) : resolveDiscordReactionClient(opts);
|
|
767
|
+
const reactions = (await getChannelMessage(rest, channelId, messageId)).reactions ?? [];
|
|
768
|
+
if (reactions.length === 0) return [];
|
|
769
|
+
const limit = typeof opts.limit === "number" && Number.isFinite(opts.limit) ? Math.min(Math.max(Math.floor(opts.limit), 1), 100) : 100;
|
|
770
|
+
const summaries = [];
|
|
771
|
+
for (const reaction of reactions) {
|
|
772
|
+
const identifier = buildReactionIdentifier(reaction.emoji);
|
|
773
|
+
if (!identifier) continue;
|
|
774
|
+
const users = await listMessageReactionUsers(rest, channelId, messageId, encodeURIComponent(identifier), { limit });
|
|
775
|
+
summaries.push({
|
|
776
|
+
emoji: {
|
|
777
|
+
id: reaction.emoji.id ?? null,
|
|
778
|
+
name: reaction.emoji.name ?? null,
|
|
779
|
+
raw: formatReactionEmoji(reaction.emoji)
|
|
780
|
+
},
|
|
781
|
+
count: reaction.count,
|
|
782
|
+
users: users.map((user) => ({
|
|
783
|
+
id: user.id,
|
|
784
|
+
username: user.username,
|
|
785
|
+
tag: user.username && user.discriminator ? `${user.username}#${user.discriminator}` : user.username
|
|
786
|
+
}))
|
|
787
|
+
});
|
|
788
|
+
}
|
|
789
|
+
return summaries;
|
|
790
|
+
}
|
|
791
|
+
//#endregion
|
|
792
|
+
//#region extensions/discord/src/send.ts
|
|
793
|
+
var send_exports = /* @__PURE__ */ __exportAll({
|
|
794
|
+
DiscordSendError: () => DiscordSendError,
|
|
795
|
+
DiscordThreadInitialMessageError: () => DiscordThreadInitialMessageError,
|
|
796
|
+
addRoleDiscord: () => addRoleDiscord,
|
|
797
|
+
banMemberDiscord: () => banMemberDiscord,
|
|
798
|
+
canViewDiscordGuildChannel: () => canViewDiscordGuildChannel,
|
|
799
|
+
createChannelDiscord: () => createChannelDiscord,
|
|
800
|
+
createScheduledEventDiscord: () => createScheduledEventDiscord,
|
|
801
|
+
createThreadDiscord: () => createThreadDiscord,
|
|
802
|
+
deleteChannelDiscord: () => deleteChannelDiscord,
|
|
803
|
+
deleteMessageDiscord: () => deleteMessageDiscord,
|
|
804
|
+
editChannelDiscord: () => editChannelDiscord,
|
|
805
|
+
editMessageDiscord: () => editMessageDiscord,
|
|
806
|
+
fetchChannelInfoDiscord: () => fetchChannelInfoDiscord,
|
|
807
|
+
fetchChannelPermissionsDiscord: () => fetchChannelPermissionsDiscord,
|
|
808
|
+
fetchMemberGuildPermissionsDiscord: () => fetchMemberGuildPermissionsDiscord,
|
|
809
|
+
fetchMemberInfoDiscord: () => fetchMemberInfoDiscord,
|
|
810
|
+
fetchMessageDiscord: () => fetchMessageDiscord,
|
|
811
|
+
fetchReactionsDiscord: () => fetchReactionsDiscord,
|
|
812
|
+
fetchRoleInfoDiscord: () => fetchRoleInfoDiscord,
|
|
813
|
+
fetchVoiceStatusDiscord: () => fetchVoiceStatusDiscord,
|
|
814
|
+
hasAllGuildPermissionsDiscord: () => hasAllGuildPermissionsDiscord,
|
|
815
|
+
hasAnyGuildPermissionDiscord: () => hasAnyGuildPermissionDiscord,
|
|
816
|
+
kickMemberDiscord: () => kickMemberDiscord,
|
|
817
|
+
listGuildChannelsDiscord: () => listGuildChannelsDiscord,
|
|
818
|
+
listGuildEmojisDiscord: () => listGuildEmojisDiscord,
|
|
819
|
+
listPinsDiscord: () => listPinsDiscord,
|
|
820
|
+
listScheduledEventsDiscord: () => listScheduledEventsDiscord,
|
|
821
|
+
listThreadsDiscord: () => listThreadsDiscord,
|
|
822
|
+
moveChannelDiscord: () => moveChannelDiscord,
|
|
823
|
+
pinMessageDiscord: () => pinMessageDiscord,
|
|
824
|
+
reactMessageDiscord: () => reactMessageDiscord,
|
|
825
|
+
readMessagesDiscord: () => readMessagesDiscord,
|
|
826
|
+
removeChannelPermissionDiscord: () => removeChannelPermissionDiscord,
|
|
827
|
+
removeOwnReactionsDiscord: () => removeOwnReactionsDiscord,
|
|
828
|
+
removeReactionDiscord: () => removeReactionDiscord,
|
|
829
|
+
removeRoleDiscord: () => removeRoleDiscord,
|
|
830
|
+
resolveEventCoverImage: () => resolveEventCoverImage,
|
|
831
|
+
searchMessagesDiscord: () => searchMessagesDiscord,
|
|
832
|
+
sendMessageDiscord: () => sendMessageDiscord,
|
|
833
|
+
sendPollDiscord: () => sendPollDiscord,
|
|
834
|
+
sendStickerDiscord: () => sendStickerDiscord,
|
|
835
|
+
sendTypingDiscord: () => sendTypingDiscord,
|
|
836
|
+
sendVoiceMessageDiscord: () => sendVoiceMessageDiscord,
|
|
837
|
+
sendWebhookMessageDiscord: () => sendWebhookMessageDiscord,
|
|
838
|
+
setChannelPermissionDiscord: () => setChannelPermissionDiscord,
|
|
839
|
+
timeoutMemberDiscord: () => timeoutMemberDiscord,
|
|
840
|
+
unpinMessageDiscord: () => unpinMessageDiscord,
|
|
841
|
+
uploadEmojiDiscord: () => uploadEmojiDiscord,
|
|
842
|
+
uploadStickerDiscord: () => uploadStickerDiscord
|
|
843
|
+
});
|
|
844
|
+
//#endregion
|
|
845
|
+
export { removeRoleDiscord as A, removeChannelPermissionDiscord as B, fetchChannelInfoDiscord as C, kickMemberDiscord as D, fetchVoiceStatusDiscord as E, uploadStickerDiscord as F, createChannelDiscord as I, deleteChannelDiscord as L, timeoutMemberDiscord as M, listGuildEmojisDiscord as N, listGuildChannelsDiscord as O, uploadEmojiDiscord as P, editChannelDiscord as R, createScheduledEventDiscord as S, fetchRoleInfoDiscord as T, setChannelPermissionDiscord as V, readMessagesDiscord as _, removeReactionDiscord as a, addRoleDiscord as b, sendWebhookMessageDiscord as c, deleteMessageDiscord as d, editMessageDiscord as f, pinMessageDiscord as g, listThreadsDiscord as h, removeOwnReactionsDiscord as i, resolveEventCoverImage as j, listScheduledEventsDiscord as k, DiscordThreadInitialMessageError as l, listPinsDiscord as m, fetchReactionsDiscord as n, sendTypingDiscord as o, fetchMessageDiscord as p, reactMessageDiscord as r, sendVoiceMessageDiscord as s, send_exports as t, createThreadDiscord as u, searchMessagesDiscord as v, fetchMemberInfoDiscord as w, banMemberDiscord as x, unpinMessageDiscord as y, moveChannelDiscord as z };
|