197 lines
7.1 KiB
JavaScript
197 lines
7.1 KiB
JavaScript
import { getEnvironmentVariable } from "@langchain/core/utils/env";
|
|
import { Tool } from "@langchain/core/tools";
|
|
/**
|
|
* Wrapper around SerpAPI.
|
|
*
|
|
* To use, you should have the `serpapi` package installed and the SERPAPI_API_KEY environment variable set.
|
|
*/
|
|
export class SerpAPI extends Tool {
|
|
static lc_name() {
|
|
return "SerpAPI";
|
|
}
|
|
toJSON() {
|
|
return this.toJSONNotImplemented();
|
|
}
|
|
constructor(apiKey = getEnvironmentVariable("SERPAPI_API_KEY"), params = {}, baseUrl = "https://serpapi.com") {
|
|
super(...arguments);
|
|
Object.defineProperty(this, "key", {
|
|
enumerable: true,
|
|
configurable: true,
|
|
writable: true,
|
|
value: void 0
|
|
});
|
|
Object.defineProperty(this, "params", {
|
|
enumerable: true,
|
|
configurable: true,
|
|
writable: true,
|
|
value: void 0
|
|
});
|
|
Object.defineProperty(this, "baseUrl", {
|
|
enumerable: true,
|
|
configurable: true,
|
|
writable: true,
|
|
value: void 0
|
|
});
|
|
Object.defineProperty(this, "name", {
|
|
enumerable: true,
|
|
configurable: true,
|
|
writable: true,
|
|
value: "search"
|
|
});
|
|
Object.defineProperty(this, "description", {
|
|
enumerable: true,
|
|
configurable: true,
|
|
writable: true,
|
|
value: "a search engine. useful for when you need to answer questions about current events. input should be a search query."
|
|
});
|
|
if (!apiKey) {
|
|
throw new Error("SerpAPI API key not set. You can set it as SERPAPI_API_KEY in your .env file, or pass it to SerpAPI.");
|
|
}
|
|
this.key = apiKey;
|
|
this.params = params;
|
|
this.baseUrl = baseUrl;
|
|
}
|
|
/**
|
|
* Builds a URL for the SerpAPI request.
|
|
* @param path The path for the request.
|
|
* @param parameters The parameters for the request.
|
|
* @param baseUrl The base URL for the request.
|
|
* @returns A string representing the built URL.
|
|
*/
|
|
buildUrl(path, parameters, baseUrl) {
|
|
const nonUndefinedParams = Object.entries(parameters)
|
|
.filter(([_, value]) => value !== undefined)
|
|
.map(([key, value]) => [key, `${value}`]);
|
|
const searchParams = new URLSearchParams(nonUndefinedParams);
|
|
return `${baseUrl}/${path}?${searchParams}`;
|
|
}
|
|
/** @ignore */
|
|
async _call(input) {
|
|
const { timeout, ...params } = this.params;
|
|
const resp = await fetch(this.buildUrl("search", {
|
|
...params,
|
|
api_key: this.key,
|
|
q: input,
|
|
}, this.baseUrl), {
|
|
signal: timeout ? AbortSignal.timeout(timeout) : undefined,
|
|
});
|
|
const res = await resp.json();
|
|
if (res.error) {
|
|
throw new Error(`Got error from serpAPI: ${res.error}`);
|
|
}
|
|
const answer_box = res.answer_box_list
|
|
? res.answer_box_list[0]
|
|
: res.answer_box;
|
|
if (answer_box) {
|
|
if (answer_box.result) {
|
|
return answer_box.result;
|
|
}
|
|
else if (answer_box.answer) {
|
|
return answer_box.answer;
|
|
}
|
|
else if (answer_box.snippet) {
|
|
return answer_box.snippet;
|
|
}
|
|
else if (answer_box.snippet_highlighted_words) {
|
|
return answer_box.snippet_highlighted_words.toString();
|
|
}
|
|
else {
|
|
const answer = {};
|
|
Object.keys(answer_box)
|
|
.filter((k) => !Array.isArray(answer_box[k]) &&
|
|
typeof answer_box[k] !== "object" &&
|
|
!(typeof answer_box[k] === "string" &&
|
|
answer_box[k].startsWith("http")))
|
|
.forEach((k) => {
|
|
answer[k] = answer_box[k];
|
|
});
|
|
return JSON.stringify(answer);
|
|
}
|
|
}
|
|
if (res.events_results) {
|
|
return JSON.stringify(res.events_results);
|
|
}
|
|
if (res.sports_results) {
|
|
return JSON.stringify(res.sports_results);
|
|
}
|
|
if (res.top_stories) {
|
|
return JSON.stringify(res.top_stories);
|
|
}
|
|
if (res.news_results) {
|
|
return JSON.stringify(res.news_results);
|
|
}
|
|
if (res.jobs_results?.jobs) {
|
|
return JSON.stringify(res.jobs_results.jobs);
|
|
}
|
|
if (res.questions_and_answers) {
|
|
return JSON.stringify(res.questions_and_answers);
|
|
}
|
|
if (res.popular_destinations?.destinations) {
|
|
return JSON.stringify(res.popular_destinations.destinations);
|
|
}
|
|
if (res.top_sights?.sights) {
|
|
const sights = res.top_sights.sights
|
|
.map((s) => ({
|
|
title: s.title,
|
|
description: s.description,
|
|
price: s.price,
|
|
}))
|
|
.slice(0, 8);
|
|
return JSON.stringify(sights);
|
|
}
|
|
if (res.shopping_results && res.shopping_results[0]?.title) {
|
|
return JSON.stringify(res.shopping_results.slice(0, 3));
|
|
}
|
|
if (res.images_results && res.images_results[0]?.thumbnail) {
|
|
return res.images_results
|
|
.map((ir) => ir.thumbnail)
|
|
.slice(0, 10)
|
|
.toString();
|
|
}
|
|
const snippets = [];
|
|
if (res.knowledge_graph) {
|
|
if (res.knowledge_graph.description) {
|
|
snippets.push(res.knowledge_graph.description);
|
|
}
|
|
const title = res.knowledge_graph.title || "";
|
|
Object.keys(res.knowledge_graph)
|
|
.filter((k) => typeof res.knowledge_graph[k] === "string" &&
|
|
k !== "title" &&
|
|
k !== "description" &&
|
|
!k.endsWith("_stick") &&
|
|
!k.endsWith("_link") &&
|
|
!k.startsWith("http"))
|
|
.forEach((k) => snippets.push(`${title} ${k}: ${res.knowledge_graph[k]}`));
|
|
}
|
|
const first_organic_result = res.organic_results?.[0];
|
|
if (first_organic_result) {
|
|
if (first_organic_result.snippet) {
|
|
snippets.push(first_organic_result.snippet);
|
|
}
|
|
else if (first_organic_result.snippet_highlighted_words) {
|
|
snippets.push(first_organic_result.snippet_highlighted_words);
|
|
}
|
|
else if (first_organic_result.rich_snippet) {
|
|
snippets.push(first_organic_result.rich_snippet);
|
|
}
|
|
else if (first_organic_result.rich_snippet_table) {
|
|
snippets.push(first_organic_result.rich_snippet_table);
|
|
}
|
|
else if (first_organic_result.link) {
|
|
snippets.push(first_organic_result.link);
|
|
}
|
|
}
|
|
if (res.buying_guide) {
|
|
snippets.push(res.buying_guide);
|
|
}
|
|
if (res.local_results?.places) {
|
|
snippets.push(res.local_results.places);
|
|
}
|
|
if (snippets.length > 0) {
|
|
return JSON.stringify(snippets);
|
|
}
|
|
else {
|
|
return "No good search result found";
|
|
}
|
|
}
|
|
}
|