From 3fa4151abc966e308cf6c1af1c9bd41fb0e4728c Mon Sep 17 00:00:00 2001 From: Ting Chien Meng Date: Fri, 22 Nov 2024 11:24:43 -0500 Subject: [PATCH 1/3] integrate tavily web search --- packages/client-direct/src/index.ts | 2 +- packages/core/src/generation.ts | 37 ++++- packages/core/src/types.ts | 18 +++ packages/plugin-web-search/package.json | 17 ++ packages/plugin-web-search/src/index.ts | 188 ++++++++++++++++++++++ packages/plugin-web-search/tsconfig.json | 9 ++ packages/plugin-web-search/tsup.config.ts | 21 +++ 7 files changed, 290 insertions(+), 2 deletions(-) create mode 100644 packages/plugin-web-search/package.json create mode 100644 packages/plugin-web-search/src/index.ts create mode 100644 packages/plugin-web-search/tsconfig.json create mode 100644 packages/plugin-web-search/tsup.config.ts diff --git a/packages/client-direct/src/index.ts b/packages/client-direct/src/index.ts index 123600bf55..65a2c4cc00 100644 --- a/packages/client-direct/src/index.ts +++ b/packages/client-direct/src/index.ts @@ -233,7 +233,7 @@ export class DirectClient { ); if (message) { - res.json([message, response]); + res.json([response, message]); } else { res.json([response]); } diff --git a/packages/core/src/generation.ts b/packages/core/src/generation.ts index 24cbf8fd38..4d1f70211d 100644 --- a/packages/core/src/generation.ts +++ b/packages/core/src/generation.ts @@ -30,6 +30,7 @@ import { ModelClass, ModelProviderName, ServiceType, + SearchResponse, } from "./types.ts"; /** @@ -881,6 +882,40 @@ export const generateCaption = async ( description: resp.description.trim(), }; }; + +export const generateWebSearch = async ( + query: string, + runtime: IAgentRuntime +): Promise => { + const apiUrl = "https://api.tavily.com/search"; + const apiKey = runtime.getSetting("TAVILY_API_KEY"); + + try { + const response = await fetch(apiUrl, { + method: "POST", + headers: { + "Content-Type": "application/json", + }, + body: JSON.stringify({ + api_key: apiKey, + query, + include_answer: true, + }), + }); + + if (!response.ok) { + throw new elizaLogger.error( + `HTTP error! status: ${response.status}` + ); + } + + const data: SearchResponse = await response.json(); + console.log("Response:", data); + return data; + } catch (error) { + elizaLogger.error("Error:", error); + } +}; /** * Configuration options for generating objects with a model. */ @@ -1047,7 +1082,7 @@ async function handleOpenAI({ mode, modelOptions, }: ProviderOptions): Promise> { - const baseURL = models.openai.endpoint || undefined + const baseURL = models.openai.endpoint || undefined; const openai = createOpenAI({ apiKey, baseURL }); return await aiGenerateObject({ model: openai.languageModel(model), diff --git a/packages/core/src/types.ts b/packages/core/src/types.ts index 0b8c43569c..0275948fb8 100644 --- a/packages/core/src/types.ts +++ b/packages/core/src/types.ts @@ -1083,6 +1083,23 @@ export interface IPdfService extends Service { convertPdfToText(pdfBuffer: Buffer): Promise; } +export type SearchResult = { + title: string; + url: string; + content: string; + score: number; + raw_content: string | null; +}; + +export type SearchResponse = { + query: string; + follow_up_questions: string[] | null; + answer: string | null; + images: string[]; + results: SearchResult[]; + response_time: number; +}; + export enum ServiceType { IMAGE_DESCRIPTION = "image_description", TRANSCRIPTION = "transcription", @@ -1091,6 +1108,7 @@ export enum ServiceType { BROWSER = "browser", SPEECH_GENERATION = "speech_generation", PDF = "pdf", + WEB_SEARCH = "web_search", } export enum LoggingLevel { diff --git a/packages/plugin-web-search/package.json b/packages/plugin-web-search/package.json new file mode 100644 index 0000000000..84e18b8a43 --- /dev/null +++ b/packages/plugin-web-search/package.json @@ -0,0 +1,17 @@ +{ + "name": "@ai16z/plugin-web-search", + "version": "0.1.3", + "main": "dist/index.js", + "type": "module", + "types": "dist/index.d.ts", + "dependencies": { + "@ai16z/eliza": "workspace:*", + "tsup": "^8.3.5" + }, + "scripts": { + "build": "tsup --format esm --dts" + }, + "peerDependencies": { + "whatwg-url": "7.1.0" + } +} diff --git a/packages/plugin-web-search/src/index.ts b/packages/plugin-web-search/src/index.ts new file mode 100644 index 0000000000..4d7d7be44c --- /dev/null +++ b/packages/plugin-web-search/src/index.ts @@ -0,0 +1,188 @@ +import { elizaLogger } from "@ai16z/eliza"; +import { + Action, + HandlerCallback, + IAgentRuntime, + Memory, + Plugin, + State, +} from "@ai16z/eliza"; +import { generateWebSearch } from "@ai16z/eliza"; + +import { SearchResult } from "@ai16z/eliza"; + +const webSearch: Action = { + name: "WEB_SEARCH", + similes: [ + "SEARCH_WEB", + "INTERNET_SEARCH", + "LOOKUP", + "QUERY_WEB", + "FIND_ONLINE", + "SEARCH_ENGINE", + "WEB_LOOKUP", + "ONLINE_SEARCH", + "FIND_INFORMATION", + ], + description: + "Perform a web search to find information related to the message.", + validate: async (runtime: IAgentRuntime, message: Memory) => { + const tavilyApiKeyOk = !!runtime.getSetting("TAVILY_API_KEY"); + + return tavilyApiKeyOk; + }, + handler: async ( + runtime: IAgentRuntime, + message: Memory, + state: State, + options: any, + callback: HandlerCallback + ) => { + elizaLogger.log("Composing state for message:", message); + state = (await runtime.composeState(message)) as State; + const userId = runtime.agentId; + elizaLogger.log("User ID:", userId); + + const webSearchPrompt = message.content.text; + elizaLogger.log("web search prompt received:", webSearchPrompt); + + elizaLogger.log("Generating image with prompt:", webSearchPrompt); + const searchResponse = await generateWebSearch( + webSearchPrompt, + runtime + ); + + if (searchResponse && searchResponse.results.length) { + const responseList = searchResponse.answer + ? `${searchResponse.answer}${ + Array.isArray(searchResponse.results) && + searchResponse.results.length > 0 + ? `\n\nFor more details, you can check out these resources:\n${searchResponse.results + .map( + (result: SearchResult, index: number) => + `${index + 1}. [${result.title}](${result.url})` + ) + .join("\n")}` + : "" + }` + : ""; + + callback({ + text: responseList, + }); + } else { + elizaLogger.error("search failed or returned no data."); + } + }, + examples: [ + [ + { + user: "{{user1}}", + content: { + text: "Find the latest news about SpaceX launches.", + }, + }, + { + user: "{{agentName}}", + content: { + text: "Here is the latest news about SpaceX launches:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { + text: "Can you find details about the iPhone 16 release?", + }, + }, + { + user: "{{agentName}}", + content: { + text: "Here are the details I found about the iPhone 16 release:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { + text: "What is the schedule for the next FIFA World Cup?", + }, + }, + { + user: "{{agentName}}", + content: { + text: "Here is the schedule for the next FIFA World Cup:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { text: "Check the latest stock price of Tesla." }, + }, + { + user: "{{agentName}}", + content: { + text: "Here is the latest stock price of Tesla I found:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { + text: "What are the current trending movies in the US?", + }, + }, + { + user: "{{agentName}}", + content: { + text: "Here are the current trending movies in the US:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { + text: "What is the latest score in the NBA finals?", + }, + }, + { + user: "{{agentName}}", + content: { + text: "Here is the latest score from the NBA finals:", + action: "WEB_SEARCH", + }, + }, + ], + [ + { + user: "{{user1}}", + content: { text: "When is the next Apple keynote event?" }, + }, + { + user: "{{agentName}}", + content: { + text: "Here is the information about the next Apple keynote event:", + action: "WEB_SEARCH", + }, + }, + ], + ], +} as Action; + +export const webSearchPlugin: Plugin = { + name: "webSearch", + description: "Search web", + actions: [webSearch], + evaluators: [], + providers: [], +}; diff --git a/packages/plugin-web-search/tsconfig.json b/packages/plugin-web-search/tsconfig.json new file mode 100644 index 0000000000..69b4220036 --- /dev/null +++ b/packages/plugin-web-search/tsconfig.json @@ -0,0 +1,9 @@ +{ + "extends": "../../tsconfig.json", + "compilerOptions": { + "outDir": "dist", + "rootDir": "src", + "types": ["node"] + }, + "include": ["src/**/*.ts"] +} diff --git a/packages/plugin-web-search/tsup.config.ts b/packages/plugin-web-search/tsup.config.ts new file mode 100644 index 0000000000..b5e4388b21 --- /dev/null +++ b/packages/plugin-web-search/tsup.config.ts @@ -0,0 +1,21 @@ +import { defineConfig } from "tsup"; + +export default defineConfig({ + entry: ["src/index.ts"], + outDir: "dist", + sourcemap: true, + clean: true, + format: ["esm"], // Ensure you're targeting CommonJS + external: [ + "dotenv", // Externalize dotenv to prevent bundling + "fs", // Externalize fs to use Node.js built-in module + "path", // Externalize other built-ins if necessary + "@reflink/reflink", + "@node-llama-cpp", + "https", + "http", + "agentkeepalive", + "zod", + // Add other modules you want to externalize + ], +}); From bd6fcbe2a856a04b0781d1688c77ade74aa681cb Mon Sep 17 00:00:00 2001 From: Ting Chien Meng Date: Fri, 22 Nov 2024 11:26:46 -0500 Subject: [PATCH 2/3] clean code --- packages/core/src/generation.ts | 1 - packages/core/src/types.ts | 1 - 2 files changed, 2 deletions(-) diff --git a/packages/core/src/generation.ts b/packages/core/src/generation.ts index 4d1f70211d..e3ced2ab18 100644 --- a/packages/core/src/generation.ts +++ b/packages/core/src/generation.ts @@ -910,7 +910,6 @@ export const generateWebSearch = async ( } const data: SearchResponse = await response.json(); - console.log("Response:", data); return data; } catch (error) { elizaLogger.error("Error:", error); diff --git a/packages/core/src/types.ts b/packages/core/src/types.ts index 0275948fb8..daa777c332 100644 --- a/packages/core/src/types.ts +++ b/packages/core/src/types.ts @@ -1108,7 +1108,6 @@ export enum ServiceType { BROWSER = "browser", SPEECH_GENERATION = "speech_generation", PDF = "pdf", - WEB_SEARCH = "web_search", } export enum LoggingLevel { From 529328f1a802815adf81b52857cde85aa2e60d72 Mon Sep 17 00:00:00 2001 From: Ting Chien Meng Date: Tue, 26 Nov 2024 11:09:25 -0500 Subject: [PATCH 3/3] add web search package --- packages/plugin-web-search/.npmignore | 6 ++++++ scripts/build.sh | 1 + 2 files changed, 7 insertions(+) create mode 100644 packages/plugin-web-search/.npmignore diff --git a/packages/plugin-web-search/.npmignore b/packages/plugin-web-search/.npmignore new file mode 100644 index 0000000000..078562ecea --- /dev/null +++ b/packages/plugin-web-search/.npmignore @@ -0,0 +1,6 @@ +* + +!dist/** +!package.json +!readme.md +!tsup.config.ts \ No newline at end of file diff --git a/scripts/build.sh b/scripts/build.sh index 1a93101dcc..5ef239cca6 100644 --- a/scripts/build.sh +++ b/scripts/build.sh @@ -36,6 +36,7 @@ PACKAGES=( "plugin-node" "plugin-bootstrap" "plugin-image-generation" + "plugin-web-search" ) # Build packages in specified order