mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-09 16:21:15 +00:00
* fix(xai): make x_search auth plugin-owned * fix(xai): restore x_search runtime migration fallback * fix(xai): narrow legacy x_search auth migration * fix(secrets): drop legacy x_search target registry entry * fix(xai): no-op knob-only x_search migration fallback
300 lines
8.1 KiB
TypeScript
300 lines
8.1 KiB
TypeScript
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import { withFetchPreconnect } from "../../test/helpers/plugins/fetch-mock.js";
|
|
import { createXSearchTool } from "./x-search.js";
|
|
|
|
function installXSearchFetch(payload?: Record<string, unknown>) {
|
|
const mockFetch = vi.fn((_input?: unknown, _init?: unknown) =>
|
|
Promise.resolve({
|
|
ok: true,
|
|
json: () =>
|
|
Promise.resolve(
|
|
payload ?? {
|
|
output: [
|
|
{
|
|
type: "message",
|
|
content: [
|
|
{
|
|
type: "output_text",
|
|
text: "Found X posts",
|
|
annotations: [{ type: "url_citation", url: "https://x.com/openclaw/status/1" }],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
citations: ["https://x.com/openclaw/status/1"],
|
|
},
|
|
),
|
|
} as Response),
|
|
);
|
|
global.fetch = withFetchPreconnect(mockFetch);
|
|
return mockFetch;
|
|
}
|
|
|
|
function parseFirstRequestBody(mockFetch: ReturnType<typeof installXSearchFetch>) {
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
const requestBody = request?.body;
|
|
return JSON.parse(typeof requestBody === "string" ? requestBody : "{}") as Record<
|
|
string,
|
|
unknown
|
|
>;
|
|
}
|
|
|
|
afterEach(() => {
|
|
vi.restoreAllMocks();
|
|
});
|
|
|
|
describe("xai x_search tool", () => {
|
|
it("enables x_search when runtime config carries the shared xAI key", () => {
|
|
const tool = createXSearchTool({
|
|
config: {},
|
|
runtimeConfig: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "x-search-runtime-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(tool?.name).toBe("x_search");
|
|
});
|
|
|
|
it("enables x_search when the xAI plugin web search key is configured", () => {
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-plugin-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(tool?.name).toBe("x_search");
|
|
});
|
|
|
|
it("uses the xAI Responses x_search tool with structured filters", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-config-test", // pragma: allowlist secret
|
|
},
|
|
xSearch: {
|
|
model: "grok-4-1-fast-non-reasoning",
|
|
maxTurns: 2,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
const result = await tool?.execute?.("x-search:1", {
|
|
query: "dinner recipes",
|
|
allowed_x_handles: ["openclaw"],
|
|
excluded_x_handles: ["spam"],
|
|
from_date: "2026-03-01",
|
|
to_date: "2026-03-20",
|
|
enable_image_understanding: true,
|
|
});
|
|
|
|
expect(mockFetch).toHaveBeenCalled();
|
|
expect(String(mockFetch.mock.calls[0]?.[0])).toContain("api.x.ai/v1/responses");
|
|
const body = parseFirstRequestBody(mockFetch);
|
|
expect(body.model).toBe("grok-4-1-fast-non-reasoning");
|
|
expect(body.max_turns).toBe(2);
|
|
expect(body.tools).toEqual([
|
|
{
|
|
type: "x_search",
|
|
allowed_x_handles: ["openclaw"],
|
|
excluded_x_handles: ["spam"],
|
|
from_date: "2026-03-01",
|
|
to_date: "2026-03-20",
|
|
enable_image_understanding: true,
|
|
},
|
|
]);
|
|
expect((result?.details as { citations?: string[] } | undefined)?.citations).toEqual([
|
|
"https://x.com/openclaw/status/1",
|
|
]);
|
|
});
|
|
|
|
it("reuses the xAI plugin web search key for x_search requests", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-plugin-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("x-search:plugin-key", {
|
|
query: "latest post from huntharo",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer xai-plugin-key",
|
|
);
|
|
});
|
|
|
|
it("prefers the active runtime config for shared xAI keys", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: { source: "env", provider: "default", id: "X_SEARCH_KEY_REF" },
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
runtimeConfig: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "x-search-runtime-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("x-search:runtime-key", {
|
|
query: "runtime key search",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer x-search-runtime-key",
|
|
);
|
|
});
|
|
|
|
it("reuses the legacy grok web search key for x_search requests", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
tools: {
|
|
web: {
|
|
search: {
|
|
grok: {
|
|
apiKey: "xai-legacy-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("x-search:legacy-key", {
|
|
query: "latest legacy-key post from huntharo",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer xai-legacy-key",
|
|
);
|
|
});
|
|
|
|
it("uses migrated runtime auth when the source config still carries legacy x_search apiKey", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
tools: {
|
|
web: {
|
|
x_search: {
|
|
apiKey: "legacy-x-search-key", // pragma: allowlist secret
|
|
enabled: true,
|
|
} as Record<string, unknown>,
|
|
},
|
|
},
|
|
},
|
|
runtimeConfig: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "migrated-runtime-key", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await tool?.execute?.("x-search:migrated-runtime-key", {
|
|
query: "migrated runtime auth",
|
|
});
|
|
|
|
const request = mockFetch.mock.calls[0]?.[1] as RequestInit | undefined;
|
|
expect((request?.headers as Record<string, string> | undefined)?.Authorization).toBe(
|
|
"Bearer migrated-runtime-key",
|
|
);
|
|
});
|
|
|
|
it("rejects invalid date ordering before calling xAI", async () => {
|
|
const mockFetch = installXSearchFetch();
|
|
const tool = createXSearchTool({
|
|
config: {
|
|
plugins: {
|
|
entries: {
|
|
xai: {
|
|
config: {
|
|
webSearch: {
|
|
apiKey: "xai-config-test", // pragma: allowlist secret
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
await expect(
|
|
tool?.execute?.("x-search:bad-dates", {
|
|
query: "dinner recipes",
|
|
from_date: "2026-03-20",
|
|
to_date: "2026-03-01",
|
|
}),
|
|
).rejects.toThrow(/from_date must be on or before to_date/i);
|
|
expect(mockFetch).not.toHaveBeenCalled();
|
|
});
|
|
});
|