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

推荐订阅源

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

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 test(crabbox): tolerate Windows shell capture · openclaw/openclaw@50d6611 fix(qa): extend config mutation Windows budget 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 cleanup Windows budget · openclaw/openclaw@8a93851 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 feat(qa): add coverage scenario matching · openclaw/openclaw@a1fe86a fix(perf): avoid duplicate docker package ui build build: enable modern TypeScript module syntax · openclaw/openclaw@bbc1772 ci: include performance evidence in release validation fix(providers): stream ordinary tool-like prose promptly fix(perf): harden gateway restart bench exits · openclaw/openclaw@82bbcf6 fix(gateway): gate talk secret bootstrap handoff (#85690) · openclaw/openclaw@c791e42 fix: suppress async media incomplete-turn errors (#85933) · openclaw/openclaw@35dcd42 migrate auth credentials · openclaw/openclaw@f036bac fix migrate auth lint · openclaw/openclaw@50e6cb0 fix migrate supported auth imports · openclaw/openclaw@44bb2be fix migrate auth opt-out precedence · openclaw/openclaw@2016a51 honor migrate auth opt-out in plan · openclaw/openclaw@17edec7 address migrate auth review comments · openclaw/openclaw@0a98c2d fix ci blockers for migrate auth docs: add migrate auth changelog (#85667) · openclaw/openclaw@f7fcbdb fix(scripts): avoid duplicate install smoke ui build · openclaw/openclaw@b1b2841 fix(telegram): preserve inbound text entities (#83873) · openclaw/openclaw@b552919 chore: ignore Python bytecode caches · openclaw/openclaw@b6b2755 fix: make autoreview progress visible · openclaw/openclaw@236edb2 ci(release): fix plugin prerelease extension batch invocation test(telegram): provide topic cache store in message context harness · openclaw/openclaw@ff1fde1 test(agents): complete provider runtime test mocks · openclaw/openclaw@be8cd12 test(telegram): type topic cache harness store · openclaw/openclaw@84ab206 test(agents): sync provider runtime mocks · openclaw/openclaw@a289dd9
fix(kilocode): normalize string stop param to array in stream wrapper… · openclaw/openclaw@abe9923
SebTardif · 2026-05-25 · via Recent Commits to openclaw:main

@@ -0,0 +1,146 @@

1+

import type { ProviderWrapStreamFnContext } from "openclaw/plugin-sdk/plugin-entry";

2+

import { resolveProviderRequestHeaders } from "openclaw/plugin-sdk/provider-http";

3+

import { normalizeOptionalLowercaseString } from "openclaw/plugin-sdk/string-coerce-runtime";

4+5+

const KILOCODE_FEATURE_HEADER = "X-KILOCODE-FEATURE";

6+

const KILOCODE_FEATURE_DEFAULT = "openclaw";

7+

const KILOCODE_FEATURE_ENV_VAR = "KILOCODE_FEATURE";

8+9+

type ThinkLevel = NonNullable<ProviderWrapStreamFnContext["thinkingLevel"]>;

10+

type ProviderStreamFn = NonNullable<ProviderWrapStreamFnContext["streamFn"]>;

11+

type ReasoningEffort = "none" | "minimal" | "low" | "medium" | "high" | "xhigh";

12+13+

function resolveKilocodeAppHeaders(): Record<string, string> {

14+

const feature = process.env[KILOCODE_FEATURE_ENV_VAR]?.trim() || KILOCODE_FEATURE_DEFAULT;

15+

return { [KILOCODE_FEATURE_HEADER]: feature };

16+

}

17+18+

function mapThinkingLevelToReasoningEffort(thinkingLevel: ThinkLevel): ReasoningEffort {

19+

if (thinkingLevel === "off") {

20+

return "none";

21+

}

22+

if (thinkingLevel === "adaptive") {

23+

return "medium";

24+

}

25+

if (thinkingLevel === "max") {

26+

return "xhigh";

27+

}

28+

return thinkingLevel;

29+

}

30+31+

function normalizeKilocodeReasoningPayload(

32+

payloadObj: Record<string, unknown>,

33+

thinkingLevel?: ThinkLevel,

34+

): void {

35+

delete payloadObj.reasoning_effort;

36+

if (!thinkingLevel || thinkingLevel === "off") {

37+

return;

38+

}

39+40+

const existingReasoning = payloadObj.reasoning;

41+

if (

42+

existingReasoning &&

43+

typeof existingReasoning === "object" &&

44+

!Array.isArray(existingReasoning)

45+

) {

46+

const reasoningObj = existingReasoning as Record<string, unknown>;

47+

if (!("max_tokens" in reasoningObj) && !("effort" in reasoningObj)) {

48+

reasoningObj.effort = mapThinkingLevelToReasoningEffort(thinkingLevel);

49+

}

50+

} else if (!existingReasoning) {

51+

payloadObj.reasoning = {

52+

effort: mapThinkingLevelToReasoningEffort(thinkingLevel),

53+

};

54+

}

55+

}

56+57+

function normalizeKilocodeStopPayload(payloadObj: Record<string, unknown>): void {

58+

if (typeof payloadObj.stop === "string") {

59+

payloadObj.stop = [payloadObj.stop];

60+

}

61+

}

62+63+

function asRecord(value: unknown): Record<string, unknown> | undefined {

64+

return value && typeof value === "object" && !Array.isArray(value)

65+

? (value as Record<string, unknown>)

66+

: undefined;

67+

}

68+69+

function normalizeKilocodeStopAfterCaller(

70+

value: unknown,

71+

fallbackPayload: Record<string, unknown> | undefined,

72+

): unknown {

73+

const replacementPayload = asRecord(value);

74+

if (replacementPayload) {

75+

normalizeKilocodeStopPayload(replacementPayload);

76+

return value;

77+

}

78+

if (fallbackPayload) {

79+

normalizeKilocodeStopPayload(fallbackPayload);

80+

}

81+

return value;

82+

}

83+84+

function isProxyReasoningUnsupported(modelId: string): boolean {

85+

const trimmed = normalizeOptionalLowercaseString(modelId);

86+

const slashIndex = trimmed?.indexOf("/") ?? -1;

87+

return slashIndex > 0 && trimmed?.slice(0, slashIndex) === "x-ai";

88+

}

89+90+

function resolveKilocodeThinkingLevel(ctx: ProviderWrapStreamFnContext): ThinkLevel | undefined {

91+

if (ctx.modelId === "kilo/auto" || isProxyReasoningUnsupported(ctx.modelId)) {

92+

return undefined;

93+

}

94+

return ctx.thinkingLevel;

95+

}

96+97+

export function createKilocodeStreamWrapper(

98+

baseStreamFn: ProviderWrapStreamFnContext["streamFn"],

99+

thinkingLevel?: ThinkLevel,

100+

): ProviderWrapStreamFnContext["streamFn"] {

101+

if (!baseStreamFn) {

102+

return undefined;

103+

}

104+

const underlying = baseStreamFn;

105+

return (model, context, options) => {

106+

const originalOnPayload = options?.onPayload;

107+

const headers = resolveProviderRequestHeaders({

108+

provider: typeof model.provider === "string" ? model.provider : "kilocode",

109+

api: model.api,

110+

baseUrl: typeof model.baseUrl === "string" ? model.baseUrl : undefined,

111+

capability: "llm",

112+

transport: "stream",

113+

callerHeaders: options?.headers,

114+

defaultHeaders: resolveKilocodeAppHeaders(),

115+

precedence: "defaults-win",

116+

});

117+

return underlying(model, context, {

118+

...options,

119+

headers,

120+

onPayload(payload, payloadModel) {

121+

const payloadObj = asRecord(payload);

122+

if (payloadObj) {

123+

// Keep Kilo thinking defaults overrideable by later caller/config payload hooks.

124+

normalizeKilocodeReasoningPayload(payloadObj, thinkingLevel);

125+

}

126+127+

const result = originalOnPayload?.(payload, payloadModel);

128+

if (result && typeof (result as Promise<unknown>).then === "function") {

129+

return Promise.resolve(result).then((resolved) =>

130+

normalizeKilocodeStopAfterCaller(resolved, payloadObj),

131+

);

132+

}

133+

return normalizeKilocodeStopAfterCaller(result, payloadObj);

134+

},

135+

});

136+

};

137+

}

138+139+

export function wrapKilocodeProviderStream(

140+

ctx: ProviderWrapStreamFnContext,

141+

): ProviderStreamFn | undefined {

142+

if (normalizeOptionalLowercaseString(ctx.provider) !== "kilocode") {

143+

return undefined;

144+

}

145+

return createKilocodeStreamWrapper(ctx.streamFn, resolveKilocodeThinkingLevel(ctx));

146+

}