Skip to content

Commit e05f6d9

Browse files
committedJan 5, 2025
Add Stargaze plugin
1 parent 62abe4c commit e05f6d9

14 files changed

+908
-0
lines changed
 

‎.env.example

+3
Original file line numberDiff line numberDiff line change
@@ -373,3 +373,6 @@ FUEL_WALLET_PRIVATE_KEY=
373373
# Tokenizer Settings
374374
TOKENIZER_MODEL= # Specify the tokenizer model to be used.
375375
TOKENIZER_TYPE= # Options: tiktoken (for OpenAI models) or auto (AutoTokenizer from Hugging Face for non-OpenAI models). Default: tiktoken.
376+
377+
# Stargaze NFT marketplace from Cosmos (You can use https://graphql.mainnet.stargaze-apis.com/graphql)
378+
STARGAZE_ENDPOINT=

‎agent/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@
4949
"@elizaos/plugin-node": "workspace:*",
5050
"@elizaos/plugin-solana": "workspace:*",
5151
"@elizaos/plugin-starknet": "workspace:*",
52+
"@elizaos/plugin-stargaze": "workspace:*",
5253
"@elizaos/plugin-ton": "workspace:*",
5354
"@elizaos/plugin-sui": "workspace:*",
5455
"@elizaos/plugin-tee": "workspace:*",

‎agent/src/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,7 @@ import { abstractPlugin } from "@elizaos/plugin-abstract";
6464
import { avalanchePlugin } from "@elizaos/plugin-avalanche";
6565
import { webSearchPlugin } from "@elizaos/plugin-web-search";
6666
import { echoChamberPlugin } from "@elizaos/plugin-echochambers";
67+
import { stargazePlugin } from "@elizaos/plugin-stargaze";
6768
import Database from "better-sqlite3";
6869
import fs from "fs";
6970
import path from "path";
@@ -609,6 +610,7 @@ export async function createAgent(
609610
getSecret(character, "ECHOCHAMBERS_API_KEY")
610611
? echoChamberPlugin
611612
: null,
613+
getSecret(character, "STARGAZE_ENDPOINT") ? stargazePlugin : null,
612614
].filter(Boolean),
613615
providers: [],
614616
actions: [],

‎packages/plugin-stargaze/README.md

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
# Plugin Stargaze
2+
3+
A plugin for fetching NFT data from the Stargaze API.
4+
5+
## Overview
6+
7+
The Plugin Stargaze provides a simple interface to get NFT data from Stargaze collections. It integrates with Stargaze's GraphQL API to fetch the latest NFTs from collections.
8+
9+
## Installation
10+
11+
```bash
12+
pnpm add @elizaos/plugin-stargaze
13+
```
14+
15+
## Configuration
16+
17+
Set up your environment with the required Stargaze API endpoint, currently Stargaze offers https://graphql.mainnet.stargaze-apis.com/graphql publically.
18+
19+
| Variable Name | Description |
20+
| ------------- | ----------- |
21+
| `STARGAZE_ENDPOINT` | Stargaze GraphQL API endpoint |
22+
23+
## Usage
24+
25+
```typescript
26+
import { stargazePlugin } from "@elizaos/plugin-stargaze";
27+
28+
// Initialize the plugin
29+
const plugin = stargazePlugin;
30+
31+
// The plugin provides the GET_LATEST_NFT action which can be used to fetch NFTs
32+
// Example: "Show me the latest NFT from ammelia collection"
33+
```
34+
35+
## License
36+
37+
MIT

‎packages/plugin-stargaze/package.json

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
{
2+
"name": "@elizaos/plugin-stargaze",
3+
"version": "0.1.0",
4+
"main": "dist/index.js",
5+
"type": "module",
6+
"types": "dist/index.d.ts",
7+
"dependencies": {
8+
"@elizaos/core": "workspace:*",
9+
"axios": "^1.6.7",
10+
"tsup": "^8.3.5",
11+
"zod": "^3.22.4"
12+
},
13+
"scripts": {
14+
"build": "tsup --format esm --dts",
15+
"dev": "tsup --format esm --dts --watch",
16+
"test": "vitest run"
17+
}
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,219 @@
1+
import {
2+
ActionExample,
3+
composeContext,
4+
Content,
5+
elizaLogger,
6+
generateObjectDeprecated,
7+
HandlerCallback,
8+
IAgentRuntime,
9+
Memory,
10+
ModelClass,
11+
State,
12+
type Action,
13+
} from "@elizaos/core";
14+
import axios from "axios";
15+
import { validateStargazeConfig } from "../environment";
16+
import { debugLog } from "../utils/debug";
17+
18+
export interface GetCollectionStatsContent extends Content {
19+
collectionAddr: string;
20+
}
21+
22+
const COLLECTION_STATS_QUERY = `
23+
query CollectionStats($collectionAddr: String!) {
24+
collection(address: $collectionAddr) {
25+
contractAddress
26+
name
27+
stats {
28+
numOwners
29+
bestOffer
30+
volumeTotal
31+
volume24Hour
32+
salesCountTotal
33+
tokensMintedPercent
34+
uniqueOwnerPercent
35+
change24HourPercent
36+
marketCap
37+
mintCount24hour
38+
mintVolume24hour
39+
volumeUsdTotal
40+
volumeUsd24hour
41+
}
42+
}
43+
}`;
44+
45+
// Add template for content generation
46+
const getCollectionStatsTemplate = `Given the message, extract the collection address for fetching Stargaze stats.
47+
48+
Format the response as a JSON object with this field:
49+
- collectionAddr: the collection address or name (required)
50+
51+
Example response for "Show me stats for ammelia collection":
52+
\`\`\`json
53+
{
54+
"collectionAddr": "ammelia"
55+
}
56+
\`\`\`
57+
58+
Example response for "Show me stats for stars10n0m58ztlr9wvwkgjuek2m2k0dn5pgrhfw9eahg9p8e5qtvn964suc995j collection":
59+
\`\`\`json
60+
{
61+
"collectionAddr": "stars10n0m58ztlr9wvwkgjuek2m2k0dn5pgrhfw9eahg9p8e5qtvn964suc995j"
62+
}
63+
\`\`\`
64+
65+
{{recentMessages}}
66+
67+
Extract the collection address from the above messages and respond with the appropriate JSON.`;
68+
69+
export default {
70+
name: "GET_COLLECTION_STATS",
71+
similes: ["CHECK_COLLECTION_STATS", "COLLECTION_INFO"],
72+
validate: async (runtime: IAgentRuntime, message: Memory) => {
73+
elizaLogger.log("🔄 Validating Stargaze configuration...");
74+
try {
75+
const config = await validateStargazeConfig(runtime);
76+
debugLog.validation(config);
77+
return true;
78+
} catch (error) {
79+
debugLog.error(error);
80+
return false;
81+
}
82+
},
83+
description: "Get detailed statistics for a Stargaze collection",
84+
handler: async (
85+
runtime: IAgentRuntime,
86+
message: Memory,
87+
state: State,
88+
_options: { [key: string]: unknown },
89+
callback?: HandlerCallback
90+
): Promise<boolean> => {
91+
elizaLogger.log("🚀 Starting Stargaze GET_COLLECTION_STATS handler...");
92+
93+
if (!state) {
94+
elizaLogger.log("Creating new state...");
95+
state = (await runtime.composeState(message)) as State;
96+
} else {
97+
elizaLogger.log("Updating existing state...");
98+
state = await runtime.updateRecentMessageState(state);
99+
}
100+
101+
try {
102+
elizaLogger.log("Composing collection stats context...");
103+
const statsContext = composeContext({
104+
state,
105+
template: getCollectionStatsTemplate,
106+
});
107+
108+
elizaLogger.log("Generating content from context...");
109+
const content = (await generateObjectDeprecated({
110+
runtime,
111+
context: statsContext,
112+
modelClass: ModelClass.LARGE,
113+
})) as unknown as GetCollectionStatsContent;
114+
115+
if (!content || !content.collectionAddr) {
116+
throw new Error("Invalid or missing collection address in parsed content");
117+
}
118+
119+
debugLog.validation(content);
120+
121+
const config = await validateStargazeConfig(runtime);
122+
123+
const requestData = {
124+
query: COLLECTION_STATS_QUERY,
125+
variables: {
126+
collectionAddr: content.collectionAddr,
127+
},
128+
};
129+
130+
debugLog.request('POST', config.STARGAZE_ENDPOINT, requestData);
131+
132+
const response = await axios.post(
133+
config.STARGAZE_ENDPOINT,
134+
requestData,
135+
{
136+
headers: {
137+
'Content-Type': 'application/json',
138+
}
139+
}
140+
);
141+
142+
debugLog.response(response);
143+
144+
const stats = response.data?.data?.collection?.stats;
145+
const name = response.data?.data?.collection?.name;
146+
147+
if (!stats) {
148+
throw new Error("No stats found for collection");
149+
}
150+
151+
// Format numerical values
152+
const formatValue = (value: number) =>
153+
value ? Number(value).toLocaleString(undefined, {
154+
maximumFractionDigits: 2
155+
}) : '0';
156+
157+
// Format percentage values
158+
const formatPercent = (value: number) =>
159+
value ? `${Number(value).toFixed(2)}%` : '0%';
160+
161+
if (callback) {
162+
const message = {
163+
text: `Collection Stats for ${name} (${content.collectionAddr}):
164+
- Total Volume: ${formatValue(stats.volumeUsdTotal)} USD
165+
- 24h Volume: ${formatValue(stats.volumeUsd24hour)} USD
166+
- Total Sales: ${formatValue(stats.salesCountTotal)}
167+
- Unique Owners: ${formatValue(stats.numOwners)}
168+
- Owner Ratio: ${formatPercent(stats.uniqueOwnerPercent)}
169+
- Minted: ${formatPercent(stats.tokensMintedPercent)}
170+
- 24h Change: ${formatPercent(stats.change24HourPercent)}
171+
- 24h Mints: ${formatValue(stats.mintCount24hour)}
172+
- Market Cap: ${formatValue(stats.marketCap)} USD`,
173+
content: stats,
174+
};
175+
elizaLogger.log("✅ Sending callback with collection stats:", message);
176+
callback(message);
177+
}
178+
179+
return true;
180+
} catch (error) {
181+
debugLog.error(error);
182+
if (callback) {
183+
callback({
184+
text: `Error fetching collection stats: ${error}`,
185+
content: { error: error },
186+
});
187+
}
188+
return false;
189+
}
190+
},
191+
examples: [[
192+
{
193+
user: "{{user1}}",
194+
content: {
195+
text: "Show me stats for collection ammelia",
196+
},
197+
},
198+
{
199+
user: "{{agent}}",
200+
content: {
201+
text: "I'll check the stats for collection ammelia...",
202+
action: "GET_COLLECTION_STATS",
203+
},
204+
},
205+
{
206+
user: "{{user1}}",
207+
content: {
208+
text: "Show me stats for collection {collection address}",
209+
},
210+
},
211+
{
212+
user: "{{agent}}",
213+
content: {
214+
text: "I'll check the stats for collection {collection address}...",
215+
action: "GET_COLLECTION_STATS",
216+
},
217+
},
218+
]],
219+
} as Action;

0 commit comments

Comments
 (0)