Files
pi-web/index.ts
Evan Reichard 42ce8730ab refactor: complete pi-web rename in paths and env vars
- Config path: ~/.pi/pi-search/config.json -> ~/.pi/pi-web/config.json
- Env vars: PI_SEARCH_PROVIDER -> PI_WEB_PROVIDER,
  PI_SEARCH_SEARXNG_URL -> PI_WEB_SEARXNG_URL
- README + flake.nix description updated to match.
2026-05-25 11:58:18 -04:00

104 lines
3.5 KiB
TypeScript

// Pi-Web Extension - Registers `web_search` and `web_fetch` tools backed by
// a shared headless Firefox session. Provider config lives in
// ~/.pi/pi-web/config.json (env overrides supported).
import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
import { Type } from "typebox";
import { ConfigError, loadConfig, resolveSettings } from "./src/config.ts";
import { fetchPage, type FetchResult } from "./src/fetch.ts";
import { searchKagi } from "./src/providers/kagi.ts";
import { searchSearxng } from "./src/providers/searxng.ts";
import { SearchError, type SearchResult } from "./src/types.ts";
function formatResults(results: SearchResult[]): string {
if (results.length === 0) return "(no results)";
return results
.map((r) => `## [${r.title}](${r.url})\n> ${r.description}`)
.join("\n\n");
}
function formatFetch(r: FetchResult): string {
const header =
r.title && r.title.trim()
? `# ${r.title}\n\n<${r.finalUrl}>\n\n`
: `<${r.finalUrl}>\n\n`;
return `${header}${r.markdown}`;
}
async function runSearch(query: string): Promise<SearchResult[]> {
const config = loadConfig();
const settings = resolveSettings(config);
switch (settings.provider) {
case "kagi":
return searchKagi({ query, token: settings.kagiToken ?? "" });
case "searxng":
return searchSearxng({ query, baseUrl: settings.searxngBaseUrl ?? "" });
}
}
export default function (pi: ExtensionAPI) {
pi.registerTool({
name: "web_search",
label: "Web Search",
description:
"Search the web. Returns a markdown list of titles, URLs, and snippets.",
promptSnippet: "Search the web for the given query",
parameters: Type.Object({
query: Type.String({ description: "Search query text" }),
}),
async execute(_toolCallId, params) {
try {
const results = await runSearch(params.query);
return {
content: [{ type: "text", text: formatResults(results) }],
details: { raw: results },
};
} catch (err) {
if (err instanceof ConfigError || err instanceof SearchError) {
return {
content: [{ type: "text", text: `Search error: ${err.message}` }],
isError: true,
details: { raw: [] as SearchResult[] },
};
}
throw err;
}
},
});
pi.registerTool({
name: "web_fetch",
label: "Web Fetch",
description:
"Fetch a URL and return the page content as readable markdown (Readability + Turndown over a headless Firefox session, so JS-rendered pages work). For raw HTML, non-text content (PDFs, images), or simple HTTP requests, use bash with curl instead.",
promptSnippet: "Fetch a URL and convert the page to readable markdown",
parameters: Type.Object({
url: Type.String({ description: "Absolute URL to fetch" }),
}),
async execute(_toolCallId, params) {
try {
const result = await fetchPage({ url: params.url });
return {
content: [{ type: "text", text: formatFetch(result) }],
details: { raw: result },
};
} catch (err) {
if (err instanceof SearchError) {
return {
content: [{ type: "text", text: `Fetch error: ${err.message}` }],
isError: true,
details: {
raw: {
url: params.url,
finalUrl: params.url,
markdown: "",
truncated: false,
} satisfies FetchResult,
},
};
}
throw err;
}
},
});
}