Using as a Library
manolo-bot is designed as a modular library, allowing you to integrate its AI capabilities into any Python application—not just Telegram bots. This section explains the core concepts and how to use the library components.
Core Concepts
To build an AI assistant with manolo-bot, you need to understand four main components:
LLM Configuration: Defining which AI model to use and providing the necessary credentials.
Bot Configuration: Defining the bot’s identity and behavioral settings.
Storage: Managing the conversation history (context) so the bot can “remember” previous messages.
The Bot/Agent: The main engine that processes messages using the LLM and storage.
LLM Configuration and Builder
The LLMConfig class holds all settings for the AI provider (OpenAI, Google, or Ollama). The LLMBuilder then takes this configuration and returns a compatible LangChain model object.
from manolo_bot.ai.config import LLMConfig
from manolo_bot.ai.llmbot import LLMBuilder
# Example: Google Gemini
llm_config = LLMConfig(google_api_key="your_api_key")
llm = LLMBuilder(llm_config).get_llm()
Bot Configuration
The BotConfig class defines settings like the bot’s name, UUID (for unique identification in storage), and features like multimodal support.
from manolo_bot.ai.config import BotConfig
bot_config = BotConfig(
bot_uuid="my-unique-bot-id",
bot_name="MyAssistant"
)
Storage
The storage component is responsible for persisting conversation history. manolo-bot uses a “per-chat” storage model. Each user or conversation should have its own storage instance identified by a chat_id.
MemoryMessagesStorage: Stores messages in RAM. Useful for testing or simple CLI tools. History is lost when the process restarts.
RedisMessagesStorage: Stores messages in a Redis database. Recommended for production and persistent history.
from manolo_bot.storage.memory_storage import MemoryMessagesStorage
# chat_id is a unique identifier for the current conversation (e.g., a user ID)
storage = MemoryMessagesStorage(bot_uuid="my-unique-bot-id", chat_id=12345)
Main Component: LLMAgent (Recommended)
The LLMAgent is the most powerful and feature-rich component in manolo-bot. It implements Agentic behavior, meaning it uses the LLM as a “reasoning engine” to complete tasks.
What is “Agentic” behavior?
Unlike a standard LLM that generates a single response, an Agent iterates through a loop:
Reasoning: The bot analyzes the user’s request.
Tool Selection: It decides if it needs external information (e.g., using a Search tool).
Iteration: It executes the tool, observes the result, and then iterates.
Completion: It continues this loop until it has enough information to provide a final answer.
Implementation Example
import asyncio
from manolo_bot.ai.llmagent import LLMAgent
from manolo_bot.ai.llmbot import LLMBuilder
from manolo_bot.ai.config import LLMConfig, BotConfig
from manolo_bot.storage.memory_storage import MemoryMessagesStorage
from manolo_bot.ai.tools import get_tools
async def main():
# 1. Setup LLM (Must support Tool Calling)
llm_config = LLMConfig(google_api_key="your_key")
llm = LLMBuilder(llm_config).get_llm()
# 2. Setup Bot Identity
bot_config = BotConfig(bot_uuid="bot-1", bot_name="Assistant")
# 3. Setup Storage
chat_id = 1001
storage = MemoryMessagesStorage(bot_uuid="bot-1", chat_id=chat_id)
await storage.refresh_messages()
# 4. Initialize the Agent with Tools
tools = get_tools()
agent = LLMAgent(
llm=llm,
config=bot_config,
system_instructions="You are a helpful assistant.",
storage=storage,
tools=tools
)
# 5. Answer a message
# The agent will automatically decide when to use search, etc.
response = await agent.answer_message(chat_id=chat_id, message="What happened in the news today?")
print(f"Agent Result: {response.content}")
await storage.commit()
if __name__ == "__main__":
asyncio.run(main())
Simple Alternative: LLMBot
The LLMBot is a simpler implementation designed for: * Simple Models: LLMs that do not support tool calling or have limited reasoning capabilities. * Direct Interaction: When you only need a straightforward chat interface without multi-step reasoning. * Performance: It is slightly faster as it doesn’t perform multiple iterations.
from manolo_bot.ai.llmbot import LLMBot
# ... (Setup is identical to LLMAgent, but without tools)
bot = LLMBot(
llm=llm,
config=bot_config,
system_instructions="You are a simple chatbot.",
storage=storage
)
response = await bot.answer_message(chat_id=chat_id, message="Hello!")
Custom Tools
You can easily provide your own tools to both LLMAgent and LLMBot. This allows you to extend the bot’s capabilities with your own domain-specific logic.
To add custom tools, use the @tool decorator from langchain_core.tools and pass a list of tools to the constructor.
from langchain_core.tools import tool
from manolo_bot.ai.llmagent import LLMAgent
@tool
def get_stock_price(symbol: str) -> str:
"""Gets the current stock price for a given symbol."""
# Your custom logic here
return f"The price of {symbol} is $150.00"
# Initialize with custom tools
custom_tools = [get_stock_price]
agent = LLMAgent(
...,
tools=custom_tools
)
Note
If you provide a tools list, it will replace the default built-in tools. If you want to extend the default tools, you can use the get_tools function:
from manolo_bot.ai.tools import get_tools
all_tools = get_tools(bot_config) + [get_stock_price]
agent = LLMAgent(..., tools=all_tools)