@use-lattice/litmus 0.121.3
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/LICENSE +19 -0
- package/dist/src/accounts-Bt1oJb1Z.cjs +219 -0
- package/dist/src/accounts-DjOU8Rm3.js +178 -0
- package/dist/src/agentic-utils-D03IiXQc.js +153 -0
- package/dist/src/agentic-utils-Dh7xaMQM.cjs +180 -0
- package/dist/src/agents-C6BIMlZa.js +231 -0
- package/dist/src/agents-DvIpNX1L.cjs +666 -0
- package/dist/src/agents-ZP0RP9vV.cjs +231 -0
- package/dist/src/agents-maJXdjbR.js +665 -0
- package/dist/src/aimlapi-BTbQjG2E.cjs +30 -0
- package/dist/src/aimlapi-CwMxqfXP.js +30 -0
- package/dist/src/audio-BBUdvsde.cjs +97 -0
- package/dist/src/audio-D5DPZ7I-.js +97 -0
- package/dist/src/base-BEysXrkq.cjs +222 -0
- package/dist/src/base-C451JQfq.js +193 -0
- package/dist/src/blobs-BY8MDmpo.js +230 -0
- package/dist/src/blobs-BgcNn97m.cjs +256 -0
- package/dist/src/cache-BBE_lsTA.cjs +4 -0
- package/dist/src/cache-BkrqU5Ba.js +237 -0
- package/dist/src/cache-DsCxFlsZ.cjs +297 -0
- package/dist/src/chat-CPJWDP6a.cjs +289 -0
- package/dist/src/chat-CXX3xzkk.cjs +811 -0
- package/dist/src/chat-CcDgZFJ4.js +787 -0
- package/dist/src/chat-Dz5ZeGO2.js +289 -0
- package/dist/src/chatkit-Dw0mKkML.cjs +1158 -0
- package/dist/src/chatkit-swAIVuea.js +1157 -0
- package/dist/src/chunk-DEq-mXcV.js +15 -0
- package/dist/src/claude-agent-sdk-BXZJtOg6.js +379 -0
- package/dist/src/claude-agent-sdk-CkfyjDoG.cjs +383 -0
- package/dist/src/cloudflare-ai-BzpJcqUH.js +161 -0
- package/dist/src/cloudflare-ai-Cmy_R1y2.cjs +161 -0
- package/dist/src/cloudflare-gateway-B9tVQKok.cjs +272 -0
- package/dist/src/cloudflare-gateway-DrD3ew3H.js +272 -0
- package/dist/src/codex-sdk-Dezj9Nwm.js +1056 -0
- package/dist/src/codex-sdk-Dl9D4k5B.cjs +1060 -0
- package/dist/src/cometapi-C-9YvCHC.js +54 -0
- package/dist/src/cometapi-DHgDKoO2.cjs +54 -0
- package/dist/src/completion-B8Ctyxpr.js +120 -0
- package/dist/src/completion-Cxrt08sj.cjs +131 -0
- package/dist/src/createHash-BwgE13yv.cjs +27 -0
- package/dist/src/createHash-DmPQkvBh.js +15 -0
- package/dist/src/docker-BiqcTwLv.js +80 -0
- package/dist/src/docker-C7tEJnP-.cjs +80 -0
- package/dist/src/esm-C62Zofr1.cjs +409 -0
- package/dist/src/esm-DMVc93eh.js +379 -0
- package/dist/src/evalResult-C3NJPQOo.cjs +301 -0
- package/dist/src/evalResult-C7JJAPBb.js +295 -0
- package/dist/src/evalResult-DoVTZZWI.cjs +2 -0
- package/dist/src/extractor-DnMD3fwt.cjs +391 -0
- package/dist/src/extractor-DtlL28vL.js +374 -0
- package/dist/src/fetch-BTxakTSg.cjs +1133 -0
- package/dist/src/fetch-DQckpUFz.js +928 -0
- package/dist/src/fileExtensions-DnqA1y9x.js +85 -0
- package/dist/src/fileExtensions-bYh77CN8.cjs +114 -0
- package/dist/src/genaiTracer-CyZrmaK0.cjs +268 -0
- package/dist/src/genaiTracer-D3fD9dNV.js +256 -0
- package/dist/src/graders-BNscxFrU.js +13644 -0
- package/dist/src/graders-D2oE9Msq.js +2 -0
- package/dist/src/graders-c0Ez_w-9.cjs +2 -0
- package/dist/src/graders-d0F2M3e9.cjs +14056 -0
- package/dist/src/image-0ZhE0VlR.cjs +280 -0
- package/dist/src/image-CWE1pdNv.js +257 -0
- package/dist/src/image-D9ZK6hwL.js +163 -0
- package/dist/src/image-DKZgZITg.cjs +163 -0
- package/dist/src/index.cjs +11366 -0
- package/dist/src/index.d.cts +19640 -0
- package/dist/src/index.d.ts +19641 -0
- package/dist/src/index.js +11306 -0
- package/dist/src/invariant-Ddh24eXh.js +25 -0
- package/dist/src/invariant-kfQ8Bu82.cjs +30 -0
- package/dist/src/knowledgeBase-BgPyGFUd.cjs +122 -0
- package/dist/src/knowledgeBase-DyHilYaP.js +122 -0
- package/dist/src/litellm-CyMeneHS.js +135 -0
- package/dist/src/litellm-DWDF73yF.cjs +135 -0
- package/dist/src/logger-C40ZGil9.js +717 -0
- package/dist/src/logger-DyfK9PBt.cjs +917 -0
- package/dist/src/luma-ray-BAU9X_ep.cjs +315 -0
- package/dist/src/luma-ray-nwVseBbv.js +313 -0
- package/dist/src/messages-B5ADWTTv.js +245 -0
- package/dist/src/messages-BCnZfqrS.cjs +257 -0
- package/dist/src/meteor-DLZZ3osF.cjs +134 -0
- package/dist/src/meteor-DUiCJRC-.js +134 -0
- package/dist/src/modelslab-00cveB8L.cjs +163 -0
- package/dist/src/modelslab-D9sCU_L7.js +163 -0
- package/dist/src/nova-reel-CTapvqYH.js +276 -0
- package/dist/src/nova-reel-DlWuuroF.cjs +278 -0
- package/dist/src/nova-sonic-5UPWfeMv.cjs +363 -0
- package/dist/src/nova-sonic-BhSwQNym.js +363 -0
- package/dist/src/openai-BWrJK9d8.cjs +52 -0
- package/dist/src/openai-DumO8WQn.js +47 -0
- package/dist/src/openclaw-B8brrjC_.cjs +577 -0
- package/dist/src/openclaw-Bkayww9q.js +571 -0
- package/dist/src/opencode-sdk-7xjoDNiM.cjs +562 -0
- package/dist/src/opencode-sdk-SGwAPxht.js +558 -0
- package/dist/src/otlpReceiver-CoAHfAN9.cjs +15 -0
- package/dist/src/otlpReceiver-oO3EQwI9.js +14 -0
- package/dist/src/providerRegistry-4yjhaEM8.js +45 -0
- package/dist/src/providerRegistry-DhV4rJIc.cjs +50 -0
- package/dist/src/providers-B5RJVG-7.cjs +33609 -0
- package/dist/src/providers-BdmZCLzV.js +33262 -0
- package/dist/src/providers-CxtRxn8e.js +2 -0
- package/dist/src/providers-DnQLNbx1.cjs +3 -0
- package/dist/src/pythonUtils-BD0druiM.cjs +275 -0
- package/dist/src/pythonUtils-IBhn5YGR.js +249 -0
- package/dist/src/quiverai-BDOwZBsM.cjs +213 -0
- package/dist/src/quiverai-D3JTF5lD.js +213 -0
- package/dist/src/responses-B2LCDCXZ.js +667 -0
- package/dist/src/responses-BvNm4Xv9.cjs +685 -0
- package/dist/src/rubyUtils-B0NwnfpY.cjs +245 -0
- package/dist/src/rubyUtils-BroxzZ7c.cjs +2 -0
- package/dist/src/rubyUtils-hqVw5UvJ.js +222 -0
- package/dist/src/sagemaker-Cno2V-Sx.js +689 -0
- package/dist/src/sagemaker-fV_KUgs5.cjs +691 -0
- package/dist/src/server-BOuAXb06.cjs +238 -0
- package/dist/src/server-CtI-EWzm.cjs +2 -0
- package/dist/src/server-Cy3DZymt.js +189 -0
- package/dist/src/slack-CP8xBePa.js +135 -0
- package/dist/src/slack-DSQ1yXVb.cjs +135 -0
- package/dist/src/store-BwDDaBjb.cjs +246 -0
- package/dist/src/store-DcbLC593.cjs +2 -0
- package/dist/src/store-IGpqMIkv.js +240 -0
- package/dist/src/tables-3Q2cL7So.cjs +373 -0
- package/dist/src/tables-Bi2fjr4W.js +288 -0
- package/dist/src/telemetry-Bg2WqF79.js +161 -0
- package/dist/src/telemetry-D0x6u5kX.cjs +166 -0
- package/dist/src/telemetry-DXNimrI0.cjs +2 -0
- package/dist/src/text-B_UCRPp2.js +22 -0
- package/dist/src/text-CW1cyrwj.cjs +33 -0
- package/dist/src/tokenUsageUtils-NYT-WKS6.js +138 -0
- package/dist/src/tokenUsageUtils-bVa1ga6f.cjs +173 -0
- package/dist/src/transcription-Cl_W16Pr.js +122 -0
- package/dist/src/transcription-yt1EecY8.cjs +124 -0
- package/dist/src/transform-BCtGrl_W.cjs +228 -0
- package/dist/src/transform-Bv6gG2MJ.cjs +1688 -0
- package/dist/src/transform-CY1wbpRy.js +1507 -0
- package/dist/src/transform-DU8rUL9P.cjs +2 -0
- package/dist/src/transform-yWaShiKr.js +216 -0
- package/dist/src/transformersAvailability-BGkzavwb.js +35 -0
- package/dist/src/transformersAvailability-DKoRtQLy.cjs +35 -0
- package/dist/src/types-5aqHpBwE.cjs +3769 -0
- package/dist/src/types-Bn6D9c4U.js +3300 -0
- package/dist/src/util-BkKlTkI2.js +293 -0
- package/dist/src/util-CTh0bfOm.cjs +1119 -0
- package/dist/src/util-D17oBwo7.cjs +328 -0
- package/dist/src/util-DsS_-v4p.js +613 -0
- package/dist/src/util-DuntT1Ga.js +951 -0
- package/dist/src/util-aWjdCYMI.cjs +667 -0
- package/dist/src/utils-CisQwpjA.js +94 -0
- package/dist/src/utils-yWamDvmz.cjs +123 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/drizzle/0000_lush_hellion.sql +36 -0
- package/drizzle/0001_wide_calypso.sql +3 -0
- package/drizzle/0002_tidy_juggernaut.sql +1 -0
- package/drizzle/0003_lively_naoko.sql +8 -0
- package/drizzle/0004_minor_peter_quill.sql +19 -0
- package/drizzle/0005_silky_millenium_guard.sql +2 -0
- package/drizzle/0006_harsh_caretaker.sql +42 -0
- package/drizzle/0007_cloudy_wong.sql +1 -0
- package/drizzle/0008_broad_boomer.sql +2 -0
- package/drizzle/0009_strong_marten_broadcloak.sql +19 -0
- package/drizzle/0010_needy_bishop.sql +11 -0
- package/drizzle/0011_moaning_millenium_guard.sql +1 -0
- package/drizzle/0012_late_marten_broadcloak.sql +2 -0
- package/drizzle/0013_previous_dormammu.sql +9 -0
- package/drizzle/0014_lazy_captain_universe.sql +2 -0
- package/drizzle/0015_zippy_wallop.sql +29 -0
- package/drizzle/0016_jazzy_zemo.sql +2 -0
- package/drizzle/0017_reflective_praxagora.sql +4 -0
- package/drizzle/0018_fat_vanisher.sql +22 -0
- package/drizzle/0019_new_clint_barton.sql +8 -0
- package/drizzle/0020_skinny_maverick.sql +1 -0
- package/drizzle/0021_mysterious_madelyne_pryor.sql +13 -0
- package/drizzle/0022_sleepy_ultimo.sql +25 -0
- package/drizzle/0023_wooden_mandrill.sql +2 -0
- package/drizzle/AGENTS.md +68 -0
- package/drizzle/CLAUDE.md +1 -0
- package/drizzle/meta/0000_snapshot.json +221 -0
- package/drizzle/meta/0001_snapshot.json +214 -0
- package/drizzle/meta/0002_snapshot.json +221 -0
- package/drizzle/meta/0005_snapshot.json +369 -0
- package/drizzle/meta/0006_snapshot.json +638 -0
- package/drizzle/meta/0007_snapshot.json +640 -0
- package/drizzle/meta/0008_snapshot.json +649 -0
- package/drizzle/meta/0009_snapshot.json +554 -0
- package/drizzle/meta/0010_snapshot.json +619 -0
- package/drizzle/meta/0011_snapshot.json +627 -0
- package/drizzle/meta/0012_snapshot.json +639 -0
- package/drizzle/meta/0013_snapshot.json +717 -0
- package/drizzle/meta/0014_snapshot.json +717 -0
- package/drizzle/meta/0015_snapshot.json +897 -0
- package/drizzle/meta/0016_snapshot.json +1031 -0
- package/drizzle/meta/0018_snapshot.json +1210 -0
- package/drizzle/meta/0019_snapshot.json +1165 -0
- package/drizzle/meta/0020_snapshot.json +1232 -0
- package/drizzle/meta/0021_snapshot.json +1311 -0
- package/drizzle/meta/0022_snapshot.json +1481 -0
- package/drizzle/meta/0023_snapshot.json +1496 -0
- package/drizzle/meta/_journal.json +174 -0
- package/package.json +240 -0
|
@@ -0,0 +1,577 @@
|
|
|
1
|
+
const require_logger = require("./logger-DyfK9PBt.cjs");
|
|
2
|
+
const require_fetch = require("./fetch-BTxakTSg.cjs");
|
|
3
|
+
const require_chat = require("./chat-CXX3xzkk.cjs");
|
|
4
|
+
const require_responses = require("./responses-BvNm4Xv9.cjs");
|
|
5
|
+
let fs = require("fs");
|
|
6
|
+
fs = require_logger.__toESM(fs);
|
|
7
|
+
let path = require("path");
|
|
8
|
+
path = require_logger.__toESM(path);
|
|
9
|
+
let os = require("os");
|
|
10
|
+
os = require_logger.__toESM(os);
|
|
11
|
+
let crypto = require("crypto");
|
|
12
|
+
crypto = require_logger.__toESM(crypto);
|
|
13
|
+
let ws = require("ws");
|
|
14
|
+
ws = require_logger.__toESM(ws);
|
|
15
|
+
let json5 = require("json5");
|
|
16
|
+
json5 = require_logger.__toESM(json5);
|
|
17
|
+
//#region src/providers/openclaw/shared.ts
|
|
18
|
+
const DEFAULT_GATEWAY_PORT = 18789;
|
|
19
|
+
const DEFAULT_GATEWAY_HOST = "127.0.0.1";
|
|
20
|
+
/**
|
|
21
|
+
* Cached config to avoid re-reading the file multiple times during provider init.
|
|
22
|
+
*/
|
|
23
|
+
let cachedConfig;
|
|
24
|
+
let cachedConfigPath;
|
|
25
|
+
function resolveConfigPath(env) {
|
|
26
|
+
return env?.OPENCLAW_CONFIG_PATH || require_logger.getEnvString("OPENCLAW_CONFIG_PATH") || path.default.join(os.default.homedir(), ".openclaw", "openclaw.json");
|
|
27
|
+
}
|
|
28
|
+
function normalizeGatewayUrl(url, transport) {
|
|
29
|
+
const trimmed = url.trim();
|
|
30
|
+
if (!trimmed) return;
|
|
31
|
+
if (transport === "http") {
|
|
32
|
+
if (trimmed.startsWith("wss://")) return `https://${trimmed.slice(6)}`;
|
|
33
|
+
if (trimmed.startsWith("ws://")) return `http://${trimmed.slice(5)}`;
|
|
34
|
+
return trimmed;
|
|
35
|
+
}
|
|
36
|
+
if (trimmed.startsWith("https://")) return `wss://${trimmed.slice(8)}`;
|
|
37
|
+
if (trimmed.startsWith("http://")) return `ws://${trimmed.slice(7)}`;
|
|
38
|
+
return trimmed;
|
|
39
|
+
}
|
|
40
|
+
function resolveGatewayHost(gatewayConfig) {
|
|
41
|
+
const bind = gatewayConfig?.bind?.trim();
|
|
42
|
+
const customBindHost = gatewayConfig?.customBindHost?.trim();
|
|
43
|
+
if (!bind || bind === "auto" || bind === "loopback" || bind === "lan" || bind === "tailnet" || bind === "0.0.0.0" || bind === "::" || bind === "127.0.0.1" || bind === "localhost" || bind === "::1") return DEFAULT_GATEWAY_HOST;
|
|
44
|
+
if (bind === "custom") {
|
|
45
|
+
if (!customBindHost || customBindHost === "0.0.0.0" || customBindHost === "::" || customBindHost === "127.0.0.1" || customBindHost === "localhost" || customBindHost === "::1") return DEFAULT_GATEWAY_HOST;
|
|
46
|
+
return customBindHost;
|
|
47
|
+
}
|
|
48
|
+
return bind;
|
|
49
|
+
}
|
|
50
|
+
function buildLocalGatewayUrl(gatewayConfig, transport) {
|
|
51
|
+
const scheme = transport === "ws" ? gatewayConfig?.tls?.enabled ? "wss" : "ws" : gatewayConfig?.tls?.enabled ? "https" : "http";
|
|
52
|
+
const port = gatewayConfig?.port ?? 18789;
|
|
53
|
+
return `${scheme}://${resolveGatewayHost(gatewayConfig)}:${port}`;
|
|
54
|
+
}
|
|
55
|
+
function resolveGatewayUrlFromConfig(openclawConfig, transport) {
|
|
56
|
+
const gatewayConfig = openclawConfig?.gateway;
|
|
57
|
+
if (!gatewayConfig) return;
|
|
58
|
+
if (gatewayConfig.mode === "remote") {
|
|
59
|
+
const remoteUrl = normalizeGatewayUrl(gatewayConfig.remote?.url ?? "", transport);
|
|
60
|
+
if (remoteUrl) return remoteUrl;
|
|
61
|
+
}
|
|
62
|
+
return buildLocalGatewayUrl(gatewayConfig, transport);
|
|
63
|
+
}
|
|
64
|
+
function toAuthSecret(kind, value) {
|
|
65
|
+
const trimmed = value?.trim();
|
|
66
|
+
return trimmed ? {
|
|
67
|
+
kind,
|
|
68
|
+
value: trimmed
|
|
69
|
+
} : void 0;
|
|
70
|
+
}
|
|
71
|
+
function resolveAuthSecretFromConfig(openclawConfig) {
|
|
72
|
+
const gatewayConfig = openclawConfig?.gateway;
|
|
73
|
+
const authMode = gatewayConfig?.auth?.mode?.trim();
|
|
74
|
+
const preferRemoteCredentials = gatewayConfig?.mode === "remote";
|
|
75
|
+
const localToken = toAuthSecret("token", gatewayConfig?.auth?.token);
|
|
76
|
+
const localPassword = toAuthSecret("password", gatewayConfig?.auth?.password);
|
|
77
|
+
const remoteToken = toAuthSecret("token", gatewayConfig?.remote?.token);
|
|
78
|
+
const remotePassword = toAuthSecret("password", gatewayConfig?.remote?.password);
|
|
79
|
+
if (preferRemoteCredentials) {
|
|
80
|
+
if (authMode === "password") return remotePassword || localPassword || remoteToken || localToken;
|
|
81
|
+
if (authMode === "token") return remoteToken || localToken || remotePassword || localPassword;
|
|
82
|
+
return remoteToken || remotePassword || localToken || localPassword;
|
|
83
|
+
}
|
|
84
|
+
if (authMode === "password") return localPassword || remotePassword || localToken || remoteToken;
|
|
85
|
+
if (authMode === "token") return localToken || remoteToken || localPassword || remotePassword;
|
|
86
|
+
return localToken || localPassword || remoteToken || remotePassword;
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Read and parse the active OpenClaw configuration file.
|
|
90
|
+
* Results are cached based on file modification time.
|
|
91
|
+
* Returns undefined if the file doesn't exist or can't be parsed.
|
|
92
|
+
*/
|
|
93
|
+
function readOpenClawConfig(env) {
|
|
94
|
+
const configPath = resolveConfigPath(env);
|
|
95
|
+
try {
|
|
96
|
+
if (!fs.default.existsSync(configPath)) return;
|
|
97
|
+
const mtime = fs.default.statSync(configPath).mtimeMs;
|
|
98
|
+
if (cachedConfig && cachedConfigPath === configPath && cachedConfig.mtime === mtime) return cachedConfig.config;
|
|
99
|
+
const raw = fs.default.readFileSync(configPath, "utf-8");
|
|
100
|
+
const config = json5.default.parse(raw);
|
|
101
|
+
cachedConfig = {
|
|
102
|
+
config,
|
|
103
|
+
mtime
|
|
104
|
+
};
|
|
105
|
+
cachedConfigPath = configPath;
|
|
106
|
+
return config;
|
|
107
|
+
} catch (err) {
|
|
108
|
+
require_logger.logger.debug(`Failed to read OpenClaw config at ${configPath}: ${err}`);
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Auto-detect the OpenClaw gateway URL from config, env overrides, or the active config file.
|
|
114
|
+
*/
|
|
115
|
+
function resolveGatewayUrl(config, env) {
|
|
116
|
+
return resolveGatewayTransportUrl(config, env, "http");
|
|
117
|
+
}
|
|
118
|
+
function resolveGatewayWsUrl(config, env) {
|
|
119
|
+
return resolveGatewayTransportUrl(config, env, "ws");
|
|
120
|
+
}
|
|
121
|
+
function resolveGatewayTransportUrl(config, env, transport) {
|
|
122
|
+
const configUrl = config?.gateway_url?.trim();
|
|
123
|
+
if (configUrl) return normalizeGatewayUrl(configUrl, transport) || configUrl;
|
|
124
|
+
const trimmedEnvUrl = (env?.OPENCLAW_GATEWAY_URL || require_logger.getEnvString("OPENCLAW_GATEWAY_URL") || env?.CLAWDBOT_GATEWAY_URL || require_logger.getEnvString("CLAWDBOT_GATEWAY_URL"))?.trim();
|
|
125
|
+
if (trimmedEnvUrl) return normalizeGatewayUrl(trimmedEnvUrl, transport) || trimmedEnvUrl;
|
|
126
|
+
const resolvedUrl = resolveGatewayUrlFromConfig(readOpenClawConfig(env), transport);
|
|
127
|
+
if (resolvedUrl) return resolvedUrl;
|
|
128
|
+
return `${transport === "ws" ? "ws" : "http"}://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}`;
|
|
129
|
+
}
|
|
130
|
+
function resolveAuthSecret(config, env) {
|
|
131
|
+
if (config?.auth_token) return {
|
|
132
|
+
kind: "token",
|
|
133
|
+
value: config.auth_token
|
|
134
|
+
};
|
|
135
|
+
if (config?.auth_password) return {
|
|
136
|
+
kind: "password",
|
|
137
|
+
value: config.auth_password
|
|
138
|
+
};
|
|
139
|
+
const envToken = env?.OPENCLAW_GATEWAY_TOKEN || require_logger.getEnvString("OPENCLAW_GATEWAY_TOKEN") || env?.CLAWDBOT_GATEWAY_TOKEN || require_logger.getEnvString("CLAWDBOT_GATEWAY_TOKEN");
|
|
140
|
+
if (envToken) return {
|
|
141
|
+
kind: "token",
|
|
142
|
+
value: envToken
|
|
143
|
+
};
|
|
144
|
+
const envPassword = env?.OPENCLAW_GATEWAY_PASSWORD || require_logger.getEnvString("OPENCLAW_GATEWAY_PASSWORD") || env?.CLAWDBOT_GATEWAY_PASSWORD || require_logger.getEnvString("CLAWDBOT_GATEWAY_PASSWORD");
|
|
145
|
+
if (envPassword) return {
|
|
146
|
+
kind: "password",
|
|
147
|
+
value: envPassword
|
|
148
|
+
};
|
|
149
|
+
return resolveAuthSecretFromConfig(readOpenClawConfig(env));
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Auto-detect the OpenClaw gateway bearer secret from config, env overrides, or the active
|
|
153
|
+
* config file. OpenClaw accepts either a token or password as the HTTP bearer secret.
|
|
154
|
+
*/
|
|
155
|
+
function resolveAuthToken(config, env) {
|
|
156
|
+
return resolveAuthSecret(config, env)?.value;
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Build common OpenClaw headers for agent-id and session-key.
|
|
160
|
+
* Note: thinking_level is only supported by the WS Agent provider and is
|
|
161
|
+
* passed as an RPC param there, not as an HTTP header.
|
|
162
|
+
*/
|
|
163
|
+
function buildOpenClawHeaders(agentId, config) {
|
|
164
|
+
const headers = { "x-openclaw-agent-id": agentId };
|
|
165
|
+
if (config?.session_key) headers["x-openclaw-session-key"] = config.session_key;
|
|
166
|
+
return headers;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Build provider options for OpenAI-compatible OpenClaw providers (chat, responses).
|
|
170
|
+
* Resolves gateway URL, auth token, and merges OpenClaw-specific headers.
|
|
171
|
+
*/
|
|
172
|
+
function buildOpenClawProviderOptions(agentId, providerOptions) {
|
|
173
|
+
const config = providerOptions.config || {};
|
|
174
|
+
const env = providerOptions.env;
|
|
175
|
+
const gatewayUrl = resolveGatewayUrl(config, env);
|
|
176
|
+
const authToken = resolveAuthToken(config, env);
|
|
177
|
+
return {
|
|
178
|
+
...providerOptions,
|
|
179
|
+
config: {
|
|
180
|
+
...config,
|
|
181
|
+
apiBaseUrl: `${gatewayUrl}/v1`,
|
|
182
|
+
...authToken && { apiKey: authToken },
|
|
183
|
+
apiKeyRequired: false,
|
|
184
|
+
headers: {
|
|
185
|
+
...config.headers,
|
|
186
|
+
...buildOpenClawHeaders(agentId, config)
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
//#endregion
|
|
192
|
+
//#region src/providers/openclaw/agent.ts
|
|
193
|
+
const OPENCLAW_PROTOCOL_VERSION = 3;
|
|
194
|
+
/**
|
|
195
|
+
* OpenClaw WebSocket Agent Provider
|
|
196
|
+
*
|
|
197
|
+
* Custom provider that uses the native OpenClaw WS RPC protocol to invoke agents.
|
|
198
|
+
* Supports full streaming with event accumulation.
|
|
199
|
+
*
|
|
200
|
+
* Protocol flow:
|
|
201
|
+
* 1. Open WS connection to gateway
|
|
202
|
+
* 2. Receive connect.challenge event → send connect request
|
|
203
|
+
* 3. Receive hello-ok response → send agent request
|
|
204
|
+
* 4. Receive agent accepted response → send agent.wait
|
|
205
|
+
* 5. Accumulate streaming "agent" events (stream: "assistant")
|
|
206
|
+
* 6. Resolve on agent.wait response
|
|
207
|
+
*
|
|
208
|
+
* Usage:
|
|
209
|
+
* openclaw:agent - default agent (main)
|
|
210
|
+
* openclaw:agent:main - explicit agent ID
|
|
211
|
+
* openclaw:agent:my-agent - custom agent ID
|
|
212
|
+
*/
|
|
213
|
+
var OpenClawAgentProvider = class {
|
|
214
|
+
agentId;
|
|
215
|
+
gatewayUrl;
|
|
216
|
+
authKind;
|
|
217
|
+
authSecret;
|
|
218
|
+
openclawConfig;
|
|
219
|
+
timeoutMs;
|
|
220
|
+
activeConnections = /* @__PURE__ */ new Set();
|
|
221
|
+
constructor(agentId, providerOptions = {}) {
|
|
222
|
+
this.agentId = agentId;
|
|
223
|
+
this.openclawConfig = providerOptions.config || {};
|
|
224
|
+
const env = providerOptions.env;
|
|
225
|
+
this.gatewayUrl = resolveGatewayWsUrl(this.openclawConfig, env);
|
|
226
|
+
const authSecret = resolveAuthSecret(this.openclawConfig, env);
|
|
227
|
+
this.authKind = authSecret?.kind;
|
|
228
|
+
this.authSecret = authSecret?.value;
|
|
229
|
+
this.timeoutMs = this.openclawConfig.timeoutMs ?? require_fetch.REQUEST_TIMEOUT_MS;
|
|
230
|
+
}
|
|
231
|
+
id() {
|
|
232
|
+
return `openclaw:agent:${this.agentId}`;
|
|
233
|
+
}
|
|
234
|
+
toString() {
|
|
235
|
+
return `[OpenClaw Agent Provider ${this.agentId}]`;
|
|
236
|
+
}
|
|
237
|
+
toJSON() {
|
|
238
|
+
return { provider: this.id() };
|
|
239
|
+
}
|
|
240
|
+
async cleanup() {
|
|
241
|
+
for (const ws$1 of this.activeConnections) ws$1.close();
|
|
242
|
+
this.activeConnections.clear();
|
|
243
|
+
}
|
|
244
|
+
async callApi(prompt) {
|
|
245
|
+
const sessionKey = this.openclawConfig.session_key || `promptfoo-${crypto.default.randomUUID()}`;
|
|
246
|
+
return new Promise((resolve) => {
|
|
247
|
+
const ws$2 = new ws.default(this.gatewayUrl);
|
|
248
|
+
this.activeConnections.add(ws$2);
|
|
249
|
+
const agentRequestId = crypto.default.randomUUID();
|
|
250
|
+
const waitRequestId = crypto.default.randomUUID();
|
|
251
|
+
const idempotencyKey = crypto.default.randomUUID();
|
|
252
|
+
let lastText = "";
|
|
253
|
+
let runId;
|
|
254
|
+
let connected = false;
|
|
255
|
+
let resolved = false;
|
|
256
|
+
const finish = (result, closeSocket = true) => {
|
|
257
|
+
if (resolved) return;
|
|
258
|
+
resolved = true;
|
|
259
|
+
clearTimeout(timeout);
|
|
260
|
+
this.activeConnections.delete(ws$2);
|
|
261
|
+
if (closeSocket) ws$2.close();
|
|
262
|
+
resolve(result);
|
|
263
|
+
};
|
|
264
|
+
const timeout = setTimeout(() => {
|
|
265
|
+
finish({ error: `OpenClaw agent request timed out after ${this.timeoutMs}ms` });
|
|
266
|
+
}, this.timeoutMs);
|
|
267
|
+
ws$2.on("error", (err) => {
|
|
268
|
+
finish({ error: `OpenClaw WebSocket error: ${err.message}` });
|
|
269
|
+
});
|
|
270
|
+
ws$2.on("close", () => {
|
|
271
|
+
this.activeConnections.delete(ws$2);
|
|
272
|
+
if (!resolved) finish({ error: "OpenClaw WebSocket connection closed unexpectedly" }, false);
|
|
273
|
+
});
|
|
274
|
+
ws$2.on("message", (data) => {
|
|
275
|
+
let frame;
|
|
276
|
+
try {
|
|
277
|
+
frame = JSON.parse(data.toString());
|
|
278
|
+
} catch {
|
|
279
|
+
require_logger.logger.debug("[OpenClaw Agent] Failed to parse WS frame");
|
|
280
|
+
return;
|
|
281
|
+
}
|
|
282
|
+
require_logger.logger.debug("[OpenClaw Agent] Frame received", {
|
|
283
|
+
type: frame.type,
|
|
284
|
+
event: frame.event,
|
|
285
|
+
id: frame.id
|
|
286
|
+
});
|
|
287
|
+
if (frame.type === "event" && frame.event === "connect.challenge") {
|
|
288
|
+
ws$2.send(JSON.stringify({
|
|
289
|
+
type: "req",
|
|
290
|
+
id: crypto.default.randomUUID(),
|
|
291
|
+
method: "connect",
|
|
292
|
+
params: {
|
|
293
|
+
minProtocol: OPENCLAW_PROTOCOL_VERSION,
|
|
294
|
+
maxProtocol: OPENCLAW_PROTOCOL_VERSION,
|
|
295
|
+
client: {
|
|
296
|
+
id: "gateway-client",
|
|
297
|
+
displayName: "promptfoo",
|
|
298
|
+
version: require_fetch.VERSION,
|
|
299
|
+
platform: process.platform,
|
|
300
|
+
mode: "cli"
|
|
301
|
+
},
|
|
302
|
+
role: "operator",
|
|
303
|
+
scopes: ["operator.read", "operator.write"],
|
|
304
|
+
caps: [],
|
|
305
|
+
commands: [],
|
|
306
|
+
permissions: {},
|
|
307
|
+
...this.authSecret && this.authKind && { auth: this.authKind === "password" ? { password: this.authSecret } : { token: this.authSecret } }
|
|
308
|
+
}
|
|
309
|
+
}));
|
|
310
|
+
return;
|
|
311
|
+
}
|
|
312
|
+
if (frame.type === "res" && !connected) {
|
|
313
|
+
if (!frame.ok) {
|
|
314
|
+
finish({ error: `OpenClaw connect failed: ${frame.error?.message || "unknown error"}` });
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
connected = true;
|
|
318
|
+
ws$2.send(JSON.stringify({
|
|
319
|
+
type: "req",
|
|
320
|
+
id: agentRequestId,
|
|
321
|
+
method: "agent",
|
|
322
|
+
params: {
|
|
323
|
+
message: prompt,
|
|
324
|
+
agentId: this.agentId,
|
|
325
|
+
idempotencyKey,
|
|
326
|
+
sessionKey,
|
|
327
|
+
...this.openclawConfig.thinking_level && { thinking: this.openclawConfig.thinking_level },
|
|
328
|
+
...this.openclawConfig.extra_system_prompt && { extraSystemPrompt: this.openclawConfig.extra_system_prompt }
|
|
329
|
+
}
|
|
330
|
+
}));
|
|
331
|
+
return;
|
|
332
|
+
}
|
|
333
|
+
if (frame.type === "res" && frame.id === agentRequestId) {
|
|
334
|
+
if (!frame.ok) {
|
|
335
|
+
finish({ error: `OpenClaw agent error: ${frame.error?.message || "unknown error"}` });
|
|
336
|
+
return;
|
|
337
|
+
}
|
|
338
|
+
const payload = frame.payload;
|
|
339
|
+
runId = typeof payload?.runId === "string" && payload.runId.trim() ? payload.runId : void 0;
|
|
340
|
+
if (!runId) {
|
|
341
|
+
require_logger.logger.warn("[OpenClaw Agent] Missing runId in accepted response", {
|
|
342
|
+
agentId: this.agentId,
|
|
343
|
+
payload
|
|
344
|
+
});
|
|
345
|
+
finish({ error: "OpenClaw agent error: gateway accepted request without a runId" });
|
|
346
|
+
return;
|
|
347
|
+
}
|
|
348
|
+
ws$2.send(JSON.stringify({
|
|
349
|
+
type: "req",
|
|
350
|
+
id: waitRequestId,
|
|
351
|
+
method: "agent.wait",
|
|
352
|
+
params: {
|
|
353
|
+
runId,
|
|
354
|
+
timeoutMs: this.timeoutMs
|
|
355
|
+
}
|
|
356
|
+
}));
|
|
357
|
+
return;
|
|
358
|
+
}
|
|
359
|
+
if (frame.type === "event" && frame.event === "agent") {
|
|
360
|
+
const payload = frame.payload;
|
|
361
|
+
if (payload?.runId && runId && payload.runId !== runId) return;
|
|
362
|
+
if (payload?.stream === "assistant" && payload?.data?.text) lastText = payload.data.text;
|
|
363
|
+
return;
|
|
364
|
+
}
|
|
365
|
+
if (frame.type === "res" && frame.id === waitRequestId) {
|
|
366
|
+
if (frame.ok) finish({ output: lastText || "No output from agent" });
|
|
367
|
+
else finish({ error: `OpenClaw agent error: ${frame.error?.message || "unknown error"}` });
|
|
368
|
+
return;
|
|
369
|
+
}
|
|
370
|
+
});
|
|
371
|
+
});
|
|
372
|
+
}
|
|
373
|
+
};
|
|
374
|
+
//#endregion
|
|
375
|
+
//#region src/providers/openclaw/chat.ts
|
|
376
|
+
/**
|
|
377
|
+
* OpenClaw chat provider extends OpenAI chat completion provider.
|
|
378
|
+
*
|
|
379
|
+
* OpenClaw exposes an OpenAI-compatible HTTP API at /v1/chat/completions.
|
|
380
|
+
* This provider auto-detects gateway URL and bearer auth from:
|
|
381
|
+
* 1. Explicit config (gateway_url, auth_token, auth_password)
|
|
382
|
+
* 2. Environment variables (OPENCLAW_GATEWAY_URL, OPENCLAW_GATEWAY_TOKEN, OPENCLAW_GATEWAY_PASSWORD)
|
|
383
|
+
* 3. The active OpenClaw config file (OPENCLAW_CONFIG_PATH or ~/.openclaw/openclaw.json),
|
|
384
|
+
* including gateway.remote.url and gateway.tls.enabled
|
|
385
|
+
*
|
|
386
|
+
* Usage:
|
|
387
|
+
* openclaw - default agent (main)
|
|
388
|
+
* openclaw:main - specific agent
|
|
389
|
+
* openclaw:coding-agent - named agent
|
|
390
|
+
*/
|
|
391
|
+
var OpenClawChatProvider = class extends require_chat.OpenAiChatCompletionProvider {
|
|
392
|
+
agentId;
|
|
393
|
+
constructor(agentId, providerOptions = {}) {
|
|
394
|
+
super(`openclaw:${agentId}`, buildOpenClawProviderOptions(agentId, providerOptions));
|
|
395
|
+
this.agentId = agentId;
|
|
396
|
+
}
|
|
397
|
+
id() {
|
|
398
|
+
return `openclaw:${this.agentId}`;
|
|
399
|
+
}
|
|
400
|
+
toString() {
|
|
401
|
+
return `[OpenClaw Provider ${this.agentId}]`;
|
|
402
|
+
}
|
|
403
|
+
getApiUrlDefault() {
|
|
404
|
+
return `http://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}/v1`;
|
|
405
|
+
}
|
|
406
|
+
getApiKey() {
|
|
407
|
+
return this.config.apiKey;
|
|
408
|
+
}
|
|
409
|
+
getApiUrl() {
|
|
410
|
+
return this.config.apiBaseUrl || this.getApiUrlDefault();
|
|
411
|
+
}
|
|
412
|
+
};
|
|
413
|
+
//#endregion
|
|
414
|
+
//#region src/providers/openclaw/responses.ts
|
|
415
|
+
/**
|
|
416
|
+
* OpenClaw Responses API Provider
|
|
417
|
+
*
|
|
418
|
+
* Extends OpenAI Responses API provider with OpenClaw-specific configuration.
|
|
419
|
+
* Routes through the OpenClaw gateway's /v1/responses endpoint.
|
|
420
|
+
*
|
|
421
|
+
* Requires `gateway.http.endpoints.responses.enabled=true` in OpenClaw config.
|
|
422
|
+
*
|
|
423
|
+
* Usage:
|
|
424
|
+
* openclaw:responses - default agent (main)
|
|
425
|
+
* openclaw:responses:main - explicit agent ID
|
|
426
|
+
* openclaw:responses:X - custom agent ID
|
|
427
|
+
*/
|
|
428
|
+
var OpenClawResponsesProvider = class extends require_responses.OpenAiResponsesProvider {
|
|
429
|
+
agentId;
|
|
430
|
+
constructor(agentId, providerOptions = {}) {
|
|
431
|
+
super(`openclaw:${agentId}`, buildOpenClawProviderOptions(agentId, providerOptions));
|
|
432
|
+
this.agentId = agentId;
|
|
433
|
+
}
|
|
434
|
+
id() {
|
|
435
|
+
return `openclaw:responses:${this.agentId}`;
|
|
436
|
+
}
|
|
437
|
+
toString() {
|
|
438
|
+
return `[OpenClaw Responses Provider ${this.agentId}]`;
|
|
439
|
+
}
|
|
440
|
+
getApiUrlDefault() {
|
|
441
|
+
return `http://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}/v1`;
|
|
442
|
+
}
|
|
443
|
+
getApiKey() {
|
|
444
|
+
return this.config.apiKey;
|
|
445
|
+
}
|
|
446
|
+
getApiUrl() {
|
|
447
|
+
return this.config.apiBaseUrl || this.getApiUrlDefault();
|
|
448
|
+
}
|
|
449
|
+
async getOpenAiBody(prompt, context, callApiOptions) {
|
|
450
|
+
const result = await super.getOpenAiBody(prompt, context, callApiOptions);
|
|
451
|
+
if ("text" in result.body) delete result.body.text;
|
|
452
|
+
return result;
|
|
453
|
+
}
|
|
454
|
+
};
|
|
455
|
+
//#endregion
|
|
456
|
+
//#region src/providers/openclaw/tools.ts
|
|
457
|
+
/**
|
|
458
|
+
* OpenClaw Tool Invoke Provider
|
|
459
|
+
*
|
|
460
|
+
* Simple HTTP provider for direct tool invocation via POST /tools/invoke.
|
|
461
|
+
* The tool name is extracted from the provider path:
|
|
462
|
+
* openclaw:tools:sessions_list → tool="sessions_list"
|
|
463
|
+
*
|
|
464
|
+
* The prompt is parsed as JSON for tool arguments. If it's not valid JSON,
|
|
465
|
+
* it's passed as a single `input` argument.
|
|
466
|
+
*
|
|
467
|
+
* Usage:
|
|
468
|
+
* openclaw:tools:sessions_list - invoke the sessions_list tool
|
|
469
|
+
* openclaw:tools:session_status - invoke the session_status tool
|
|
470
|
+
*
|
|
471
|
+
* Optional config:
|
|
472
|
+
* action - tool sub-action, forwarded as body.action
|
|
473
|
+
* dry_run - dry-run hint, forwarded as body.dryRun
|
|
474
|
+
*/
|
|
475
|
+
var OpenClawToolInvokeProvider = class {
|
|
476
|
+
toolName;
|
|
477
|
+
gatewayUrl;
|
|
478
|
+
authToken;
|
|
479
|
+
openclawConfig;
|
|
480
|
+
timeoutMs;
|
|
481
|
+
constructor(toolName, providerOptions = {}) {
|
|
482
|
+
this.toolName = toolName;
|
|
483
|
+
this.openclawConfig = providerOptions.config || {};
|
|
484
|
+
const env = providerOptions.env;
|
|
485
|
+
this.gatewayUrl = resolveGatewayUrl(this.openclawConfig, env);
|
|
486
|
+
this.authToken = resolveAuthToken(this.openclawConfig, env);
|
|
487
|
+
this.timeoutMs = this.openclawConfig.timeoutMs ?? require_fetch.REQUEST_TIMEOUT_MS;
|
|
488
|
+
}
|
|
489
|
+
id() {
|
|
490
|
+
return `openclaw:tools:${this.toolName}`;
|
|
491
|
+
}
|
|
492
|
+
toString() {
|
|
493
|
+
return `[OpenClaw Tool Provider ${this.toolName}]`;
|
|
494
|
+
}
|
|
495
|
+
toJSON() {
|
|
496
|
+
return { provider: this.id() };
|
|
497
|
+
}
|
|
498
|
+
async callApi(prompt) {
|
|
499
|
+
let args;
|
|
500
|
+
try {
|
|
501
|
+
args = JSON.parse(prompt);
|
|
502
|
+
} catch {
|
|
503
|
+
args = { input: prompt };
|
|
504
|
+
}
|
|
505
|
+
const url = `${this.gatewayUrl}/tools/invoke`;
|
|
506
|
+
const headers = {
|
|
507
|
+
"Content-Type": "application/json",
|
|
508
|
+
...this.openclawConfig.headers || {}
|
|
509
|
+
};
|
|
510
|
+
if (this.authToken) headers["Authorization"] = `Bearer ${this.authToken}`;
|
|
511
|
+
const body = {
|
|
512
|
+
tool: this.toolName,
|
|
513
|
+
...this.openclawConfig.action && { action: this.openclawConfig.action },
|
|
514
|
+
args,
|
|
515
|
+
...typeof this.openclawConfig.dry_run === "boolean" && { dryRun: this.openclawConfig.dry_run },
|
|
516
|
+
...this.openclawConfig.session_key && { sessionKey: this.openclawConfig.session_key }
|
|
517
|
+
};
|
|
518
|
+
require_logger.logger.debug(`[OpenClaw Tool] POST ${url}`, {
|
|
519
|
+
tool: this.toolName,
|
|
520
|
+
args
|
|
521
|
+
});
|
|
522
|
+
try {
|
|
523
|
+
const controller = new AbortController();
|
|
524
|
+
const fetchTimeout = setTimeout(() => controller.abort(), this.timeoutMs);
|
|
525
|
+
const response = await require_fetch.fetchWithProxy(url, {
|
|
526
|
+
method: "POST",
|
|
527
|
+
headers,
|
|
528
|
+
body: JSON.stringify(body),
|
|
529
|
+
signal: controller.signal
|
|
530
|
+
}).finally(() => clearTimeout(fetchTimeout));
|
|
531
|
+
if (!response.ok) {
|
|
532
|
+
const text = await response.text();
|
|
533
|
+
return { error: `OpenClaw tool invoke failed (${response.status}): ${text}` };
|
|
534
|
+
}
|
|
535
|
+
const data = await response.json();
|
|
536
|
+
if (data.ok) return { output: typeof data.result === "string" ? data.result : JSON.stringify(data.result) };
|
|
537
|
+
return { error: data.error || "Unknown tool error" };
|
|
538
|
+
} catch (err) {
|
|
539
|
+
return { error: `OpenClaw tool invoke error: ${err instanceof Error ? err.message : String(err)}` };
|
|
540
|
+
}
|
|
541
|
+
}
|
|
542
|
+
};
|
|
543
|
+
//#endregion
|
|
544
|
+
//#region src/providers/openclaw/index.ts
|
|
545
|
+
/**
|
|
546
|
+
* Create an OpenClaw provider from a provider path string.
|
|
547
|
+
*
|
|
548
|
+
* Routing:
|
|
549
|
+
* openclaw → OpenClawChatProvider('main')
|
|
550
|
+
* openclaw:main → OpenClawChatProvider('main')
|
|
551
|
+
* openclaw:my-agent → OpenClawChatProvider('my-agent')
|
|
552
|
+
* openclaw:responses → OpenClawResponsesProvider('main')
|
|
553
|
+
* openclaw:responses:X → OpenClawResponsesProvider('X')
|
|
554
|
+
* openclaw:agent → OpenClawAgentProvider('main')
|
|
555
|
+
* openclaw:agent:X → OpenClawAgentProvider('X')
|
|
556
|
+
* openclaw:tools:sessions_list → OpenClawToolInvokeProvider('sessions_list')
|
|
557
|
+
*/
|
|
558
|
+
function createOpenClawProvider(providerPath, providerOptions = {}, env) {
|
|
559
|
+
const splits = providerPath.split(":");
|
|
560
|
+
const keyword = splits[1];
|
|
561
|
+
const opts = {
|
|
562
|
+
...providerOptions,
|
|
563
|
+
env
|
|
564
|
+
};
|
|
565
|
+
if (keyword === "responses") return new OpenClawResponsesProvider(splits[2] || "main", opts);
|
|
566
|
+
if (keyword === "agent") return new OpenClawAgentProvider(splits[2] || "main", opts);
|
|
567
|
+
if (keyword === "tools") {
|
|
568
|
+
const toolName = splits.slice(2).join(":");
|
|
569
|
+
if (!toolName) throw new Error("OpenClaw tools provider requires a tool name: openclaw:tools:<tool-name>");
|
|
570
|
+
return new OpenClawToolInvokeProvider(toolName, opts);
|
|
571
|
+
}
|
|
572
|
+
return new OpenClawChatProvider(splits.length > 1 ? splits.slice(1).join(":") : "main", opts);
|
|
573
|
+
}
|
|
574
|
+
//#endregion
|
|
575
|
+
exports.createOpenClawProvider = createOpenClawProvider;
|
|
576
|
+
|
|
577
|
+
//# sourceMappingURL=openclaw-B8brrjC_.cjs.map
|