Unified communication layer for Django (Telegram, WhatsApp, Email)
Project description
Django Unicom
Unified communication layer for Django — easily integrate Telegram bots, WhatsApp bots, and Email bots with a consistent API across all platforms.
📑 Table of Contents
- Quick Start
- Available Platforms
- Core Models & Usage
- Advanced Features
- Production Setup
- Management Commands
- Contributing
- License
- Release Automation
🚀 Quick Start
-
Install the package (plus Playwright browser binaries):
pip install django-unicom # Install the headless Chromium browser that powers PDF export python -m playwright install --with-deps
-
Add required apps to your Django settings:
INSTALLED_APPS = [ ... 'django_ace', # Required for the JSON configuration editor 'unicom', ]
-
Include
unicomURLs in your project'surls.py:This is required so that webhook URLs can be constructed correctly.
from django.urls import path, include urlpatterns = [ ... path('unicom/', include('unicom.urls')), ]
-
Define your public origin: In your Django
settings.py:DJANGO_PUBLIC_ORIGIN = "https://yourdomain.com"
Or via environment variable:
DJANGO_PUBLIC_ORIGIN=https://yourdomain.com
-
Set up media file handling: In your Django
settings.py:MEDIA_URL = '/media/' MEDIA_ROOT = os.path.join(BASE_DIR, '')
In your main project
urls.py:from django.conf import settings from django.conf.urls.static import static urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
-
(Optional, but recommended) Set your TinyMCE Cloud API key — required if you plan to compose Email messages from the Django admin UI.
Obtain a free key at https://www.tiny.cloud, then add it to your
settings.py:UNICOM_TINYMCE_API_KEY = "your-tinymce-api-key"
Or via environment variable:
UNICOM_TINYMCE_API_KEY=your-tinymce-api-key
and then you would still have to load it in settings.py
UNICOM_TINYMCE_API_KEY = os.getenv('UNICOM_TINYMCE_API_KEY', '')
-
(Optional) Set your OpenAI API key — required if you plan to use the AI-powered template population service.
Obtain a key from https://platform.openai.com/api-keys, then set it as an environment variable:
OPENAI_API_KEY="your-openai-api-key"
The application will automatically pick it up from the environment.
-
Install ffmpeg:
ffmpegis required for converting audio files (e.g., Telegram voice notes) to formats compatible with OpenAI and other services. Make sureffmpegis installed on your system or Docker image.
That's it! Unicom can now register and manage public-facing webhooks (e.g., for Telegram bots) based on your defined base URL and can automatically sync with email clients.
📱 Available Platforms
Django Unicom supports the following communication platforms:
- Email - SMTP/IMAP with auto-discovery, rich HTML content, link tracking
- Telegram - Bot API integration with webhooks, media support, typing indicators
- WhatsApp - Business API integration, template messages, delivery status
- Internal - System-to-system messaging within your application
Throughout this documentation, features will be marked as:
- ✅ All platforms: Works across all communication channels
- 📧 Email only: Specific to email channels
- 📱 Telegram only: Specific to Telegram channels
- 💬 WhatsApp only: Specific to WhatsApp channels
- 🤖 LLM features: AI integration (platform-agnostic)
📝 Core Models & Usage
Channel Model
Channels represent communication endpoints for different platforms.
Creating Channels Programmatically
from unicom.models import Channel
# Email Channel - Auto-discovers SMTP/IMAP settings
email_channel = Channel.objects.create(
name="Customer Support Email",
platform="Email",
config={
"EMAIL_ADDRESS": "support@example.com",
"EMAIL_PASSWORD": "your-app-password"
}
)
# Email Channel - Custom SMTP/IMAP settings
email_channel_custom = Channel.objects.create(
name="Marketing Email",
platform="Email",
config={
"EMAIL_ADDRESS": "marketing@example.com",
"EMAIL_PASSWORD": "password",
"IMAP": {
"host": "imap.example.com",
"port": 993,
"use_ssl": True,
"protocol": "IMAP"
},
"SMTP": {
"host": "smtp.example.com",
"port": 587,
"use_ssl": True,
"protocol": "SMTP"
},
"TRACKING_PARAMETER_ID": "utm_source", # 📧 Custom tracking parameter
"MARK_SEEN_WHEN": "on_request_completed" # 📧 When to mark emails as seen
}
)
# Telegram Channel - Auto-generates webhook secret
telegram_channel = Channel.objects.create(
name="Customer Bot",
platform="Telegram",
config={
"API_TOKEN": "your-bot-token-from-botfather"
}
)
# Validate channel (sets up webhooks/connections)
channel.validate() # Returns True if successful
Creating Channels via Admin Interface
- Go to Django Admin > Unicom > Channels
- Click "Add Channel"
- Fill in the name, select platform, and add configuration JSON
- Save - the channel will automatically validate and set up webhooks
Sending Messages with Channels
# ✅ All platforms: Basic message sending
message = channel.send_message({
'chat_id': 'recipient_chat_id',
'text': 'Hello from Django Unicom!'
})
# 📧 Email only: New email thread
message = email_channel.send_message({
'to': ['recipient@example.com'],
'subject': 'Welcome!',
'html': '<h1>Welcome to our service!</h1>'
})
# 📧 Email only: Email with CC/BCC
message = email_channel.send_message({
'to': ['primary@example.com'],
'cc': ['manager@example.com'],
'bcc': ['archive@example.com'],
'subject': 'Team Update',
'html': '<p>Here is the latest update...</p>'
})
# 📧 Email only: Reply to existing email thread
message = email_channel.send_message({
'chat_id': 'existing_email_thread_id',
'html': '<p>Thanks for your message!</p>'
# Subject is automatically derived from thread
})
# 📱 Telegram only: Message with interactive buttons
from unicom.services.telegram.create_inline_keyboard import create_callback_button, create_inline_keyboard
message = telegram_channel.send_message({
'chat_id': 'user_chat_id',
'text': 'Choose an option:',
'reply_markup': create_inline_keyboard([
[create_callback_button("Confirm", {"action": "confirm"}, message=message)],
[create_callback_button("Cancel", {"action": "cancel"}, message=message)]
])
})
# See "Interactive Buttons & Callbacks" section for handling button clicks
Message Model
Messages represent individual communications across all platforms with rich metadata and tracking capabilities.
Key Message Fields by Platform
The Message model contains many important fields that provide detailed information about message status, tracking, and content. Important: Each field is only populated by specific platforms:
from unicom.models import Message
# Content fields (✅ All platforms)
message.text # Plain text content
message.sender_name # Display name of sender
message.timestamp # When message was created
message.is_outgoing # True=outgoing, False=incoming, None=system
message.platform # 'Email', 'Telegram', 'WhatsApp', 'Internal'
message.media_type # 'text', 'html', 'image', 'audio', 'tool_call', 'tool_response'
message.media # Attached media file
message.raw # Raw platform-specific data (JSON)
# 📧 Email-only content fields
message.html # HTML content
message.subject # Email subject line
message.to # List of recipient email addresses (array)
message.cc # List of CC email addresses (array)
message.bcc # List of BCC email addresses (array)
message.imap_uid # IMAP UID for server operations
# 💬 WhatsApp-only status tracking
message.sent # Updated when WhatsApp confirms message sent
message.delivered # Updated when WhatsApp confirms message delivered
message.seen # Updated when WhatsApp confirms message read
message.time_sent # When WhatsApp confirmed message sent
message.time_delivered # When WhatsApp confirmed message delivered
message.time_seen # When WhatsApp confirmed message read
# 📧 Email-only tracking (via tracking pixels & links)
message.opened # Set to True when recipient opens email
message.time_opened # When email was first opened (via tracking pixel)
message.link_clicked # Set to True when any tracked link is clicked
message.time_link_clicked # When first link was clicked
message.clicked_links # Array of all URLs that have been clicked
message.tracking_id # UUID used for tracking pixel and link tracking
Platform-Specific Usage Examples
# 💬 WhatsApp: Check delivery status (only WhatsApp provides this data)
whatsapp_msg = Message.objects.get(id='whatsapp_message_id')
if whatsapp_msg.delivered:
print(f"WhatsApp message delivered at: {whatsapp_msg.time_delivered}")
if whatsapp_msg.seen:
print(f"WhatsApp message read at: {whatsapp_msg.time_seen}")
# 📧 Email: Check tracking data (only emails have open/click tracking)
email_msg = Message.objects.get(id='email_message_id')
if email_msg.opened:
print(f"Email opened at: {email_msg.time_opened}")
if email_msg.link_clicked:
print(f"Links clicked: {email_msg.clicked_links}")
print(f"First click at: {email_msg.time_link_clicked}")
# ✅ All platforms: Basic message info
for message in Message.objects.filter(channel=channel):
print(f"{message.platform}: {message.sender_name} - {message.text}")
if message.is_outgoing:
print(" (Outgoing message)")
elif message.is_outgoing is False:
print(" (Incoming message)")
else:
print(" (System message)")
# 💬 WhatsApp-specific queries
unread_whatsapp = Message.objects.filter(
platform='WhatsApp',
is_outgoing=True,
seen=False # Only WhatsApp populates this field
)
# 📧 Email-specific queries
opened_emails = Message.objects.filter(
platform='Email',
opened=True # Only emails have open tracking
)
clicked_emails = Message.objects.filter(
platform='Email',
link_clicked=True # Only emails have click tracking
).values('subject', 'clicked_links', 'time_link_clicked')
Understanding Field Limitations
Important Notes:
- Delivery tracking (
delivered,time_delivered): Only WhatsApp provides delivery confirmations - Read tracking (
seen,time_seen): Only WhatsApp provides read receipts - Email open tracking (
opened,time_opened): Only works when recipient loads images/tracking pixels - Email click tracking (
link_clicked,time_link_clicked,clicked_links): Only works for links that go through tracking system - Email "seen" status: Use
imap_uidfield and IMAP operations, not theseenfield
Accessing Messages
from unicom.models import Message
# Get message by ID
message = Message.objects.get(id='message_id')
# Get recent messages for a channel
recent_messages = Message.objects.filter(
channel=channel
).order_by('-timestamp')[:10]
# Get conversation history
chat_messages = Message.objects.filter(
chat_id='chat_id'
).order_by('timestamp')
Replying to Messages
# ✅ All platforms: Reply with text
reply = message.reply_with({
'text': 'Thanks for your message!'
})
# 📧 Email only: Reply with HTML
reply = message.reply_with({
'html': '<p>Thank you for contacting us!</p><p>We will get back to you soon.</p>'
})
# ✅ All platforms: Reply with media
reply = message.reply_with({
'text': 'Here is the file you requested',
'file_path': '/path/to/file.pdf'
})
# 📱 Telegram only: Reply with interactive buttons
from unicom.services.telegram.create_inline_keyboard import create_simple_keyboard
reply = message.reply_with({
'text': 'Would you like to continue?',
'reply_markup': create_simple_keyboard(
"Yes", "continue_yes",
"No", "continue_no"
)
})
# See "Interactive Buttons & Callbacks" section for handling button clicks
Via Admin Interface
- Go to Django Admin > Unicom > Messages
- Find the message you want to reply to
- Click on the message ID to open details
- Use the "Reply" button in the interface
- Compose your reply using the rich text editor (📧 email) or plain text
Chat Model
Chats represent conversations/threads across platforms.
Working with Chats
from unicom.models import Chat
# Get chat by ID
chat = Chat.objects.get(id='chat_id')
# Send message to chat
message = chat.send_message({
'text': 'Hello everyone!'
})
# 📧 Email only: Reply to last incoming message in email thread
reply = chat.send_message({
'html': '<p>Following up on our previous conversation...</p>'
})
# Reply to specific message in chat
reply = chat.send_message({
'reply_to_message_id': 'specific_message_id',
'text': 'Replying to your specific question...'
})
Template System
Create reusable message templates for consistent communication.
Creating Templates Programmatically
from unicom.models import MessageTemplate
# Create a basic template
template = MessageTemplate.objects.create(
title='Welcome Email',
content='<h1>Welcome {{name}}!</h1><p>Thank you for joining {{company}}.</p>',
category='Onboarding'
)
# Make template available for specific channels
template.channels.add(email_channel)
template.channels.add(telegram_channel)
# 🤖 AI-powered template population (requires OpenAI API key)
populated_content = template.populate(
html_prompt="User name is John Doe, company is Acme Corp",
model="gpt-4"
)
Creating Templates via Admin Interface
- Go to Django Admin > Unicom > Message Templates
- Click "Add Message Template"
- Fill in title, description, category
- Create your HTML content using TinyMCE editor (📧 email templates get rich editor)
- Select which channels can use this template
- Save template
Using Templates in Messages
# Get template and use its content
template = MessageTemplate.objects.get(title='Welcome Email')
# Use template content directly
message = channel.send_message({
'to': ['newuser@example.com'],
'subject': 'Welcome!',
'html': template.content.replace('{{name}}', 'John Doe')
})
# Or use AI population
populated = template.populate("User is John Doe from Acme Corp")
message = channel.send_message({
'to': ['john@acme.com'],
'subject': 'Welcome!',
'html': populated
})
Draft Messages & Scheduling
Create draft messages and schedule them for later sending.
Creating Draft Messages
from unicom.models import DraftMessage
from django.utils import timezone
# Create a scheduled email
draft = DraftMessage.objects.create(
channel=email_channel,
to=['customer@example.com'],
subject='Weekly Newsletter',
html='<h1>This week\'s updates...</h1>',
send_at=timezone.now() + timezone.timedelta(hours=24),
is_approved=True,
status='scheduled'
)
# Create a Telegram draft
telegram_draft = DraftMessage.objects.create(
channel=telegram_channel,
chat_id='telegram_chat_id',
text='Scheduled announcement for tomorrow',
send_at=timezone.now() + timezone.timedelta(days=1),
is_approved=True,
status='scheduled'
)
# Send draft immediately (if approved and time has passed)
sent_message = draft.send()
Creating Drafts via Admin Interface
- Go to Django Admin > Unicom > Draft Messages
- Click "Add Draft Message"
- Select channel and fill in recipient details
- Compose message content
- Set "Send at" time for scheduling
- Mark as "Approved" when ready to send
- Status will automatically update to "Scheduled"
🚀 Advanced Features
Email-Specific Features
📧 Link Tracking
Email channels automatically track which links recipients click:
# Send email with trackable links
message = email_channel.send_message({
'to': ['user@example.com'],
'subject': 'Check out our new features',
'html': '''
<p>Visit our <a href="https://example.com/features">features page</a></p>
<p>Or check the <a href="https://example.com/docs">documentation</a></p>
'''
})
# Check tracking data later
if message.link_clicked:
print(f"First link clicked at: {message.time_link_clicked}")
print(f"Clicked links: {message.clicked_links}")
if message.opened:
print(f"Email opened at: {message.time_opened}")
📧 Rich HTML Content with TinyMCE
The admin interface provides a rich text editor for composing HTML emails with features like:
- Font formatting, colors, styles
- Image uploads and inline images
- Tables, lists, links
- Template insertion
- AI-powered content generation
📧 DKIM and SPF Verification
Email channels automatically validate DKIM and SPF records for incoming messages, ensuring email authenticity and preventing spoofing.
Telegram-Specific Features
📱 Typing Indicators
from unicom.services.telegram import start_typing_in_telegram, stop_typing_in_telegram
# Show typing indicator
start_typing_in_telegram(telegram_channel, chat_id="telegram_chat_id")
# Your processing logic here
import time
time.sleep(2)
# Stop typing and send message
stop_typing_in_telegram(telegram_channel, chat_id="telegram_chat_id")
message = telegram_channel.send_message({
'chat_id': 'telegram_chat_id',
'text': 'Here is your response!'
})
📱 Interactive Messages with Action Buttons
Telegram channels support inline keyboard buttons. Pass any JSON-serializable callback_data:
from unicom.services.telegram.create_inline_keyboard import (
create_inline_keyboard, create_callback_button, create_url_button
)
# Create message with buttons
message = telegram_channel.send_message({
'chat_id': 'telegram_chat_id',
'text': 'Do you want to continue?',
'reply_markup': create_inline_keyboard([
[create_callback_button("Yes", {"action": "confirm"}, message=message)],
[create_callback_button("No", {"action": "cancel"}, message=message)],
[create_url_button("Visit Website", "https://example.com")]
])
})
# Rich data structures
product_buttons = create_inline_keyboard([
[create_callback_button("Product A", {"product_id": 123, "price": 29.99}, message=message)],
[create_callback_button("Product B", {"product_id": 456, "price": 49.99}, message=message)]
])
# Optional expiration
from django.utils import timezone
from datetime import timedelta
expiring_button = create_callback_button(
"Limited Offer",
{"offer_id": 789},
message=message,
expires_at=timezone.now() + timedelta(hours=24)
)
📱 Handling Button Clicks
When users click buttons, handle them with Django signals - no configuration needed:
from django.dispatch import receiver
from unicom.signals import telegram_callback_received
@receiver(telegram_callback_received)
def handle_button_clicks(sender, callback_execution, clicking_account, original_message, tool_call, **kwargs):
"""
Handle button clicks.
Args:
callback_execution: CallbackExecution instance with callback_data
clicking_account: The unicom.Account that clicked the button
original_message: The Message containing the buttons
tool_call: Optional ToolCall if button was from a tool (None otherwise)
Note: unicom.Account represents a platform user (e.g., Telegram user, email address).
To access Django auth.User: clicking_account.member.user (if member exists)
"""
data = callback_execution.callback_data
# Handle dict callback_data
if isinstance(data, dict):
if data.get('action') == 'confirm':
process_confirmation(clicking_account)
original_message.reply_with({'text': '✅ Confirmed!'})
elif data.get('action') == 'buy_product':
product_id = data['product_id']
product = get_product(product_id)
# Create new buttons with callback_data
original_message.reply_with({
'text': f'Product: {product.name}\nPrice: ${product.price}',
'reply_markup': create_inline_keyboard([
[create_callback_button('Confirm Purchase', {'action': 'confirm_purchase', 'product_id': product_id}, message=original_message, account=clicking_account)],
[create_callback_button('Cancel', {'action': 'cancel'}, message=original_message, account=clicking_account)]
])
})
# Handle string callback_data
elif data == 'cancel':
original_message.reply_with({'text': '❌ Cancelled'})
# Access Django User if needed
if clicking_account.member and clicking_account.member.user:
django_user = clicking_account.member.user
# Do something with django_user
# If button was from a tool, you can respond to the tool call
if tool_call and data.get('action') == 'confirm':
# Inform the LLM that the user confirmed
tool_call.respond({'confirmed': True, 'user_id': clicking_account.id})
Where to put your callback handler:
Create a file like your_app/callback_handlers.py and import it in your app's apps.py:
# your_app/apps.py
from django.apps import AppConfig
class YourAppConfig(AppConfig):
name = 'your_app'
def ready(self):
import your_app.callback_handlers # Register signal handlers
Make sure your app is in INSTALLED_APPS in settings.py.
Key Features:
- Flexible Data: Store any JSON-serializable data (dict, list, str, int, bool, None)
- Security: Only the intended account can click the button
- Expiration: Optional
expires_atparameter for time-limited buttons - Reusable: Buttons can be clicked multiple times (developers control behavior)
- Efficient: Callback data stored in DB, only ID sent to Telegram
- No Message Creation: Button clicks don't create Message objects - they only trigger handlers
- Tool Integration: When buttons are from tools, handlers can use
tool_call.respond()to inform the LLM
📱 Tool-Generated Buttons (Advanced)
When a tool sends buttons, you can link them to the ToolCall so handlers can respond to the LLM:
# In your tool code (e.g., definitions/tools/my_tool.py)
def my_interactive_tool(question: str) -> str:
"""Ask user a question with buttons and wait for response."""
# Get the tool_call object - available in tool context
# This requires your tool system to pass tool_call to tools
from unicom.models import ToolCall
tool_call = ToolCall.objects.filter(
tool_name='my_interactive_tool',
status='PENDING'
).order_by('-created_at').first()
# Or if your tool system provides it directly:
# tool_call = context.get('tool_call') # depends on your implementation
# Send message with buttons linked to this tool call
message.reply_with({
'text': f'Question: {question}',
'reply_markup': create_inline_keyboard([
[create_callback_button(
"Yes",
{"tool": "my_interactive_tool", "action": "answer", "value": "yes"},
message=message,
tool_call=tool_call # Link button to tool call
)],
[create_callback_button(
"No",
{"tool": "my_interactive_tool", "action": "answer", "value": "no"},
message=message,
tool_call=tool_call
)]
])
})
return "Question sent to user, waiting for response..."
# In your callback handler (callback_handlers.py)
@receiver(telegram_callback_received)
def handle_tool_buttons(sender, callback_execution, clicking_account, original_message, tool_call, **kwargs):
data = callback_execution.callback_data
# Route to correct handler based on tool name
if isinstance(data, dict) and data.get('tool') == 'my_interactive_tool':
if data.get('action') == 'answer':
# User clicked a button from the tool
answer = data['value']
# Respond to the tool call - this will notify the LLM
if tool_call:
tool_call.respond({
'question_answered': True,
'answer': answer,
'user_id': clicking_account.id
})
# Also send confirmation to user
original_message.reply_with({
'text': f'✅ You answered: {answer}'
})
📱 Button Routing Best Practices
When building applications with multiple button types, use a consistent routing strategy:
Recommended Pattern: Use a "type" or "handler" field
# Define button types as constants for consistency
BUTTON_TYPES = {
'PRODUCT': 'product_handler',
'NAVIGATION': 'nav_handler',
'SETTINGS': 'settings_handler',
'TOOL': 'tool_handler'
}
# Create buttons with type field
create_callback_button(
"Buy Product A",
{
"type": "product_handler", # Routes to product handler
"action": "buy",
"product_id": 123
},
message=message
)
create_callback_button(
"Settings",
{
"type": "settings_handler", # Routes to settings handler
"action": "show_settings"
},
message=message
)
# In your callback handler - route based on type
@receiver(telegram_callback_received)
def handle_all_buttons(sender, callback_execution, clicking_account, original_message, tool_call, **kwargs):
data = callback_execution.callback_data
if not isinstance(data, dict):
return # Skip non-dict data
# Route to appropriate handler based on type
handler_type = data.get('type')
if handler_type == 'product_handler':
handle_product_buttons(data, clicking_account, original_message)
elif handler_type == 'settings_handler':
handle_settings_buttons(data, clicking_account, original_message)
elif handler_type == 'tool_handler':
handle_tool_buttons(data, clicking_account, original_message, tool_call)
else:
# Unknown type - log or handle gracefully
print(f"Unknown button type: {handler_type}")
def handle_product_buttons(data, account, message):
"""Handle product-related button clicks"""
if data.get('action') == 'buy':
product_id = data['product_id']
# Process purchase...
message.reply_with({'text': f'Processing purchase for product {product_id}'})
def handle_settings_buttons(data, account, message):
"""Handle settings-related button clicks"""
if data.get('action') == 'show_settings':
# Show settings menu...
message.edit_original_message({'text': '⚙️ Settings Menu'})
def handle_tool_buttons(data, account, message, tool_call):
"""Handle tool-generated button clicks"""
if tool_call and data.get('action') == 'confirm':
tool_call.respond({'confirmed': True})
message.reply_with({'text': '✅ Confirmed'})
Alternative Pattern: Multiple Signal Receivers
# Register separate handlers for different button types
@receiver(telegram_callback_received)
def handle_product_buttons(sender, callback_execution, **kwargs):
data = callback_execution.callback_data
# Only handle product buttons
if isinstance(data, dict) and data.get('type') == 'product':
# Handle product actions...
pass
@receiver(telegram_callback_received)
def handle_navigation_buttons(sender, callback_execution, **kwargs):
data = callback_execution.callback_data
# Only handle navigation buttons
if isinstance(data, dict) and data.get('type') == 'nav':
# Handle navigation...
pass
Scalable Data Structure Example:
# Well-structured callback data for complex applications
callback_data = {
"type": "product_handler", # Routes to correct handler
"action": "add_to_cart", # Specific action
"entity_type": "product", # Type of entity
"entity_id": 123, # Entity identifier
"metadata": { # Additional context
"source": "search_results",
"page": 2
}
}
📱 Editing Telegram Messages
Telegram allows you to edit messages in place instead of sending new ones:
# Edit a message you sent
message = telegram_channel.send_message({
'chat_id': 'user_chat_id',
'text': 'Processing your request...'
})
# Later, update the same message
from unicom.services.telegram.edit_telegram_message import edit_telegram_message
edit_telegram_message(telegram_channel, message, {
'text': '✅ Request completed!'
})
# Edit messages with buttons (common in callback handlers)
from django.dispatch import receiver
from unicom.signals import telegram_callback_received
from unicom.services.telegram.create_inline_keyboard import create_inline_keyboard, create_callback_button
@receiver(telegram_callback_received)
def handle_navigation(sender, callback_execution, clicking_account, original_message, tool_call, **kwargs):
button_data = callback_execution.callback_data
if button_data == 'show_settings':
# Edit the original message to show settings
original_message.edit_original_message({
'text': '⚙️ Settings Menu',
'reply_markup': create_inline_keyboard([
[create_callback_button("Account", "settings_account", message=original_message, account=clicking_account)],
[create_callback_button("Privacy", "settings_privacy", message=original_message, account=clicking_account)],
[create_callback_button("🔙 Back", "main_menu", message=original_message, account=clicking_account)]
])
})
Common use cases:
- Updating status messages (e.g., "Processing..." → "Complete!")
- Creating navigation menus that update in place
- Building interactive forms without spamming the chat
- Showing real-time progress updates
📱 File Downloads and Voice Messages
Telegram channels automatically handle file downloads and voice message processing:
# Voice messages are automatically converted to compatible formats
# and can be processed by LLM services
if message.media_type == 'audio':
# Voice message is available in message.media
# Converted to MP3 format for compatibility
llm_response = message.reply_using_llm(
model="gpt-4-vision-preview",
multimodal=True # Enables audio processing
)
LLM Integration
🤖 AI-Powered Responses (Platform-Agnostic)
# Basic LLM reply to any message
response = message.reply_using_llm(
model="gpt-4",
system_instruction="You are a helpful customer service assistant",
depth=10 # Include last 10 messages for context
)
# 🤖 Multimodal support (images, audio)
response = message.reply_using_llm(
model="gpt-4-vision-preview",
multimodal=True, # Process images and audio
voice="alloy" # Voice for audio responses
)
🤖 Tool Call System
The LLM system can call external functions and tools:
# Log tool interactions
message.log_tool_interaction(
tool_call={
"name": "search_database",
"arguments": {"query": "user orders", "limit": 5},
"id": "call_123"
}
)
# Log tool response
message.log_tool_interaction(
tool_response={
"call_id": "call_123",
"result": {"orders": [...], "count": 3}
}
)
# Get LLM-ready conversation including tool calls
conversation = message.as_llm_chat(depth=20, mode="chat")
🤖 Chat-Level Tool Interactions
# System-initiated tool call
chat.log_tool_interaction(
tool_call={"name": "cleanup_cache", "arguments": {}, "id": "call_456"}
)
# With specific reply target
chat.log_tool_interaction(
tool_call={"name": "fetch_data", "arguments": {"user_id": 123}, "id": "call_789"},
reply_to=some_message
)
Delayed Tool Calls
🤖 Request-Based Tool Call Management
The LLM system supports delayed tool calls that can take hours or days to complete, perfect for reminders, monitoring, and long-running processes.
from unicom.models import Request, ToolCall
# Submit multiple tool calls from a request (atomic operation)
request = Request.objects.get(id='request_id')
tool_calls = request.submit_tool_calls([
{
"name": "set_reminder",
"arguments": {"text": "Meeting tomorrow", "delay_hours": 24},
"id": "call_123" # Optional, auto-generated if omitted
},
{
"name": "monitor_system",
"arguments": {"threshold": 90}
}
])
# Days later... respond to tool calls
reminder_call = ToolCall.objects.get(call_id="call_123")
msg, child_request = reminder_call.respond("Reminder: Meeting in 1 hour")
# Creates new child request for further processing
# For periodic/ongoing tools, set status to ACTIVE
monitor_call = tool_calls[1]
monitor_call.status = 'ACTIVE'
monitor_call.save()
# Now it can respond indefinitely without creating child requests
monitor_call.respond("CPU usage: 95%") # Just logs, no child request
monitor_call.respond("CPU usage: 92%") # Just logs, no child request
🤖 Request Hierarchy and Final Response Logic
# Only when ALL pending tool calls respond does system create child request
request = Request.objects.get(id='parent_request')
# Submit 3 tool calls
calls = request.submit_tool_calls([
{"name": "search", "arguments": {"query": "data"}},
{"name": "analyze", "arguments": {"input": "results"}},
{"name": "report", "arguments": {"format": "pdf"}}
])
# Respond to each (no child request yet)
calls[0].respond("search results") # No child - not final
calls[1].respond("analysis complete") # No child - not final
calls[2].respond("report generated") # Child request created!
# Child request inherits context from initial request
child = Request.objects.filter(parent_request=request).first()
print(f"Child inherits: {child.account}, {child.category}, {child.member}")
🤖 Request Tracking Fields
New fields added to Request model for LLM and tool call tracking:
request.parent_request # Parent request that spawned this one
request.initial_request # Root request that started the chain
request.tool_call_count # Number of tool calls made from this request
request.llm_calls_count # Number of LLM API calls made
request.llm_token_usage # Total tokens consumed by LLM
Message Scheduling
Automated Scheduling System
# Check and process scheduled messages manually
from unicom.services.crossplatform.scheduler import process_scheduled_messages
result = process_scheduled_messages()
print(f"Processed {result['total_due']} messages")
print(f"Sent: {result['sent']}, Failed: {result['failed']}")
⚙️ Production Setup
IMAP Listeners
Email channels require IMAP listeners to receive incoming emails in real-time.
Development (Django runserver)
When using python manage.py runserver, IMAP listeners start automatically with the server.
Production (Gunicorn, uWSGI, etc.)
In production deployments, you need to run IMAP listeners as a separate process:
# Start IMAP listeners for all active email channels
python manage.py start_imap_listeners
This command will:
- Start IMAP IDLE connections for all active email channels
- Keep running until stopped with Ctrl+C
- Automatically reconnect if connections drop
- Process incoming emails in real-time
Docker/Containerized Deployments
Add a separate service in your docker-compose.yml:
services:
web:
# Your main Django app
imap_listener:
# Same image as your web service
build: .
command: python manage.py start_imap_listeners
volumes:
- .:/app
environment:
# Same environment as web service
depends_on:
- db
Scheduled Message Processing
For automated sending of scheduled messages:
# Process scheduled messages every 10 seconds (default)
python manage.py send_scheduled_messages
# Custom interval (30 seconds)
python manage.py send_scheduled_messages --interval 30
Add this as a background service or cron job in production.
🛠️ Management Commands
Available management commands for production and development:
start_imap_listeners
Starts IMAP listeners for all active email channels. Required in production when not using runserver.
python manage.py start_imap_listeners
send_scheduled_messages
Continuously processes and sends scheduled messages.
# Default 10-second interval
python manage.py send_scheduled_messages
# Custom interval
python manage.py send_scheduled_messages --interval 30
run_as_llm_chat
Triggers an LLM response to a specific message (useful for testing AI features).
python manage.py run_as_llm_chat <message_id>
🧑💻 Contributing
We ❤️ contributors!
Requirements:
- Docker & Docker Compose installed
Getting Started:
-
Clone the repo:
git clone https://github.com/meena-erian/unicom.git cd unicom
-
Create a
db.envfile in the root:POSTGRES_DB=unicom_test POSTGRES_USER=unicom POSTGRES_PASSWORD=unicom DJANGO_PUBLIC_ORIGIN=https://yourdomain.com # Needed if you want to use the rich-text email composer in the admin UNICOM_TINYMCE_API_KEY=your-tinymce-api-key # Needed if you want to use the AI template population service OPENAI_API_KEY=your-openai-api-key
-
Start the dev environment:
docker-compose up --build
-
Run tests:
docker-compose exec app pytest
or just
pytest
Note: To run
test_telegram_livetests you need to createtelegram_credentials.pyin the tests folder and define in itTELEGRAM_API_TOKENandTELEGRAM_SECRET_TOKENand to runtest_email_liveyou need to createemail_credentials.pyin the tests folder and define in itEMAIL_CONFIGdict with the propertiesEMAIL_ADDRESS: str,EMAIL_PASSWORD: str, andIMAP: dict, andSMTP: dict, each ofIMAPandSMTPcontainshost:str ,port:int,use_ssl:bool,protocol: (IMAP|SMTP)
No need to modify settings.py — everything is pre-wired to read from db.env.
📄 License
MIT License © Meena (Menas) Erian
📦 Release Automation
To release a new version to PyPI:
-
Ensure your changes are committed and pushed.
-
Run:
make release VERSION=1.2.3
This will:
- Tag the release as v1.2.3 in Git
- Push the tag
- Build the package
- Upload to PyPI using your .pypirc
-
For an auto-generated version based on date/time, just run:
make releaseThis will use the current date/time as the version (e.g., 2024.06.13.1530).
The version is automatically managed by setuptools_scm from Git tags and is available at runtime as unicom.__version__.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file django_unicom-25.2.31.tar.gz.
File metadata
- Download URL: django_unicom-25.2.31.tar.gz
- Upload date:
- Size: 190.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
061678d1b9c355189298bdfab6bdcbeaa59b5ee28932635f169b10c3edb0b1e1
|
|
| MD5 |
c6e4b478dae8f8c6ad47a1e63f27e876
|
|
| BLAKE2b-256 |
2ce293172a5621c1d974df0c75f8ef4a30232e2160a1119bd4299f0ba91b1d62
|
File details
Details for the file django_unicom-25.2.31-py3-none-any.whl.
File metadata
- Download URL: django_unicom-25.2.31-py3-none-any.whl
- Upload date:
- Size: 209.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e718d117b3096aa826466decf100eef79c8ac8a5ed32279b6e32ef4f0d4ae2f8
|
|
| MD5 |
1492001eda4e2d9a76a8479351df6ec6
|
|
| BLAKE2b-256 |
f8d886c87b789d7fcb7921bc937e1bcda81b4134591041f1aad2bf7c87e7201e
|