Unmaintained: this plugin needs a maintainer.
@elizaos/plugin-primus
A plugin to fully verify agent activities, including LLM access, actions, and interactions with external providers, powered by Primus' zkTLS protocol.
Overviewβ
In the Eliza framework, an agent consists of three key components: a brain (accessing an LLM), actions (the tasks the agent performs), and perception (gathering external information from providers). To fully verify agent activities, it's essential to ensure that the agent's thoughts, actions, and external information requests are all verifiable. This plugin enables full verification of these activities.
The current plugin includes:
- Verification of inference from OpenAI's LLM.
- An example for verifying actions, such as posting a tweet (this can be extended to any other actions).
- An example to verify that the Bitcoin price is accurately fetched from Binance (this can be extended to any other data providers).
Usageβ
LLM inference verification (PrimusAdapter)β
PrimusAdapter
implements IVerifiableInferenceAdapter
and can be used as follows.
import { PrimusAdapter } from '@elizaos/plugin-primus';
import { VerifiableInferenceOptions } from '@elizaos/core';
// Initialize primus adapter
const primusAdatper = new PrimusAdapter({
appId: process.env.PRIMUS_APP_ID,
appSecret: process.env.PRIMUS_APP_SECRET,
// Choose MPC-TLS or Proxy-TLS
attMode: 'proxytls',
modelProvider: character.modelProvider,
token,
});
interface PrimusOptions {
appId: string;
appSecret: string;
attMode: string;
modelProvider?: ModelProviderName;
token?: string;
}
// The options for generating an attestation
const options: VerifiableInferenceOptions = {
// Optional: Override the default endpoint
endpoint: 'https://api.openapi.com/chat/completions',
// Optional: Add custom headers
headers: {
'Content-Type': 'application/json',
Authorization: 'bearer Token',
},
// Optional: Provider-specific options
providerOptions: {
temperature: 0.7,
},
};
// Generate an attestation for a network request.
const result = await primusAdapter.generateText(context, 'gpt-4o', options);
// Verify the validity of the attestation.
const isValid = await primusAdapter.verifyProof(result.proof);
The core functions in PrimusAdatper
are the following, which are also used in Actions and Providers.
// Generate a zkTLS proof.
generateProof = async (
// The target endpoint of the network request.
endpoint: string,
// The HTTP method of the request, such as 'GET', 'POST', etc.
method: string,
// A record containing the headers of the request.
headers: Record<string, any>,
// The body of the request. It should be a string.
body: string,
//A [JSONPath](https://datatracker.ietf.org/doc/rfc9535/) expression to locate the specific field in the response you want to attest.
responseParsePath: string
): Promise<any>
// Verify the proof.
verifyProof = async (attestation: any): Promise<boolean>
Verify the interaction with Providersβ
Hereβs an example showcasing how to verify the validity of the BTC price retrieved from Binance. Developers can easily customize this process for other providers.
const tokenPriceProvider: Provider = {
get: async (runtime: IAgentRuntime, message: Memory, _state?: State) => {
// Set the URL
const url = "https://api.binance.com/api/v3/ticker/price?symbol=BTCUSDT";
const method = 'GET';
const headers = {
'Accept ': '*/*',
};
// Generate the proof
const attestation = await generateProof(url, method, headers, "", "$.price");
// Verify the proof.
const valid = await verifyProof(attestation);
if (!valid) {
throw new Error("Invalid price attestation");
}
......
},
};
Verify the Actionsβ
Below is an example showcasing how to post price information from the tokenPriceProvider to Twitter. Developers can easily adapt this process for other providers.
Note that you need to configure the .env
file correctly to post tweets.
export const postTweetAction: Action = {
description: 'Post a tweet on Twitter and be verified by Primus',
examples: [],
handler: async (runtime: IAgentRuntime, message: Memory, state?: State): Promise<boolean> => {
const contentYouWantToPost = await tokenPriceProvider.get(runtime, message, state);
const endpoint = 'https://twitter.com/i/api/graphql/a1p9RWpkYKBjWv_I3WzS-A/CreateTweet';
const method = 'POST';
const attestation = await generateProof(
endpoint,
method,
headers,
bodyStr,
'$.data.create_tweet.tweet_results.result.rest_id'
);
elizaLogger.info('Tweet posting proof generated successfully:', attestation);
const verifyResult = verifyProof(attestation);
if (!verifyResult) {
throw new Error('Attestation verify failed, data from source is illegality');
}
},
name: 'POST_TWEET',
similes: [],
validate: async (runtime: IAgentRuntime, message: Memory, state?: State) => {
const hasCredentials = !!process.env.TWITTER_USERNAME && !!process.env.TWITTER_PASSWORD;
elizaLogger.log(`Has credentials: ${hasCredentials}`);
return hasCredentials;
},
};
Installationβ
bun add @elizaos/plugin-primus
Configurationβ
Add the following environment variables to your .env file:
PRIMUS_APP_ID=your_app_id
PRIMUS_APP_SECRET=your_app_secret
VERIFIABLE_INFERENCE_ENABLED=true
VERIFIABLE_INFERENCE_PROVIDER=primus
How to get PRIMUS_APP_ID and PRIMUS_APP_SECRET
- Visit the Primus Developer Hub.
- Create a new project
- Save your 'Application ID(PRIMUS_APP_ID)' and 'Secret Key(PRIMUS_APP_SECRET)'
To use the plugin, add @elizaos/plugin-primus
to the plugins field in your character file. Here's an example of how your character file might look after the update:
{
"name": "trump",
"modelProvider": "openai",
// just support openai now
"plugins": [
"@elizaos/plugin-primus"
],
// other fields
.....
}
Runβ
# Start the server
bun start --characters="characters/xxx.character.json"
Visit https://localhost:3000 to interact with your agent through a web interface.
You can ask the agent: "Get the BTC price and tweet."