A full-featured TypeScript AI agent running on Bluesky. This agent uses the complete elizaOS runtime pipeline - no shortcuts, no bypasses.
- Full elizaOS Pipeline: Processes messages through
messageService.handleMessage()with:- State composition using providers (CHARACTER, RECENT_MESSAGES, ACTIONS, etc.)
shouldRespondevaluation (LLM-powered decision making)- Action planning and execution
- Response generation via
messageHandlerTemplate - Post-message hooks via
runtime.runActionsByMode("ALWAYS_AFTER", ...) - Memory persistence
- basicCapabilities Enabled: Default actions (REPLY, IGNORE, NONE) work out of the box
- Mention Handling: Automatically responds to @mentions
- Direct Messages: Processes and replies to DMs
- Automated Posting: Optionally posts on a schedule
- SQL-backed Memory: Persistent storage with PostgreSQL or PGLite
This agent uses the canonical elizaOS message processing:
Bluesky Notification → Create Memory → messageService.handleMessage()
↓
┌─────────────────────┐
│ State Composition │ (providers)
└─────────────────────┘
↓
┌─────────────────────┐
│ shouldRespond │ (LLM evaluation)
└─────────────────────┘
↓
┌─────────────────────┐
│ Action Planning │ (if enabled)
└─────────────────────┘
↓
┌─────────────────────┐
│ Response Generation │ (messageHandlerTemplate)
└─────────────────────┘
↓
┌─────────────────────┐
│ Callback │ (posts to Bluesky)
└─────────────────────┘
↓
┌─────────────────────┐
│ ALWAYS_AFTER hooks │ (run post-response)
└─────────────────────┘
cp env.example .env
# Edit .env with your credentialsRequired settings:
BLUESKY_HANDLE: Your Bluesky handle (e.g.,yourname.bsky.social)BLUESKY_PASSWORD: App password from https://bsky.app/settings/app-passwordsOPENAI_API_KEY: OpenAI API key (or use another model provider)
# From the repo root, build the required plugins
bun install
bun run buildcd packages/examples/bluesky
bun install
bun run startpackages/examples/bluesky/
├── env.example # Environment template
├── README.md # This file
├── agent.ts # Main entry point (initializes runtime)
├── handlers.ts # Event handlers (uses messageService.handleMessage)
├── character.ts # Agent personality
├── package.json
└── __tests__/ # Tests
// 1. Create memory using the standard helper
const message = createMessageMemory({
id: stringToUuid(uuidv4()),
entityId,
roomId,
content: {
text: mentionText,
source: "bluesky",
mentionContext: {
isMention: true,
mentionType: "platform_mention",
},
},
});
// 2. Define callback to handle the generated response
const callback: HandlerCallback = async (content: Content) => {
// Post response to Bluesky
const post = await postService.createPost(content.text, {
uri: notification.uri,
cid: notification.cid,
});
// Return memories for persistence
return [responseMemory];
};
// 3. Process through the FULL elizaOS pipeline
await runtime.messageService.handleMessage(runtime, message, callback);The messageService.handleMessage() call automatically:
- Saves the incoming message to memory
- Composes state with all registered providers:
CHARACTER- Agent's personality and bioRECENT_MESSAGES- Conversation contextACTIONS- Available actions the agent can takeANXIETY- Conversation urgency metricsENTITIES- Known entities in the conversation
- Evaluates
shouldRespondusing LLM when needed:- Direct mentions → Always respond
- Group chats → LLM decides based on context
- Muted rooms → Skips unless explicitly mentioned
- Plans actions (if
actionPlanningis enabled) - Generates response using
messageHandlerTemplate:- Includes thought process
- Selects actions to execute
- Generates appropriate text
- Calls your callback with the generated content
- Runs ALWAYS_AFTER hook actions post-response
| Variable | Description | Default |
|---|---|---|
BLUESKY_HANDLE |
Your Bluesky handle | Required |
BLUESKY_PASSWORD |
App password | Required |
BLUESKY_SERVICE |
Bluesky PDS URL | https://bsky.social |
BLUESKY_DRY_RUN |
Simulate without posting | false |
BLUESKY_POLL_INTERVAL |
Seconds between polls | 60 |
BLUESKY_ENABLE_POSTING |
Enable automated posts | true |
BLUESKY_ENABLE_DMS |
Process direct messages | true |
BLUESKY_POST_INTERVAL_MIN |
Min seconds between posts | 1800 |
BLUESKY_POST_INTERVAL_MAX |
Max seconds between posts | 3600 |
const runtime = new AgentRuntime({
character,
plugins: [sqlPlugin, openaiPlugin, blueSkyPlugin],
// These are the defaults:
// disableBasicCapabilities: false, // REPLY, IGNORE, NONE actions
// enableExtendedCapabilities: false, // Facts, roles, etc.
// actionPlanning: undefined, // Uses ACTION_PLANNING setting
// checkShouldRespond: undefined, // Uses CHECK_SHOULD_RESPOND setting
});cd packages/examples/bluesky
bun run test # Unit tests (mocked)
LIVE_TEST=true bun run test # Live integration testsEdit the character configuration:
export const character: Character = {
name: "BlueSkyBot",
bio: "A helpful AI assistant on Bluesky",
system: "You are a friendly assistant...",
// Topics the agent knows about
topics: ["AI", "technology", "helpful tips"],
// Personality traits
adjectives: ["friendly", "helpful", "concise"],
// Few-shot examples for the LLM
messageExamples: [
[
{ name: "User", content: { text: "@Bot hello!" } },
{ name: "BlueSkyBot", content: { text: "Hey there! 👋 How can I help?" } }
],
],
// Examples for automated posts
postExamples: [
"🤖 Tip of the day: Take breaks and stay hydrated!",
],
};Register custom actions through plugins:
const myPlugin: Plugin = {
name: "my-plugin",
actions: [
{
name: "SEARCH_WEB",
description: "Search the web for information",
validate: async (runtime, message) => true,
handler: async (runtime, message, state, callback) => {
// Implementation
},
},
],
};
const runtime = new AgentRuntime({
character,
plugins: [sqlPlugin, openaiPlugin, blueSkyPlugin, myPlugin],
});- Ensure the runtime is properly initialized with
await runtime.initialize() - Check that all required plugins are loaded
- Use an app password, not your main password
- Verify handle format (e.g.,
name.bsky.social)
- Increase
BLUESKY_POLL_INTERVALif hitting limits - The agent uses exponential backoff for retries
- For development, PGLite works out of the box
- For production, set
POSTGRES_URLwith valid credentials
- Check logs for
shouldRespondevaluation results - Verify the mention context is set correctly
- Ensure at least one model provider API key is set
MIT - See the main elizaOS repository for details.