feat: add wolfram alpha client

pull/643/head^2
Travis Fischer 2024-06-04 18:59:44 -05:00
rodzic 6045e2323c
commit 7d11f7632a
4 zmienionych plików z 110 dodań i 11 usunięć

Wyświetl plik

@ -10,7 +10,8 @@ import restoreCursor from 'restore-cursor'
// import { PerigonClient } from '../src/index.js' // import { PerigonClient } from '../src/index.js'
// import { FirecrawlClient } from '../src/index.js' // import { FirecrawlClient } from '../src/index.js'
// import { ExaClient } from '../src/index.js' // import { ExaClient } from '../src/index.js'
import { DiffbotClient } from '../src/index.js' // import { DiffbotClient } from '../src/index.js'
import { WolframClient } from '../src/index.js'
/** /**
* Scratch pad for testing. * Scratch pad for testing.
@ -65,15 +66,24 @@ async function main() {
// }) // })
// console.log(JSON.stringify(res, null, 2)) // console.log(JSON.stringify(res, null, 2))
const diffbot = new DiffbotClient() // const diffbot = new DiffbotClient()
// // const res = await diffbot.analyzeUrl({
// // url: 'https://www.bbc.com/news/articles/cp4475gwny1o'
// // })
// const res = await diffbot.enhanceEntity({
// type: 'Person',
// name: 'Kevin Raheja'
// })
// console.log(JSON.stringify(res, null, 2))
const wolfram = new WolframClient()
// const res = await diffbot.analyzeUrl({ // const res = await diffbot.analyzeUrl({
// url: 'https://www.bbc.com/news/articles/cp4475gwny1o' // url: 'https://www.bbc.com/news/articles/cp4475gwny1o'
// }) // })
const res = await diffbot.enhanceEntity({ const res = await wolfram.ask({
type: 'Person', input: 'population of new york city'
name: 'Kevin Raheja'
}) })
console.log(JSON.stringify(res, null, 2)) console.log(res)
} }
try { try {

Wyświetl plik

@ -149,24 +149,22 @@ The SDK-specific imports are all isolated to keep the main `@agentic/stdlib` as
## TODO ## TODO
- rename this repo to agentic - rename this repo to agentic
- change license to MIT
- sdks - sdks
- instructor-js
- TODO - TODO
- services - services
- e2b - e2b
- search-and-scrape - search-and-scrape
- replicate - replicate
- huggingface - huggingface
- wolfram alpha - [skyvern](https://github.com/Skyvern-AI/skyvern)
- midjourney - midjourney
- unstructured - unstructured
- pull from [langchain](https://github.com/langchain-ai/langchainjs/tree/main/langchain) - pull from [langchain](https://github.com/langchain-ai/langchainjs/tree/main/langchain)
- provide a converter for langchain `DynamicStructuredTool` - provide a converter for langchain `DynamicStructuredTool`
- pull from other libs - pull from other libs
- pull from [nango](https://docs.nango.dev/integrations/overview) - pull from [nango](https://docs.nango.dev/integrations/overview)
- tools - pull from [activepieces](https://github.com/activepieces/activepieces/tree/main/packages/pieces/community)
- calculator - general openapi support ala [workgpt](https://github.com/team-openpm/workgpt)
- tools / chains / flows / runnables - tools / chains / flows / runnables
- market maps - market maps
- https://github.com/causaly/zod-validation-error - https://github.com/causaly/zod-validation-error

Wyświetl plik

@ -14,3 +14,4 @@ export * from './serper-client.js'
export * from './twitter-client.js' export * from './twitter-client.js'
export * from './weather-client.js' export * from './weather-client.js'
export * from './wikipedia-client.js' export * from './wikipedia-client.js'
export * from './wolfram-client.js'

Wyświetl plik

@ -0,0 +1,90 @@
import defaultKy, { type KyInstance } from 'ky'
import { z } from 'zod'
import { aiFunction, AIFunctionsProvider } from '../fns.js'
import { assert, getEnv } from '../utils.js'
export namespace wolfram {
export const API_BASE_URL = 'https://www.wolframalpha.com/api/'
export const AskWolframAlphaOptionsSchema = z.object({
input: z.string().describe('english query'),
maxchars: z
.number()
.int()
.positive()
.default(6000)
.optional()
.describe('max characters to generate in the response')
})
export type AskWolframAlphaOptions = z.infer<
typeof AskWolframAlphaOptionsSchema
>
}
/**
* Wolfram Alpha LLM API client for answering computational, mathematical, and
* scientific questions.
*
* @see https://products.wolframalpha.com/llm-api/documentation
*/
export class WolframClient extends AIFunctionsProvider {
readonly ky: KyInstance
readonly appId: string
readonly apiBaseUrl: string
constructor({
appId = getEnv('WOLFRAM_APP_ID'),
apiBaseUrl = wolfram.API_BASE_URL,
ky = defaultKy
}: {
appId?: string
apiBaseUrl?: string
ky?: KyInstance
} = {}) {
assert(
appId,
'WolframClient missing required "appId" (defaults to "WOLFRAM_APP_ID")'
)
super()
this.appId = appId
this.apiBaseUrl = apiBaseUrl
this.ky = ky.extend({
prefixUrl: apiBaseUrl,
headers: {
Authorization: `Bearer ${appId}`
}
})
}
@aiFunction({
name: 'ask_wolfram_alpha',
description: `
- WolframAlpha understands natural language queries about entities in chemistry, physics, geography, history, art, astronomy, and more.
- WolframAlpha performs mathematical calculations, date and unit conversions, formula solving, etc.
- Convert inputs to simplified keyword queries whenever possible (e.g. convert "how many people live in France" to "France population").
- Send queries in English only; translate non-English queries before sending, then respond in the original language.
- ALWAYS use this exponent notation: \`6*10^14\`, NEVER \`6e14\`.
- ALWAYS use proper Markdown formatting for all math, scientific, and chemical formulas, symbols, etc.: '$$\n[expression]\n$$' for standalone cases and '( [expression] )' when inline.
- Use ONLY single-letter variable names, with or without integer subscript (e.g., n, n1, n_1).
- Use named physical constants (e.g., 'speed of light') without numerical substitution.
- Include a space between compound units (e.g., "Ω m" for "ohm*meter").
- To solve for a variable in an equation with units, consider solving a corresponding equation without units; exclude counting units (e.g., books), include genuine units (e.g., kg).
- If a WolframAlpha result is not relevant to the query:
- If Wolfram provides multiple 'Assumptions' for a query, choose the more relevant one(s) without explaining the initial result. If you are unsure, ask the user to choose.
- Re-send the exact same 'input' with NO modifications, and add the 'assumption' parameter, formatted as a list, with the relevant values.
- ONLY simplify or rephrase the initial query if a more relevant 'Assumption' or other input suggestions are not provided.
`.trim(),
inputSchema: wolfram.AskWolframAlphaOptionsSchema
})
async ask(queryOrOptions: string | wolfram.AskWolframAlphaOptions) {
const options =
typeof queryOrOptions === 'string'
? { input: queryOrOptions }
: queryOrOptions
return this.ky.get('v1/llm-api', { searchParams: { ...options } }).text()
}
}