惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

aimingoo的专栏
aimingoo的专栏
量子位
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
S
Schneier on Security
Cisco Talos Blog
Cisco Talos Blog
T
ThreatConnect
J
Java Code Geeks
博客园 - 司徒正美
A
Arctic Wolf
T
True Tiger Recordings
C
Cybersecurity and Infrastructure Security Agency CISA
Cyberwarzone
Cyberwarzone
Know Your Adversary
Know Your Adversary
T
Threat Research - Cisco Blogs
V
Vulnerabilities – Threatpost
Recorded Future
Recorded Future
P
Palo Alto Networks Blog
The Hacker News
The Hacker News
The Register - Security
The Register - Security
S
Securelist
www.infosecurity-magazine.com
www.infosecurity-magazine.com
C
CXSECURITY Database RSS Feed - CXSecurity.com
Application and Cybersecurity Blog
Application and Cybersecurity Blog
I
Intezer
P
Privacy & Cybersecurity Law Blog
Scott Helme
Scott Helme
K
Kaspersky official blog
博客园 - 聂微东
Last Week in AI
Last Week in AI
V
V2EX
小众软件
小众软件
F
Fox-IT International blog
Martin Fowler
Martin Fowler
Apple Machine Learning Research
Apple Machine Learning Research
T
Tenable Blog
F
Future of Privacy Forum
Microsoft Security Blog
Microsoft Security Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
腾讯CDC
Stack Overflow Blog
Stack Overflow Blog
C
Check Point Blog
阮一峰的网络日志
阮一峰的网络日志
GbyAI
GbyAI
T
Threatpost
I
InfoQ
P
Proofpoint News Feed
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
T
Tor Project blog
G
GRAHAM CLULEY
D
DataBreaches.Net

Recent Commits to openclaw:main

fix(crabbox): bootstrap macos shell js commands · openclaw/openclaw@7cd15d2 fix: preflight malformed openshell exec commands fix: tighten openshell exec preflight docs: clarify unshipped compat policy fix(update): allow package-manager hardlinks in swaps docs: ban repo-hosted proof artifacts fix(installer): handle alpine apk runtime floors fix(security): audit Claude permission overrides under YOLO (#86557) · openclaw/openclaw@bd65b42 fix: speed up Discord voice wake consults · openclaw/openclaw@5ae91f0 fix(qa): harden restart inflight Windows scenario · openclaw/openclaw@3eb06e3 Recover Codex context overflow prompt errors (#85542) · openclaw/openclaw@5cfa577 docs: update changelog for #70473 · openclaw/openclaw@d967760 fix(agents): derive overflow budgets from provider errors · openclaw/openclaw@d5b0174 fix(plugins): only memoize complete metadata snapshots · openclaw/openclaw@3137622 docs: update changelog for media wake fallback (#85489) · openclaw/openclaw@a11d4e6 fix: fallback after active media wake failure (#85489) · openclaw/openclaw@1b64ccb perf(plugins): reuse derived metadata snapshots · openclaw/openclaw@159e440 docs: require maintainer-editable PR branches · openclaw/openclaw@f271f00 fix: scan OpenClaw sessions in agent transcript finder · openclaw/openclaw@4012ae4 docs: note agent transcript OpenClaw session scan · openclaw/openclaw@dd375f9 docs: require generic local fixes · openclaw/openclaw@fc93af5 fix: broaden leading voice wake fuzzing · openclaw/openclaw@a9c91ca test(agents): preserve provider hook mock exports (#86523) · openclaw/openclaw@657b246 Policy: add agent-scoped policy overlays (#85817) · openclaw/openclaw@fbb6340 fix(kilocode): normalize string stop param to array in stream wrapper… · openclaw/openclaw@abe9923 Doctor: expose shell completion health findings (#85566) · openclaw/openclaw@dc17412 fix(agents): honor effective exec policy for Claude live Bash (#86330) fix(test): stabilize e2e runtime imports fix(test): clean plugin gauntlet temp roots · openclaw/openclaw@633e4b8 perf: cache plugin package realpaths (#86517) · openclaw/openclaw@69d728a fix(qa): settle restart races with live budget · openclaw/openclaw@2cac9e5 fix(crabbox): sync full sparse lease runs · openclaw/openclaw@e97e831 fix(qa): extend config mutation Windows budget fix(qa): extend config cleanup Windows budget · openclaw/openclaw@8a93851 test(crabbox): tolerate Windows shell capture · openclaw/openclaw@50d6611 fix(sessions): stop doctor OOM on large session stores and reclaim st… · openclaw/openclaw@89aea9b fix(ollama): strip inline kimi cloud reasoning leak (#86515) fix(discord): merge media captions into one message (#86487) · openclaw/openclaw@bc10fad fix(utils): clamp fetch timeout timers (#85985) fix(ui): preserve user code block rendering (#85942) fix(memory): prevent silent vector index degradation when embedding p… docs: clarify agent transcript placeholders · openclaw/openclaw@8da8bc4 test(qa): annotate live transport RTT measurements · openclaw/openclaw@bb6f37e fix(qqbot): derive outbound watchdog from configured timeouts (#85267… · openclaw/openclaw@aa702cf fix(test): clean kitchen sink rpc temp state · openclaw/openclaw@6f695c1 fix: quiet missing daily memory reads fix: tighten empty plugin registry reuse · openclaw/openclaw@026cfb6 perf: speed up agent transcript lookup · openclaw/openclaw@e7ad116 fix: guard QMD session stem fallback (#86482) · openclaw/openclaw@2e3b59b Guard OpenAI chat payload turns (#86497) · openclaw/openclaw@489e415 fix(gateway): keep session tool mirrors under pressure · openclaw/openclaw@459e89a docs: route github creation through agent transcript test(tools): add unmocked image custom-provider auth regression (#85733) · openclaw/openclaw@f0bfb3f refactor(plugin-sdk): rename plain text tool-call compat wrapper docs(skills): defer private release locators · openclaw/openclaw@23d38e4 Replace Sharp image backend with Photon (#86437) · openclaw/openclaw@b9f975b fix(agents): release embedded-attempt session lock on every exit path… · openclaw/openclaw@32ddfc2 fix: accept OpenClaw voice wake confusions (#86507) fix(crabbox): bootstrap macos js toolchain chore: add agent transcript skill · openclaw/openclaw@d63e8d4 fix(gateway): dedupe session tool fanout · openclaw/openclaw@89a21db fix: Hook ingress token unlocks password-mode gateway auth (#86453) · openclaw/openclaw@d51f268 fix #86077: keep fallback errors candidate scoped (#86134) · openclaw/openclaw@d6b7fe8 fix(diagnostics): reclaim wedged session lanes with a stale leaked ac… · openclaw/openclaw@6f76d9f fix: derive plugin media trust from metadata (#86410) · openclaw/openclaw@e761eb8 fix(media-understanding): normalize HEIC before image descriptions (#… · openclaw/openclaw@75c7236 fix: accept leading fuzzy Discord voice wake names (#86484) · openclaw/openclaw@8fe4f34 feat: promote provider tool call stream wrapper (#86489) fix(test): dedupe kitchen sink command assertions test: derive deprecated sdk usage guard (#86403) fix(qa): extend memory fallback Windows budget fix(ui): move control ui chunking helper out of runtime source · openclaw/openclaw@968c87d fix: quiet retained lost task noise (#86475) fix(build): keep control ui chunking out of deadcode · openclaw/openclaw@dc26069 fix: rotate realtime voice sessions on max duration · openclaw/openclaw@dc2c4aa fix(test): stream bundled plugin sweep logs · openclaw/openclaw@fc3cd49 docs: add bugfix changelog credits · openclaw/openclaw@2e7e4bc fix(models): show oauth marker auth status (#86378) · openclaw/openclaw@a6df39d fix: seed cron task progress summaries (#86313) · openclaw/openclaw@92afd8b fix(update): exclude prerelease tags from stable git channel (#86260) · openclaw/openclaw@28f169b fix(doctor): warn and continue when cron job store is unreadable (#86… fix(gateway): clear runtime config snapshot before in-process restart… · openclaw/openclaw@90caa3b fix(scripts): restore sparse crabbox changed gates · openclaw/openclaw@d270879 fix(build): support Windows UI builds · openclaw/openclaw@0bb9b42 Fix local embedding worker safety (#85348) · openclaw/openclaw@7ff29a9 fix(ui): scope chat session picker to active agent (#85965) · openclaw/openclaw@70c7d6f [codex] improve iOS realtime talk mode (#86355) · openclaw/openclaw@9ca52ce fix(scripts): dedupe docker lane resources · openclaw/openclaw@5e94469 docs: add code size guidance · openclaw/openclaw@9a60fcf fix(test): avoid source gateway import in rpc walk · openclaw/openclaw@e9b8a6e docs: add bugfix changelog entries · openclaw/openclaw@f950132 Fix heartbeat response loop guard (#86324) (#86357) · openclaw/openclaw@e2c174e fix(memory-core): filter REM dreaming candidates to light-staged entr… · openclaw/openclaw@8b42771 fix(telegram): propagate forum topic names into agent context (#86299) fix(slack): keep downloaded files out of reply media (#86318) · openclaw/openclaw@2fcd481 fix(cron): accept plus durations for one-shot jobs (#86341) · openclaw/openclaw@9239f94 fix(plugins): clear metadata memo at lifecycle boundaries · openclaw/openclaw@e7c696a chore(skills): normalize release skill routing · openclaw/openclaw@4737e19 docs(release): require early performance regression check · openclaw/openclaw@0336938 fix(qa): capture Windows gateway metrics · openclaw/openclaw@9afbfc1
fix(discord): restore bare numeric channel sends (#86571) · openclaw/openclaw@c5b9872
joshavant · 2026-05-26 · via Recent Commits to openclaw:main
Original file line numberDiff line numberDiff line change

@@ -60,6 +60,7 @@ Docs: https://docs.openclaw.ai

6060

- xAI/LM Studio: avoid buffering ordinary bracketed or `final` prose until stream completion while watching for plain-text tool-call fallbacks.

6161

- Doctor: warn and continue when the cron job store exists but cannot be read so later health checks still run. Fixes #86102. (#86384) Thanks @1052326311.

6262

- Discord: suppress a bot's previous reply body and referenced media from prompt context when a user replies to that bot message, while keeping reply metadata for routing. (#86238) Thanks @fuller-stack-dev.

63+

- Discord: restore bare numeric channel IDs for outbound message-tool sends while keeping explicit DM targets unambiguous. (#86571) Thanks @joshavant.

6364

- Docker E2E: avoid rebuilding the Control UI twice while preparing the shared OpenClaw package tarball for package-backed scenario runs.

6465

- Tests: avoid rebuilding the Control UI twice during the installer Docker smoke now that `pnpm build` includes `ui:build`.

6566

- Tests: give QA config mutation RPCs enough native Windows budget to finish gateway config writes and restart settle after hot scenario runs.

Original file line numberDiff line numberDiff line change

@@ -39,4 +39,21 @@ describe("resolveDiscordOutboundSessionRoute", () => {

3939

});

4040

expect(route?.threadId).toBeUndefined();

4141

});

42+
43+

it("treats bare numeric outbound targets as channel routes", () => {

44+

const route = resolveDiscordOutboundSessionRoute({

45+

cfg: {},

46+

agentId: "main",

47+

target: "123",

48+

});

49+
50+

expect(route).toMatchObject({

51+

baseSessionKey: "agent:main:discord:channel:123",

52+

chatType: "channel",

53+

from: "discord:channel:123",

54+

peer: { kind: "channel", id: "123" },

55+

sessionKey: "agent:main:discord:channel:123",

56+

to: "channel:123",

57+

});

58+

});

4259

});

Original file line numberDiff line numberDiff line change

@@ -68,5 +68,5 @@ function resolveDiscordOutboundTargetKindHint(params: {

6868

if (/^(user:|discord:|@|<@!?)/i.test(target)) {

6969

return "user";

7070

}

71-

return undefined;

71+

return "channel";

7272

}

Original file line numberDiff line numberDiff line change

@@ -37,3 +37,13 @@ export async function parseAndResolveRecipient(

3737

);

3838

return { kind: resolved.kind, id: resolved.id };

3939

}

40+
41+

export async function parseAndResolveChannelRecipient(

42+

raw: string,

43+

cfg: OpenClawConfig,

44+

accountId?: string,

45+

): Promise<DiscordRecipient> {

46+

return await parseAndResolveRecipient(raw, cfg, accountId, {

47+

defaultKind: "channel",

48+

});

49+

}

Original file line numberDiff line numberDiff line change

@@ -170,6 +170,34 @@ describe("sendDiscordComponentMessage", () => {

170170

);

171171

});

172172
173+

it("treats bare numeric component edit targets as channels", async () => {

174+

const { rest, patchMock, getMock } = makeDiscordRest();

175+

getMock.mockResolvedValueOnce({

176+

type: ChannelType.GuildText,

177+

id: "273512430271856640",

178+

});

179+

patchMock.mockResolvedValueOnce({ id: "msg1", channel_id: "273512430271856640" });

180+
181+

await editDiscordComponentMessage(

182+

"273512430271856640",

183+

"msg1",

184+

{

185+

text: "Updated picker",

186+

blocks: [{ type: "actions", buttons: [{ label: "Tap" }] }],

187+

},

188+

{

189+

cfg: DISCORD_TEST_CFG,

190+

rest,

191+

token: "t",

192+

sessionKey: "agent:main:discord:channel:273512430271856640",

193+

agentId: "main",

194+

},

195+

);

196+
197+

expect(patchMock).toHaveBeenCalledTimes(1);

198+

expect(readMockCall(patchMock, 0)[0]).toContain("/channels/273512430271856640/messages/msg1");

199+

});

200+
173201

it("registers a prebuilt component message against an edited message id", () => {

174202

registerBuiltDiscordComponentMessage({

175203

messageId: "msg1",

@@ -312,6 +340,36 @@ describe("sendDiscordComponentMessage classic message downgrade", () => {

312340

expect(modals[0]?.fields?.[0]?.label).toBe("Notes");

313341

});

314342
343+

it("treats bare numeric component send targets as channels", async () => {

344+

const { rest, postMock, getMock } = makeDiscordRest();

345+

getMock.mockResolvedValueOnce({

346+

type: ChannelType.GuildText,

347+

id: "273512430271856640",

348+

});

349+

postMock.mockResolvedValueOnce({ id: "msg1", channel_id: "273512430271856640" });

350+
351+

await sendDiscordComponentMessage(

352+

"273512430271856640",

353+

{

354+

text: "report",

355+

modal: {

356+

title: "Feedback",

357+

fields: [{ type: "text", label: "Notes" }],

358+

},

359+

},

360+

{

361+

cfg: DISCORD_TEST_CFG,

362+

rest,

363+

token: "t",

364+

mediaUrl: "https://example.com/report.pdf",

365+

},

366+

);

367+
368+

expect(sendMessageDiscordMock).not.toHaveBeenCalled();

369+

expect(postMock).toHaveBeenCalledTimes(1);

370+

expect(readMockCall(postMock, 0)[0]).toContain("/channels/273512430271856640/messages");

371+

});

372+
315373

it("keeps spoiler file blocks on the component path", async () => {

316374

const { rest, postMock, getMock } = makeDiscordRest();

317375

getMock.mockResolvedValueOnce({

Original file line numberDiff line numberDiff line change

@@ -21,7 +21,7 @@ import {

2121

type MessagePayloadObject,

2222

type RequestClient,

2323

} from "./internal/discord.js";

24-

import { parseAndResolveRecipient } from "./recipient-resolution.js";

24+

import { parseAndResolveChannelRecipient } from "./recipient-resolution.js";

2525

import { loadOutboundMediaFromUrl } from "./runtime-api.js";

2626

import { sendMessageDiscord } from "./send.outbound.js";

2727

import { createDiscordSendResult } from "./send.receipt.js";

@@ -290,7 +290,7 @@ export async function sendDiscordComponentMessage(

290290

const cfg = requireRuntimeConfig(opts.cfg, "Discord component send");

291291

const accountInfo = resolveDiscordAccount({ cfg, accountId: opts.accountId });

292292

const { token, rest, request } = createDiscordClient({ ...opts, cfg });

293-

const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);

293+

const recipient = await parseAndResolveChannelRecipient(to, cfg, opts.accountId);

294294

const { channelId } = await resolveChannelId(rest, recipient, request);

295295
296296

const channelType = await resolveDiscordChannelType(rest, channelId);

@@ -353,7 +353,7 @@ export async function editDiscordComponentMessage(

353353

const cfg = requireRuntimeConfig(opts.cfg, "Discord component edit");

354354

const accountInfo = resolveDiscordAccount({ cfg, accountId: opts.accountId });

355355

const { token, rest, request } = createDiscordClient({ ...opts, cfg });

356-

const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);

356+

const recipient = await parseAndResolveChannelRecipient(to, cfg, opts.accountId);

357357

const { channelId } = await resolveChannelId(rest, recipient, request);

358358

const { body, buildResult } = await buildDiscordComponentPayload({

359359

spec,

Original file line numberDiff line numberDiff line change

@@ -11,7 +11,7 @@ import { convertMarkdownTables } from "openclaw/plugin-sdk/text-chunking";

1111

import { resolveDiscordAccount } from "./accounts.js";

1212

import { createChannelMessage, createThread, type RequestClient } from "./internal/discord.js";

1313

import { rewriteDiscordKnownMentions } from "./mentions.js";

14-

import { parseAndResolveRecipient } from "./recipient-resolution.js";

14+

import { parseAndResolveChannelRecipient } from "./recipient-resolution.js";

1515

import { createDiscordSendResult, type DiscordReceiptResultSource } from "./send.receipt.js";

1616

import {

1717

buildDiscordMessageRequest,

@@ -140,7 +140,7 @@ async function resolveDiscordSendTarget(

140140

): Promise<{ rest: RequestClient; request: DiscordClientRequest; channelId: string }> {

141141

const cfg = requireRuntimeConfig(opts.cfg, "Discord send target resolution");

142142

const { rest, request } = createDiscordClient({ ...opts, cfg });

143-

const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);

143+

const recipient = await parseAndResolveChannelRecipient(to, cfg, opts.accountId);

144144

const { channelId } = await resolveChannelId(rest, recipient, request);

145145

return { rest, request, channelId };

146146

}

@@ -181,7 +181,7 @@ export async function sendMessageDiscord(

181181

mentionAliases: accountInfo.config.mentionAliases,

182182

});

183183

const { token, rest, request } = createDiscordClient({ ...opts, cfg });

184-

const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);

184+

const recipient = await parseAndResolveChannelRecipient(to, cfg, opts.accountId);

185185

const { channelId } = await resolveChannelId(rest, recipient, request);

186186
187187

// Forum/Media channels reject POST /messages; auto-create a thread post instead.

Original file line numberDiff line numberDiff line change

@@ -469,29 +469,23 @@ describe("sendMessageDiscord", () => {

469469

expect(res.channelId).toBe("chan1");

470470

});

471471
472-

it("rejects bare numeric IDs as ambiguous", async () => {

473-

const { rest } = makeDiscordRest();

474-

await expect(

475-

sendMessageDiscord("273512430271856640", "hello", {

476-

rest,

477-

token: "t",

478-

cfg: DISCORD_TEST_CFG,

479-

}),

480-

).rejects.toThrow(/Ambiguous Discord recipient/);

481-

await expect(

482-

sendMessageDiscord("273512430271856640", "hello", {

483-

rest,

484-

token: "t",

485-

cfg: DISCORD_TEST_CFG,

486-

}),

487-

).rejects.toThrow(/user:273512430271856640/);

488-

await expect(

489-

sendMessageDiscord("273512430271856640", "hello", {

490-

rest,

491-

token: "t",

492-

cfg: DISCORD_TEST_CFG,

493-

}),

494-

).rejects.toThrow(/channel:273512430271856640/);

472+

it("treats bare numeric outbound IDs as channels", async () => {

473+

const { rest, postMock, getMock } = makeDiscordRest();

474+

getMock.mockResolvedValueOnce({ type: ChannelType.GuildText });

475+

postMock.mockResolvedValueOnce({

476+

id: "msg1",

477+

channel_id: "273512430271856640",

478+

});

479+
480+

const result = await sendMessageDiscord("273512430271856640", "hello", {

481+

rest,

482+

token: "t",

483+

cfg: DISCORD_TEST_CFG,

484+

});

485+
486+

expect(result.channelId).toBe("273512430271856640");

487+

expectRestRoute(postMock, 0, Routes.channelMessages("273512430271856640"));

488+

expect(requireRestBody(postMock).content).toBe("hello");

495489

});

496490
497491

it("adds missing permission hints on 50013", async () => {

Original file line numberDiff line numberDiff line change

@@ -0,0 +1,63 @@

1+

import { beforeAll, beforeEach, describe, expect, it, vi } from "vitest";

2+

import { makeDiscordRest } from "./send.test-harness.js";

3+
4+

const loadWebMediaRawMock = vi.hoisted(() => vi.fn());

5+

vi.mock("openclaw/plugin-sdk/web-media", () => ({

6+

loadWebMediaRaw: loadWebMediaRawMock,

7+

}));

8+
9+

const voiceMocks = vi.hoisted(() => ({

10+

ensureOggOpus: vi.fn(),

11+

getVoiceMessageMetadata: vi.fn(),

12+

sendDiscordVoiceMessage: vi.fn(),

13+

}));

14+

vi.mock("./voice-message.js", () => voiceMocks);

15+
16+

const DISCORD_TEST_CFG = {

17+

channels: { discord: { token: "t" } },

18+

};

19+
20+

let sendVoiceMessageDiscord: typeof import("./send.voice.js").sendVoiceMessageDiscord;

21+
22+

describe("sendVoiceMessageDiscord", () => {

23+

beforeAll(async () => {

24+

({ sendVoiceMessageDiscord } = await import("./send.voice.js"));

25+

});

26+
27+

beforeEach(() => {

28+

vi.clearAllMocks();

29+

loadWebMediaRawMock.mockResolvedValue({

30+

buffer: Buffer.from("voice"),

31+

fileName: "voice.ogg",

32+

contentType: "audio/ogg",

33+

kind: "audio",

34+

});

35+

voiceMocks.ensureOggOpus.mockImplementation(async (inputPath: string) => ({

36+

path: inputPath,

37+

cleanup: false,

38+

}));

39+

voiceMocks.getVoiceMessageMetadata.mockResolvedValue({ duration_secs: 1, waveform: "" });

40+

voiceMocks.sendDiscordVoiceMessage.mockResolvedValue({

41+

id: "msg1",

42+

channel_id: "273512430271856640",

43+

});

44+

});

45+
46+

it("treats bare numeric voice targets as channels", async () => {

47+

const { rest } = makeDiscordRest();

48+
49+

const result = await sendVoiceMessageDiscord(

50+

"273512430271856640",

51+

"https://example.com/voice.ogg",

52+

{

53+

cfg: DISCORD_TEST_CFG,

54+

rest,

55+

token: "t",

56+

},

57+

);

58+
59+

expect(result.channelId).toBe("273512430271856640");

60+

expect(voiceMocks.sendDiscordVoiceMessage).toHaveBeenCalledTimes(1);

61+

expect(voiceMocks.sendDiscordVoiceMessage.mock.calls[0]?.[1]).toBe("273512430271856640");

62+

});

63+

});

Original file line numberDiff line numberDiff line change

@@ -13,7 +13,7 @@ import { tempWorkspace, resolvePreferredOpenClawTmpDir } from "openclaw/plugin-s

1313

import { loadWebMediaRaw } from "openclaw/plugin-sdk/web-media";

1414

import { resolveDiscordAccount } from "./accounts.js";

1515

import type { RequestClient } from "./internal/discord.js";

16-

import { parseAndResolveRecipient } from "./recipient-resolution.js";

16+

import { parseAndResolveChannelRecipient } from "./recipient-resolution.js";

1717

import { createDiscordSendResult } from "./send.receipt.js";

1818

import { buildDiscordSendError, createDiscordClient, resolveChannelId } from "./send.shared.js";

1919

import type { DiscordSendResult } from "./send.types.js";

@@ -95,7 +95,7 @@ export async function sendVoiceMessageDiscord(

9595

token = client.token;

9696

rest = client.rest;

9797

const request = client.request;

98-

const recipient = await parseAndResolveRecipient(to, cfg, opts.accountId);

98+

const recipient = await parseAndResolveChannelRecipient(to, cfg, opts.accountId);

9999

channelId = (await resolveChannelId(rest, recipient, request)).channelId;

100100
101101

const ogg = await ensureOggOpus(localInputPath);