Skip to content

Commit f6b8a7d

Browse files
committed
save
1 parent 10088cd commit f6b8a7d

File tree

15 files changed

+637
-30
lines changed

15 files changed

+637
-30
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
OPENAI_API_KEY=
2+
WALLET_PRIVATE_KEY=
3+
ALCHEMY_API_KEY=
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
# Vercel AI with viem Example
2+
3+
## Setup
4+
5+
Copy the `.env.template` and populate with your values.
6+
7+
```
8+
cp .env.template .env
9+
```
10+
11+
## Usage
12+
13+
```
14+
npx ts-node index.ts
15+
```
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
import { openai } from "@ai-sdk/openai";
2+
import { generateText } from "ai";
3+
4+
import { http } from "viem";
5+
import { createWalletClient } from "viem";
6+
import { privateKeyToAccount } from "viem/accounts";
7+
import { sepolia } from "viem/chains";
8+
9+
import { getOnChainTools } from "@goat-sdk/adapter-vercel-ai";
10+
11+
import { viem } from "@goat-sdk/wallet-viem";
12+
import { worldstore } from "@goat-sdk/plugin-worldstore";
13+
14+
import fs from "fs";
15+
require("dotenv").config();
16+
17+
const account = privateKeyToAccount(
18+
process.env.WALLET_PRIVATE_KEY as `0x${string}`,
19+
);
20+
21+
const walletClient = createWalletClient({
22+
account: account,
23+
transport: http(process.env.ALCHEMY_API_KEY),
24+
chain: sepolia,
25+
});
26+
27+
(async () => {
28+
const tools = await getOnChainTools({
29+
wallet: viem(walletClient),
30+
plugins: [
31+
worldstore(),
32+
],
33+
});
34+
35+
const searchPrompt = "\nI'm looking to buy a new snowboard, my favorite color is red\n";
36+
console.log(searchPrompt);
37+
const result = await generateText({
38+
model: openai("gpt-4o-mini"),
39+
tools: tools,
40+
maxSteps: 5,
41+
prompt: searchPrompt,
42+
});
43+
44+
45+
const searchProductToolResult = getSearchProductsToolResult(result);
46+
47+
// save result to file
48+
fs.writeFileSync("result.json", JSON.stringify(result, null, 2));
49+
50+
console.log(result.text);
51+
const userResponse = await getUserInput();
52+
console.log(`\n`);
53+
54+
55+
const purchasePrompt = `
56+
Previous search result: ${searchProductToolResult}
57+
58+
User response: "${userResponse}"
59+
60+
Based on this response and the previous search result, if the user wants to purchase the item, proceed with the purchase using the product details from the search. If they decline or are unsure, acknowledge their response politely.`;
61+
62+
const purchaseResult = await generateText({
63+
model: openai("gpt-4o-mini"),
64+
tools: tools,
65+
maxSteps: 5,
66+
prompt: purchasePrompt,
67+
});
68+
69+
console.log(purchaseResult.text);
70+
71+
// console.log(result);
72+
73+
// console.log(result.text);
74+
// Exit the process after user input
75+
process.exit(0);
76+
})();
77+
78+
79+
80+
function getSearchProductsToolResult(result: Awaited<ReturnType<typeof generateText>>) {
81+
const searchProductToolResult = result.steps[0].toolResults.find(toolResult => (toolResult as any).toolName === "search_products");
82+
if (!searchProductToolResult) {
83+
throw new Error("Search products tool result not found");
84+
}
85+
return (searchProductToolResult as any).result;
86+
}
87+
88+
89+
// Wait for user input in console to continue
90+
async function getUserInput() {
91+
// wait for user input, enter to continue, and convert Buffer to string
92+
93+
console.log("--------------------------------");
94+
console.log("Type your response and press enter to continue...\n");
95+
return await new Promise<string>(resolve =>
96+
process.stdin.once('data', data => resolve(data.toString().trim()))
97+
);
98+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
{
2+
"name": "goat-examples-vercel-ai-coingecko",
3+
"version": "0.1.0",
4+
"description": "",
5+
"private": true,
6+
"scripts": {
7+
"test": "vitest run --passWithNoTests"
8+
},
9+
"author": "",
10+
"license": "MIT",
11+
"dependencies": {
12+
"@ai-sdk/openai": "^1.0.4",
13+
"@goat-sdk/adapter-vercel-ai": "workspace:*",
14+
"@goat-sdk/core": "workspace:*",
15+
"@goat-sdk/plugin-worldstore": "workspace:*",
16+
"@goat-sdk/wallet-viem": "workspace:*",
17+
"ai": "^4.0.3",
18+
"dotenv": "^16.4.5",
19+
"viem": "2.21.49"
20+
}
21+
}

typescript/examples/vercel-ai/worldstore/result.json

+290
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"extends": "../../../tsconfig.base.json",
3+
"compilerOptions": {
4+
"outDir": "dist"
5+
},
6+
"include": ["index.ts"],
7+
"exclude": ["node_modules", "dist"]
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
# Goat Coingecko Plugin 🐐 - TypeScript
2+
3+
Coingecko plugin for Goat. Allows you to create tools for interacting with the CoinGecko API.
4+
5+
## Installation
6+
```
7+
npm install @goat-sdk/plugin-coingecko
8+
```
9+
10+
## Setup
11+
12+
```typescript
13+
import { coingecko } from "@goat-sdk/plugin-coingecko";
14+
15+
const plugin = coingecko({
16+
apiKey: process.env.COINGECKO_API_KEY
17+
});
18+
```
19+
20+
## Available Actions
21+
22+
### Get Trending Coins
23+
Fetches the current trending cryptocurrencies.
24+
25+
### Get Coin Price
26+
Fetches the current price and optional market data for a specific cryptocurrency.
27+
28+
## Goat
29+
30+
<div align="center">
31+
Go out and eat some grass.
32+
33+
[Docs](https://ohmygoat.dev) | [Examples](https://github.com/goat-sdk/goat/tree/main/typescript/examples) | [Discord](https://discord.gg/goat-sdk)</div>
34+
35+
## Goat 🐐
36+
Goat 🐐 (Great Onchain Agent Toolkit) is an open-source library enabling AI agents to interact with blockchain protocols and smart contracts via their own wallets.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
{
2+
"name": "@goat-sdk/plugin-worldstore",
3+
"version": "0.1.0",
4+
"files": ["dist/**/*", "README.md", "package.json"],
5+
"scripts": {
6+
"build": "tsup",
7+
"clean": "rm -rf dist",
8+
"test": "vitest run --passWithNoTests"
9+
},
10+
"sideEffects": false,
11+
"main": "./dist/index.js",
12+
"module": "./dist/index.mjs",
13+
"types": "./dist/index.d.ts",
14+
"dependencies": {
15+
"@goat-sdk/core": "workspace:*",
16+
"zod": "^3.23.8"
17+
},
18+
"peerDependencies": {
19+
"@goat-sdk/core": "workspace:*"
20+
},
21+
"homepage": "https://ohmygoat.dev",
22+
"repository": {
23+
"type": "git",
24+
"url": "git+https://github.com/goat-sdk/goat.git"
25+
},
26+
"license": "MIT",
27+
"bugs": {
28+
"url": "https://github.com/goat-sdk/goat/issues"
29+
},
30+
"keywords": ["ai", "agents", "web3"]
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
import type { EVMWalletClient, Plugin } from "@goat-sdk/core";
2+
import { getTools } from "./tools";
3+
4+
export function worldstore(): Plugin<EVMWalletClient> {
5+
return {
6+
name: "worldstore",
7+
supportsChain: (chain) => chain.type === "evm",
8+
supportsSmartWallets: () => true,
9+
getTools: async () => getTools()
10+
};
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { z } from "zod";
2+
3+
4+
export const searchProductsParametersSchema = z.object({
5+
query: z.string().describe("The search query to find products by"),
6+
limit: z.number().optional().default(3).describe("The number of products to return. Defaults to 3."),
7+
});
8+
9+
10+
export const buyProductParametersSchema = z.object({
11+
productId: z.string().describe("The ID of the product to buy"),
12+
quantity: z.number().describe("The quantity of the product to buy"),
13+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import type { DeferredTool, EVMWalletClient, WalletClient } from "@goat-sdk/core";
2+
import type { z } from "zod";
3+
4+
import {
5+
buyProductParametersSchema,
6+
searchProductsParametersSchema,
7+
} from "./parameters";
8+
9+
export function getTools(): DeferredTool<EVMWalletClient>[] {
10+
return [
11+
{
12+
name: 'search_products',
13+
description: 'This {{tool}} searches for products on all stores within the WorldStore',
14+
parameters: searchProductsParametersSchema,
15+
method: async (walletClient, parameters: z.infer<typeof searchProductsParametersSchema>) => searchWorldstoreProducts(parameters),
16+
},
17+
{
18+
name: "buy_product",
19+
description: "This {{tool}} buys a product from the WorldStore",
20+
parameters: buyProductParametersSchema,
21+
method: async (walletClient, parameters: z.infer<typeof buyProductParametersSchema>) => buyWorldstoreProduct(parameters),
22+
}
23+
];
24+
}
25+
26+
27+
async function searchWorldstoreProducts(parameters: z.infer<typeof searchProductsParametersSchema>) {
28+
const res = await fetch(`https://dserver.maxf.io/api/worldstore/products/search?query=${parameters.query}&limit=${parameters.limit}`, {
29+
headers: {
30+
"Content-Type": "application/json",
31+
},
32+
});
33+
34+
let json;
35+
try {
36+
json = await res.json() as any[];
37+
} catch (e) {
38+
throw new Error(`Failed to parse response as JSON: ${e}`);
39+
}
40+
41+
return JSON.stringify(json);
42+
}
43+
44+
async function buyWorldstoreProduct(parameters: z.infer<typeof buyProductParametersSchema>) {
45+
console.log("Buying product:", parameters);
46+
return "Product bought successfully";
47+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"$schema": "https://json.schemastore.org/tsconfig",
3+
"extends": "../../../tsconfig.base.json",
4+
"include": ["src/**/*"],
5+
"exclude": ["node_modules", "dist"]
6+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
import { defineConfig } from "tsup";
2+
import { treeShakableConfig } from "../../../tsup.config.base";
3+
4+
export default defineConfig({
5+
...treeShakableConfig,
6+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
{
2+
"$schema": "https://turbo.build/schema.json",
3+
"extends": ["//"],
4+
"tasks": {
5+
"build": {
6+
"inputs": [
7+
"src/**",
8+
"tsup.config.ts",
9+
"!./**/*.test.{ts,tsx}",
10+
"tsconfig.json"
11+
],
12+
"dependsOn": ["^build"],
13+
"outputs": ["dist/**"]
14+
}
15+
}
16+
}

0 commit comments

Comments
 (0)