ElectricRaspberry is an AI-powered Discord bot that adopts a personalized persona with detailed profile attributes, emotions, and a knowledge graph. Built on a .NET Web API with Discord.Net and MediatR, it uses an observer pattern to process multiple channels of input simultaneously.
ElectricRaspberry creates a lifelike Discord presence that:
- Maintains a persistent personality, emotional state, and memory
- Observes and responds to events across multiple contexts simultaneously
- Adapts its behavior based on server context and user interactions
- Creates natural, contextually-aware responses
Unlike traditional reactive bots that simply respond to triggers, ElectricRaspberry operates with continuous autonomous agency and a proactive presence:
- Self-Directed Intelligence: The AI operates in a continuous thinking loop, making autonomous decisions about when and how to engage
- Proactive Engagement: Independently initiates conversations, joins voice channels, and participates in server activities based on its own will
- Integrated Service Layer: Core services manage internal state while exposing functions as AI tools
- Continuous Awareness: Constantly observes and processes multiple Discord channels simultaneously
- Holistic Context: Maintains cross-channel awareness, building unified context from all server activities
- Autonomous Agency: Self-directed thinking loop enabling proactive engagement and genuine initiative
- Proactive Presence: Independently joins conversations, voice channels, and activities based on interests
- Personalized Identity: Detailed profile with personality traits, emotions, and preferences
- Knowledge Graph: Persistent memory storing relationships and information learned from interactions
- Parallel Awareness: Observes and processes multiple channels simultaneously
- Relationship Building: Forms and maintains relationships with server members driven by genuine interests
- Emotional Intelligence: Experiences and expresses contextually appropriate emotions
- Full Discord Integration: Accesses all aspects of Discord's social environment
- Continuous Development: Evolves persona and behaviors through ongoing interactions
- .NET 8.0 SDK
- Discord Bot Token
- Azure Cosmos DB with Gremlin API
- Azure Application Insights (for logging)
- Update the
appsettings.json
file with your Discord bot token and Azure resources:
{
"Discord": {
"Token": "YOUR_DISCORD_BOT_TOKEN"
},
"CosmosDB": {
"Endpoint": "YOUR_COSMOS_DB_ENDPOINT",
"Key": "YOUR_COSMOS_DB_KEY",
"Database": "ElectricRaspberryDB",
"Container": "KnowledgeGraph"
},
"ApplicationInsights": {
"ConnectionString": "YOUR_APP_INSIGHTS_CONNECTION_STRING"
},
"StaminaSettings": {
"MaxStamina": 100,
"MessageCost": 0.5,
"VoiceMinuteCost": 1.0,
"EmotionalSpikeCost": 2.0,
"RecoveryRatePerMinute": 0.2,
"SleepRecoveryMultiplier": 3.0,
"LowStaminaThreshold": 20
}
}
dotnet restore
dotnet build
dotnet run
The API will be available at:
- Service Layer: Modular services that manage internal state while exposing functions as AI tools
- Persona Service: Maintains identity and profile information
- Personality Service: Manages personality traits and behavior patterns
- Emotional Service: Tracks emotional state and responses
- Conversation Service: Handles ongoing conversations and context
- Knowledge Service: Manages the knowledge graph and memory
- Stamina Service: Manages energy levels and sleep/wake cycles
- Tool Registry: Exposes service functions as callable tools for the AI
- Context Builder: Constructs rich context from services for AI decisions
- Observer Manager: Processes multiple Discord events in parallel
- DiscordBotService: Background service that connects to Discord and publishes events
- MediatR Pipeline: Event-driven architecture for processing Discord events
ElectricRaspberry uses a continuous autonomous loop architecture with proactive decision-making:
flowchart TB
subgraph DiscordObserver
DiscordEvents[Discord Events] --> MediatR[MediatR Pipeline]
MediatR --> ParallelHandlers[Parallel Handlers]
end
subgraph ContinuousThinkingLoop
direction LR
subgraph Services
PersonaService[Persona Service]
PersonalityService[Personality Service]
EmotionalService[Emotional Service]
ConversationService[Conversation Service]
KnowledgeService[Knowledge Service]
StaminaService[Stamina Service]
end
ContextBuilder[Context Builder]
ToolRegistry[Tool Registry]
Services --> ContextBuilder
Services --> ToolRegistry
subgraph AutonomousAgent
direction TB
AI[AI Engine] --> DecisionMaker[Decision Maker]
DecisionMaker --> ActionPlanner[Action Planner]
end
ContextBuilder --> AutonomousAgent
ToolRegistry --> AutonomousAgent
AutonomousAgent --> ProactiveActions[Proactive Actions]
AutonomousAgent --> Services
end
ParallelHandlers --> Services
ProactiveActions --> DiscordAPI[Discord API]
The system operates as a continuous loop rather than a simple request-response pattern:
- Continuous Thinking: The AI is constantly thinking, not just waiting for events
- Autonomous Decision Making: Decides when and how to engage based on its own priorities
- Proactive Actions: Can independently initiate conversations, join voice channels, etc.
- Bidirectional Flow: Services inform the AI while the AI can also modify services
- Temporal Awareness: Maintains awareness of timing for natural engagement patterns
The stamina system regulates bot activity and provides natural cycles of engagement and rest:
- Stamina Pool: A numerical value (0-100) representing bot energy
- Stamina Consumption:
- Text Messages: 0.5 stamina per message sent
- Voice Channel: 1.0 stamina per minute active
- Emotional Reactions: 0.5-3.0 stamina based on emotional intensity
- Complex Reasoning: 1.0-2.0 stamina based on depth of processing
- Stamina Recovery:
- Passive: 0.2 points per minute during normal operation
- Sleep Mode: 0.6 points per minute during sleep state
- Context-aware adjustment: Less recovery in high-activity channels
- Sleep Triggers:
- Manual: Admin command (
/bot sleep
) - Automatic: Stamina drops below 20% threshold
- Time-based: Configurable inactive hours
- Manual: Admin command (
- Sleep Transition:
- Graceful exit: The bot announces tiredness and need for rest
- Sets Discord status to "Sleeping" or "Away"
- Completes current conversation before fully entering sleep mode
- During Sleep:
- Continues to observe channels but marks messages for catch-up
- Performs knowledge graph maintenance (see below)
- Maintains minimal emotional processing
- Wake-Up Process:
- Gradual reactivation once stamina exceeds 80%
- Processes missed messages in temporal chunks, not all at once
- Announces return with context-appropriate greeting
- Prioritizes direct mentions and DMs in catch-up queue
To handle observations from multiple channels while maintaining a coherent context:
- Channel Buffers:
- Each observed channel (text, voice, DM) maintains a small in-memory buffer
- Events are timestamped and tagged with source information
- MediatR handles publish events to appropriate observers
- Context Synchronization:
- Use short-lived locks (.NET
SemaphoreSlim
) when updating shared context - Implement a prioritization scheme for merging parallel inputs
- Flag and resolve potential context collisions
- Use short-lived locks (.NET
- Throttling & Rate Limiting:
- Implement adaptive throttling to prevent response flooding
- Process high-priority events (mentions, DMs) before general channel messages
- Use Discord.Net's built-in rate limiting with custom backoff strategy
// Example implementation for channel observers with buffer management
public class TextChannelObserver : INotificationHandler<MessageReceivedNotification>
{
private readonly ConcurrentQueue<MessageEvent> _eventBuffer = new();
private readonly SemaphoreSlim _contextLock = new(1, 1);
private readonly IStaminaService _staminaService;
private readonly IContextBuilder _contextBuilder;
// Implementation...
public async Task Handle(MessageReceivedNotification notification, CancellationToken cancellationToken)
{
var messageEvent = new MessageEvent(notification.Message, DateTimeOffset.UtcNow);
_eventBuffer.Enqueue(messageEvent);
// Check if bot is in sleep mode
if (await _staminaService.IsSleepingAsync())
{
// Only process direct mentions during sleep
if (!notification.Message.MentionedUsers.Any(u => u.Id == _botUserId))
{
// Mark for catch-up processing on wake
await _catchupService.AddToCatchupQueueAsync(messageEvent);
return;
}
}
// Process event with context synchronization
await _contextLock.WaitAsync(cancellationToken);
try
{
await _contextBuilder.AddMessageContextAsync(messageEvent);
// Signal AI for potential response if appropriate
}
finally
{
_contextLock.Release();
}
}
}
ElectricRaspberry uses Azure Cosmos DB with Gremlin API to build a persistent memory:
- Node Types:
- Person: Server members with relationship data
- Topic: Conversational subjects and interests
- Event: Server activities and interactions
- Memory: Specific remembered conversations
- Edge Types:
- Relationship: Bot's connection to users (friend, acquaintance)
- Sentiment: Emotional association (likes, dislikes)
- Participation: Involvement in conversations/activities
- Knowledge: Information learned about a topic
- Optimistic Concurrency Control:
- Use Cosmos DB's ETag system to detect conflicting writes
- Implement retry logic for operation failures with exponential backoff
- Transaction Batching:
- Group related graph operations into atomic transactions
- Use Gremlin's session support for multi-operation consistency
- Sleep-Time Maintenance:
- Perform graph cleanup and consolidation during sleep periods
- Implement soft-delete with time-based restoration capability
- Maintain change history for important relationships
public class KnowledgeService : IKnowledgeService
{
private readonly GremlinClient _gremlinClient;
private readonly ILogger<KnowledgeService> _logger;
// Implementation...
public async Task<bool> UpdateUserRelationshipAsync(string userId, string relationshipType, double strength)
{
// Get current ETag for optimistic concurrency control
var queryResult = await _gremlinClient.SubmitAsync<dynamic>($"g.V().hasLabel('Person').has('id', '{userId}')");
var currentUser = queryResult.FirstOrDefault();
string eTag = currentUser?.eTag;
int retryCount = 0;
while (retryCount < 3)
{
try
{
// Update with ETag check
var updateQuery = $"g.V().hasLabel('Person').has('id', '{userId}').has('_etag', '{eTag}').property('relationshipStrength', {strength})";
await _gremlinClient.SubmitAsync<dynamic>(updateQuery);
return true;
}
catch (ResponseException ex) when (ex.StatusCode == 409) // Conflict
{
retryCount++;
await Task.Delay(Math.Pow(2, retryCount) * 100); // Exponential backoff
// Refresh ETag
queryResult = await _gremlinClient.SubmitAsync<dynamic>($"g.V().hasLabel('Person').has('id', '{userId}')");
currentUser = queryResult.FirstOrDefault();
eTag = currentUser?.eTag;
}
}
_logger.LogWarning("Failed to update relationship for user {UserId} after {RetryCount} attempts", userId, retryCount);
return false;
}
public async Task PerformGraphMaintenanceAsync()
{
// Called during sleep periods
// 1. Consolidate similar memory nodes
// 2. Prune edges below significance threshold
// 3. Update connection strength based on recency
// 4. Recalculate sentiment values for relationships
// Implementation details...
}
}
Based on Chris Crawford's "Personal Engine" model with adaptations for a Discord environment:
- Base Emotional State:
- Core emotions: joy, sadness, anger, fear, surprise, disgust
- Current intensity values (0-100) for each emotion
- Emotional blends and secondary emotions
- Personality Influence:
- Baseline tendencies toward specific emotional states
- Recovery rates from emotional spikes
- Thresholds for emotional expression
- Emotional Triggers:
- User interactions and relationships
- Message content sentiment analysis
- Server events and activities
- Expression Mapping:
- Translation of emotional state to message tone
- Emoji usage patterns based on emotional state
- Activity choices influenced by current emotions
- Bidirectional Effects:
- High emotional intensity accelerates stamina drain
- Low stamina increases irritability or sadness
- Sleep state gradually resets emotional extremes
- Recovery Mechanics:
- Emotional state gradually returns to personality baseline
- Speed of recovery tied to stamina levels
- Major emotional events have longer recovery curves
public class EmotionalService : IEmotionalService
{
private readonly EmotionalState _currentState = new();
private readonly PersonalityProfile _personalityProfile;
private readonly IStaminaService _staminaService;
private readonly SemaphoreSlim _stateLock = new(1, 1);
// Implementation...
public async Task ProcessEmotionalTriggerAsync(EmotionalTrigger trigger)
{
await _stateLock.WaitAsync();
try
{
// Calculate emotional impact
var impact = CalculateEmotionalImpact(trigger);
// Update emotional state
foreach (var (emotion, change) in impact.Changes)
{
_currentState.AdjustEmotion(emotion, change);
}
// Calculate stamina effect
double staminaCost = CalculateEmotionalStaminaCost(impact);
await _staminaService.ConsumeStaminaAsync(staminaCost);
// Log significant emotional changes
if (impact.Significance > 0.5)
{
_logger.LogInformation(
"Significant emotional change: {TriggerType}, Impact: {Significance}",
trigger.Type,
impact.Significance);
}
}
finally
{
_stateLock.Release();
}
}
public async Task<EmotionalExpression> GetCurrentExpressionAsync()
{
await _stateLock.WaitAsync();
try
{
// Get current stamina to influence expression
var stamina = await _staminaService.GetCurrentStaminaAsync();
// Apply personality and stamina modifiers to raw emotional state
return _expressionMapper.MapStateToExpression(_currentState, _personalityProfile, stamina);
}
finally
{
_stateLock.Release();
}
}
public async Task PerformEmotionalMaintenanceAsync()
{
// Called periodically and during sleep periods
await _stateLock.WaitAsync();
try
{
// Gradually return emotions to baseline
foreach (var emotion in _currentState.Emotions)
{
var baselineValue = _personalityProfile.GetEmotionalBaseline(emotion.Key);
var currentValue = emotion.Value;
var recoveryRate = _personalityProfile.GetRecoveryRate(emotion.Key);
// Apply stamina modifier to recovery rate
var stamina = await _staminaService.GetCurrentStaminaAsync();
var staminaModifier = 1.0 + ((100 - stamina) / 200); // 1.0 to 1.5x slower when tired
// Calculate new value moving toward baseline
var step = (baselineValue - currentValue) * (recoveryRate / staminaModifier);
_currentState.SetEmotion(emotion.Key, currentValue + step);
}
}
finally
{
_stateLock.Release();
}
}
}
-
Bot Control Commands:
/bot sleep [duration]
- Force bot to enter sleep mode/bot wake
- Force bot to wake from sleep mode/bot silence [duration]
- Prevent bot from sending messages but continue observation/bot reset
- Reset emotional and conversational state (maintains knowledge)/bot emergency-stop
- Immediately stop all bot activities and disconnect
-
Implementation Details:
- Commands use Discord's slash command system
- All admin commands receive the highest priority and override any active processes
- Use
CancellationTokenSource
to cancel any ongoing operations - Admin actions are logged in Application Insights with special tags
-
Structured Logging Strategy:
- Use consistent schema across all components
- Include source channel, event type, emotion info, and stamina readings
- Log correlation IDs to track events across parallel processes
- Special tags for admin actions and emotional spikes
-
Azure Application Insights Integration:
- Real-time monitoring dashboard for bot activity
- Anomaly detection for unusual behavior patterns
- Performance tracking for response times and resource usage
- Custom metrics for emotional state and stamina levels
public class AdminCommandHandler : IRequest<IResult>
{
private readonly Dictionary<string, CancellationTokenSource> _operationTokens = new();
private readonly IStaminaService _staminaService;
private readonly ILogger<AdminCommandHandler> _logger;
// Implementation...
[SlashCommand("sleep", "Force the bot to enter sleep mode")]
public async Task<IResult> ForceSleepAsync(
[Option("duration", "Sleep duration in minutes")] int? durationMinutes = null)
{
try
{
// Create operation token and register it
var sleepToken = new CancellationTokenSource();
_operationTokens["sleep"] = sleepToken;
// Log the admin command with special tag
_logger.LogInformation(
"Admin command executed: Force Sleep, Duration: {Duration}",
durationMinutes);
// Cancel any active operations
CancelActiveOperations(except: "sleep");
// Set sleep mode
await _staminaService.ForceSleepModeAsync(
durationMinutes.HasValue ? TimeSpan.FromMinutes(durationMinutes.Value) : null);
return Results.Ok("Bot is now sleeping");
}
catch (Exception ex)
{
_logger.LogError(ex, "Error executing admin sleep command");
return Results.Problem("Failed to enter sleep mode");
}
}
private void CancelActiveOperations(string except = null)
{
foreach (var (key, tokenSource) in _operationTokens)
{
if (key != except && !tokenSource.IsCancellationRequested)
{
tokenSource.Cancel();
_logger.LogInformation("Canceled operation: {OperationType}", key);
}
}
}
}
-
Presence Balancing:
- Dynamic throttling of message frequency based on channel activity
- Sensitivity to user engagement cues (short replies, topic changes)
- Natural idle behaviors during low engagement periods
- Stamina-based reduction in proactive engagement when tired
-
Relationship-Driven Interaction:
- Stronger relationship values increase bot's engagement with specific users
- Interest alignment influences topic selection and conversation depth
- Natural development of preferences and social circles within the server
- Appropriate personal boundaries based on relationship stage
public class InteractionDecisionService : IInteractionDecisionService
{
private readonly IKnowledgeService _knowledgeService;
private readonly IStaminaService _staminaService;
private readonly IEmotionalService _emotionalService;
private readonly Random _random = new();
// Implementation...
public async Task<bool> ShouldEngageInConversationAsync(string channelId, IEnumerable<string> participantIds)
{
// Check stamina level
var currentStamina = await _staminaService.GetCurrentStaminaAsync();
// Higher stamina = more likely to engage proactively
double staminaFactor = currentStamina / 100.0;
// Check relationship strength with participants
double relationshipFactor = 0;
foreach (var userId in participantIds)
{
var relationship = await _knowledgeService.GetUserRelationshipAsync(userId);
relationshipFactor += relationship.Strength;
}
relationshipFactor = Math.Min(relationshipFactor / participantIds.Count(), 1.0);
// Check emotional state - more likely to engage when positive
var emotionalState = await _emotionalService.GetCurrentExpressionAsync();
double emotionalFactor = emotionalState.IsPositive ? 1.2 : 0.8;
// Check channel activity level
var channelActivity = await _activityTrackingService.GetChannelActivityLevelAsync(channelId);
double activityFactor = NormalizeActivityLevel(channelActivity);
// Calculate final engagement probability
double engagementProbability =
staminaFactor * 0.4 +
relationshipFactor * 0.3 +
emotionalFactor * 0.2 +
activityFactor * 0.1;
// Add random variation to make behavior less predictable
engagementProbability += (_random.NextDouble() * 0.1) - 0.05;
// Decision with bias toward natural pauses in conversation
return _random.NextDouble() < engagementProbability;
}
private double NormalizeActivityLevel(ActivityLevel level)
{
// Convert activity level to a factor between 0.5 and 1.5
// Higher activity = lower chance to interject (avoid spamming busy channels)
return level switch
{
ActivityLevel.Inactive => 1.5, // More likely to start conversation
ActivityLevel.Low => 1.2, // Good time to engage
ActivityLevel.Moderate => 1.0, // Normal engagement
ActivityLevel.High => 0.7, // Less likely to interject
ActivityLevel.VeryHigh => 0.5, // Minimal interruption of active conversations
_ => 1.0
};
}
}
-
Resource Management:
- Implement background task processing for knowledge graph updates
- Use Azure Functions for scheduled maintenance tasks
- Implement caching for frequently accessed personality and relationship data
- Control AI service token usage with budget-aware request limiting
-
Scalability Design:
- Horizontal scaling of services through containerization
- Separate high-frequency services (observation) from intensive services (analysis)
- Stateless design for core components with state persistence in managed services
- Optimized MediatR pipeline with performance-focused handlers
- Observer system implementation with parallel processing
- Stamina and sleep mechanics
- Basic emotional model integration
- Knowledge graph foundation
- Admin command interface
- Advanced relationship tracking
- Self-regulation tuning
- Voice channel participation
- Emotional expression diversity
- Context-aware conversation patterns
- Thread and forum participation
- Stage events and presentations
- Enhanced voice channel interactions
- Server management capabilities
- Advanced knowledge graph with relationship mapping
- Multimedia content creation and sharing
Future versions will expose API endpoints for:
- Discord Activity and Linked Roles integration
- External persona management and monitoring
- Analytics and conversation insights
- Integration with other AI services
MIT