Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: metaplex bubblegum plugin #1532

Open
wants to merge 20 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions .env.example
Original file line number Diff line number Diff line change
Expand Up @@ -178,6 +178,10 @@ SOLANA_ADMIN_PRIVATE_KEY= # This wallet is used to verify NFTs
SOLANA_ADMIN_PUBLIC_KEY= # This wallet is used to verify NFTs
SOLANA_VERIFY_TOKEN= # Authentication token for calling the verification API

# Metaplex Bubblegum
MPL_BUBBLEGUM_PRIVATE_KEY= # Metaplex Bubblegum private key
MPL_BUBBLEGUM_RPC_URL= # Metaplex Bubblegum RPC URL (Needs support for DAS API)

# Fallback Wallet Configuration (deprecated)
WALLET_PRIVATE_KEY=
WALLET_PUBLIC_KEY=
Expand Down
5 changes: 5 additions & 0 deletions agent/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ import { nearPlugin } from "@elizaos/plugin-near";
import { nftGenerationPlugin } from "@elizaos/plugin-nft-generation";
import { createNodePlugin } from "@elizaos/plugin-node";
import { solanaPlugin } from "@elizaos/plugin-solana";
import { mplBubblegumPlugin } from "@elizaos/plugin-mpl-bubblegum";
import { suiPlugin } from "@elizaos/plugin-sui";
import { TEEMode, teePlugin } from "@elizaos/plugin-tee";
import { tonPlugin } from "@elizaos/plugin-ton";
Expand Down Expand Up @@ -530,6 +531,10 @@ export async function createAgent(
!getSecret(character, "WALLET_PUBLIC_KEY")?.startsWith("0x"))
? solanaPlugin
: null,
getSecret(character, "MPL_BUBBLEGUM_PRIVATE_KEY") &&
getSecret(character, "MPL_BUBBLEGUM_RPC_URL")
? mplBubblegumPlugin
: null,
(getSecret(character, "NEAR_ADDRESS") ||
getSecret(character, "NEAR_WALLET_PUBLIC_KEY")) &&
getSecret(character, "NEAR_WALLET_SECRET_KEY")
Expand Down
6 changes: 6 additions & 0 deletions packages/plugin-mpl-bubblegum/.npmignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
*

!dist/**
!package.json
!readme.md
!tsup.config.ts
3 changes: 3 additions & 0 deletions packages/plugin-mpl-bubblegum/eslint.config.mjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import eslintGlobalConfig from "../../eslint.config.mjs";

export default [...eslintGlobalConfig];
27 changes: 27 additions & 0 deletions packages/plugin-mpl-bubblegum/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
{
"name": "@elizaos/plugin-mpl-bubblegum",
"version": "0.1.5-alpha.5",
"main": "dist/index.js",
"type": "module",
"types": "dist/index.d.ts",
"dependencies": {
"@elizaos/core": "workspace:*",
"@elizaos/plugin-tee": "workspace:*",
"@metaplex-foundation/digital-asset-standard-api": "1.0.4",
"@metaplex-foundation/mpl-bubblegum": "4.2.1",
"@metaplex-foundation/umi-bundle-defaults": "0.9.2",
"@metaplex-foundation/umi-web3js-adapters": "0.9.2",
"@metaplex-foundation/umi": "0.9.2",
"@solana/web3.js": "1.95.8"
},
"devDependencies": {
"@types/node": "20.0.0",
"tsup": "8.3.5"
},
"scripts": {
"build": "tsup --format esm --dts",
"dev": "tsup --format esm --dts --watch",
"lint": "eslint --fix --cache .",
"test": "vitest run"
}
}
165 changes: 165 additions & 0 deletions packages/plugin-mpl-bubblegum/src/actions/transfer.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,165 @@
import {
ActionExample,
composeContext,
Content,
elizaLogger,
generateObject,
HandlerCallback,
IAgentRuntime,
Memory,
ModelClass,
State,
} from "@elizaos/core";
import { validateMplBubblegumConfig } from "../environment";
import { getWalletKey } from "../utils";
import { MplBubblegumProvider } from "../providers/bubblegumProvider";
import { fromWeb3JsKeypair } from "@metaplex-foundation/umi-web3js-adapters";
import { PublicKey } from "@metaplex-foundation/umi";

export interface TransferContent extends Content {
assetId: string;
newLeafOwner: string;
}

function isTransferContent(
_runtime: IAgentRuntime,
content: any
): content is TransferContent {
console.log("Content for transfer", content);
return (
typeof content.assetId === "string" &&
typeof content.newLeafOwner === "string"
);
}

const transferTemplate = `Respond with a JSON markdown block containing only the extracted values. Ensure that all extracted values, especially public key addresses, are complete and include every character as they appear in the messages. Use null for any values that cannot be determined.

Example response:
\`\`\`json
{
"assetId": "54dQ8cfHsW1YfKYpmdVZhWpb9iSi6Pac82Nf7sg3bVb",
"newLeafOwner": "G2FAbFQPFa5qKXCetoFZQEvF9BVvCKbvUZvodpVidnoY"
}
\`\`\`

{{recentMessages}}

Given the recent messages, extract the following information about the requested token transfer:
- Asset Id
- New Leaf Owner (new owner public key)

Respond with a JSON markdown block containing only the extracted values.`;

export default {
name: "SEND_COMPRESSED_NFT",
similes: ["TRANSFER_COMPRESSED_NFT", "SEND_CNFT", "TRANSFER_CNFT"],
validate: async (runtime: IAgentRuntime, message: Memory) => {
console.log("Validating config for user:", message.userId);
await validateMplBubblegumConfig(runtime);
return true;
},
description:
"Transfer compressed NFT (cNFT) from the agent wallet to another address",
handler: async (
runtime: IAgentRuntime,
message: Memory,
state: State,
_options: { [key: string]: unknown },
callback?: HandlerCallback
) => {
elizaLogger.log("Starting SEND_COMPRESSED_NFT handler...");

if (!state) {
state = (await runtime.composeState(message)) as State;
} else {
state = await runtime.updateRecentMessageState(state);
}

const transferContext = composeContext({
state,
template: transferTemplate,
});

const content = await generateObject({
runtime,
context: transferContext,
modelClass: ModelClass.LARGE,
});

if (!isTransferContent(runtime, content)) {
console.error("Invalid content for SEND_COMPRESSED_NFT action.");
if (callback) {
callback({
text: "Unable to process transfer request. Invalid content provided.",
content: { error: "Invalid transfer content" },
});
}
return false;
}

try {
const { keypair: agentKeypair } = await getWalletKey(runtime, true);

const RPC_URL = runtime.getSetting("MPL_BUBBLEGUM_RPC_URL");

const agentKeypairAdapter = fromWeb3JsKeypair(agentKeypair);

const mplBubblegumProvider = new MplBubblegumProvider(
agentKeypairAdapter,
RPC_URL
);

const { signature, result } = await mplBubblegumProvider.transfer(
content.assetId as PublicKey,
content.newLeafOwner as PublicKey
);

console.log("Transfer successful", signature);

if (callback) {
callback({
text: `Successfully transferred ${content.assetId} to ${content.newLeafOwner}\nTransaction: ${signature}`,
content: {
success: true,
signature,
recipient: content.newLeafOwner,
},
});
}

return true;
} catch (error) {
console.error("Error in transfer", error);
if (callback) {
callback({
text: `Error transferring tokens: ${error.message}`,
content: { error: error.message },
});
}
return false;
}
},
examples: [
[
{
user: "{{user1}}",
content: {
text: "Send BieefG47jAHCGZBxi2q87RDuHyGZyYC3vAzxpyu8pump to 9jW8FPr6BSSsemWPV22UUCzSqkVdTp6HTyPqeqyuBbCa",
},
},
{
user: "{{user2}}",
content: {
text: "I'll send BieefG47jAHCGZBxi2q87RDuHyGZyYC3vAzxpyu8pump now...",
action: "SEND_COMPRESSED_NFT",
},
},
{
user: "{{user2}}",
content: {
text: "Successfully sent BieefG47jAHCGZBxi2q87RDuHyGZyYC3vAzxpyu8pump to 9jW8FPr6BSSsemWPV22UUCzSqkVdTp6HTyPqeqyuBbCa\nTransaction: 5KtPn3DXXzHkb7VAVHZGwXJQqww39ASnrf7YkyJoF2qAGEpBEEGvRHLnnTG8ZVwKqNHMqSckWVGnsQAgfH5pbxEb",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I know nothing about metaplex bubblegum but assuming our system is full of similar plugins are these examples going to be specific enough?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TL;DR
Bubblegum is Solana's protocol for compressed NFTs, allowing for high-scale, low-cost creation of NFTs by storing "non-essential" data off-chain while maintaining proofs on-chain.

It requires a custom implementation separate from the Solana SPL Token standard.

Yes, I agree it could lead to issues when used alongside similar plugins.

},
},
],
] as ActionExample[][],
};
40 changes: 40 additions & 0 deletions packages/plugin-mpl-bubblegum/src/environment.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import { IAgentRuntime } from "@elizaos/core";
import { z } from "zod";

export const mplBubblegumEnvSchema = z.object({
MPL_BUBBLEGUM_PRIVATE_KEY: z
.string()
.min(1, "Private key for Bubblegum is required."),
MPL_BUBBLEGUM_RPC_URL: z
.string()
.min(1, "Bubblegum RPC URL with DAS API support is required."),
});

export type MplBubblegumConfig = z.infer<typeof mplBubblegumEnvSchema>;

export async function validateMplBubblegumConfig(
runtime: IAgentRuntime
): Promise<MplBubblegumConfig> {
try {
const config = {
MPL_BUBBLEGUM_PRIVATE_KEY:
runtime.getSetting("MPL_BUBBLEGUM_PRIVATE_KEY") ||
process.env.MPL_BUBBLEGUM_PRIVATE_KEY,
MPL_BUBBLEGUM_RPC_URL:
runtime.getSetting("MPL_BUBBLEGUM_RPC_URL") ||
process.env.MPL_BUBBLEGUM_RPC_URL,
};

return mplBubblegumEnvSchema.parse(config);
} catch (error) {
if (error instanceof z.ZodError) {
const errorMessages = error.errors
.map((err) => `${err.path.join(".")}: ${err.message}`)
.join("\n");
throw new Error(
`Mpl Bubblegum configuration validation failed:\n${errorMessages}`
);
}
throw error;
}
}
12 changes: 12 additions & 0 deletions packages/plugin-mpl-bubblegum/src/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import { Plugin } from "@elizaos/core";
import transfer from "./actions/transfer";

export const mplBubblegumPlugin: Plugin = {
name: "mpl-bubblegum",
description: "Bubblegum Plugin for Eliza",
actions: [transfer],
providers: [],
evaluators: [],
services: [],
clients: [],
};
Loading
Loading