A lightweight, developer-friendly toolkit for building AI agents with Web3 integration. Built with Nest.js (TypeScript), Rukh makes it easy to create, deploy, and scale AI applications with:
- 🚀 Quick setup and minimal configuration
- 🔄 Built-in session management and persistent storage
- đź”— Seamless Web3 integration
- 🛠️ Modular architecture for easy LLM integration (Mistral, Anthropic, OpenAI, etc.)
- đź“ť Auto-generated OpenAPI documentation
- 🎮 Token-gated access control built-in
- ⚡ Production-ready with rate limiting and error handling
- đź”’ Password-protected contexts for secure data management
Live at: http://rukh.w3hc.org/api
Solidity contracts:
- Nest.js
v11
(TypeScript-based API framework) - Ethers
v6
- OpenAPI/Swagger docs
- JSON-based persistent storage
- LangChain.js
- Built-in Web Reader API for fetching webpage content
- Pre-integrated LLM models:
- Mistral
mistral-large-2411
- Anthropic
claude-3-7-sonnet-20250219
- Mistral
Any other LLM service can be easily added (OpenAI, DeepSeek, or any).
pnpm i
Create and edit your .env
file based on .env.template
. You can also deploy your own token contract.
# development
pnpm start
# watch mode
pnpm start:dev
# production mode
pnpm start:prod
The Swagger UI should be available at http://localhost:3000/api
# unit tests
pnpm test
# e2e tests
pnpm test:e2e
# test coverage
pnpm test:cov
Rukh provides a secure context management system that allows you to create and manage separate contexts for different use cases or clients. Each context is password-protected to ensure data security.
Each context has its own index.json
file that stores metadata including password, description, file list, and usage statistics.
The data folder has the following structure:
data/
├── chat-history.json # Stores conversation history for all sessions
├── contexts/ # Contains all context folders
│ └── rukh/ # Example context folder
│ ├── index.json # Context metadata and configuration
│ ├── file1.md # Context content file
│ └── file2.md # Context content file
└── costs.json # Usage tracking and cost data
Each context has an index.json
file with the following structure:
{
"name": "rukh",
"password": "rukh",
"description": "Just Rukh.",
"numberOfFiles": 2,
"totalSize": 16,
"files": [
{
"name": "file1.md",
"description": "File #1",
"size": 8
},
{
"name": "file2.md",
"description": "File #2",
"size": 8
}
],
"queries": [
{
"timestamp": "2025-03-16T12:34:06.046Z",
"origin": "0x...",
"contextFilesUsed": [
"file1.md",
"file2.md"
]
}
]
}
Usage and cost data is stored in costs.json
:
{
"requests": [
{
"timestamp": "2025-03-16T11:58:17.934Z",
"inputCost": 0.0054,
"outputCost": 0.0004,
"totalCost": 0.0058,
"inputTokens": 1794,
"outputTokens": 138,
"message": "What's Rukh",
"sessionId": "e7ebac4f-e177-4461-a0f7-8266f78ff1f9",
"model": "mistral-large-2411"
}
],
"global": {
"totalInputCost": 0.29499999999999993,
"totalOutputCost": 0.06150000000000002,
"totalCost": 0.35649999999999993,
"totalInputTokens": 98289,
"totalOutputTokens": 19530,
"totalRequests": 46,
"lastUpdated": "2025-03-16T14:32:06.276Z",
"modelsUsage": {
"mistral-large-2411": {
"requests": 46,
"inputTokens": 98158,
"outputTokens": 19440,
"cost": 0.35309999999999997
},
"claude-3-7-sonnet-20250219": {
"requests": 2,
"inputTokens": 3849,
"outputTokens": 488,
"cost": 0.0189
}
}
}
}
To create a new context:
curl -X 'POST' \
'http://localhost:3000/context' \
-H 'Content-Type: application/json' \
-d '{
"name": "my-context",
"password": "my-secure-password",
"description": "Optional description for this context"
}'
All context operations require the context password to be provided in the x-context-password
header:
# Upload a file to a context
curl -X 'POST' \
'http://localhost:3000/context/upload' \
-H 'x-context-password: my-secure-password' \
-F 'contextName=my-context' \
-F 'fileDescription=Optional file description' \
-F '[email protected]'
# List files in a context
curl -X 'GET' \
'http://localhost:3000/context/my-context/files' \
-H 'x-context-password: my-secure-password'
# Get a specific file's content
curl -X 'GET' \
'http://localhost:3000/context/my-context/file/myfile.md' \
-H 'x-context-password: my-secure-password'
# Delete a context
curl -X 'DELETE' \
'http://localhost:3000/context/my-context' \
-H 'x-context-password: my-secure-password'
# Delete a file from a context
curl -X 'DELETE' \
'http://localhost:3000/context/my-context/file' \
-H 'x-context-password: my-secure-password' \
-H 'Content-Type: application/json' \
-d '{
"filename": "myfile.md"
}'
Rukh now supports adding web links to your contexts. When you query a context, the content of these links will be automatically fetched, processed, and included in the LLM's context alongside your context files.
The context's index.json
now includes a links
array:
{
"name": "rukh",
"password": "rukh",
"description": "Just Rukh.",
"numberOfFiles": 2,
"totalSize": 16,
"files": [
{
"name": "file1.md",
"description": "File #1",
"size": 8
},
{
"name": "file2.md",
"description": "File #2",
"size": 8
}
],
"links": [
{
"title": "Rukh GitHub Repository",
"url": "https://github.com/w3hc/rukh",
"description": "Official GitHub repository for the Rukh project",
"timestamp": "2025-03-16T12:34:06.046Z"
}
],
"queries": [
{
"timestamp": "2025-03-16T12:34:06.046Z",
"origin": "0x...",
"contextFilesUsed": [
"file1.md",
"file2.md",
"link:https://github.com/w3hc/rukh"
]
}
]
}
All link operations require the context password to be provided in the x-context-password
header:
# Add a link to a context
curl -X 'POST' \
'http://localhost:3000/context/my-context/link' \
-H 'x-context-password: my-secure-password' \
-H 'Content-Type: application/json' \
-d '{
"title": "Rukh GitHub Repository",
"url": "https://github.com/w3hc/rukh",
"description": "Official GitHub repository for the Rukh project"
}'
# List all links in a context
curl -X 'GET' \
'http://localhost:3000/context/my-context/links' \
-H 'x-context-password: my-secure-password'
# Delete a link from a context
curl -X 'DELETE' \
'http://localhost:3000/context/my-context/link' \
-H 'x-context-password: my-secure-password' \
-H 'Content-Type: application/json' \
-d '{
"url": "https://github.com/w3hc/rukh"
}'
When you query an API with a specific context, Rukh will:
- Process all the markdown files in the context folder
- Fetch and extract content from all links in the context using the Web Reader API
- Combine all this information into a single context for the LLM
- Record which files and links were used in the context's query history
This allows you to include live web content in your AI responses without having to manually update your context files. The content is fetched fresh with each query, ensuring you always have the most up-to-date information.
Query tracking now includes links in the contextFilesUsed
array with a link:
prefix to distinguish them from files.
- Context passwords are stored in plain text in each context's
index.json
file. For production use, consider implementing encryption. - Only
.md
files are allowed to be uploaded to contexts. - All operations on a context require the correct password in the
x-context-password
header. - File size is limited to 5MB by default.
- All context usage is tracked and recorded in the context's index file.
Simple request with default model (Mistral):
curl -X 'POST' \
'http://localhost:3000/ask' \
-H 'accept: application/json' \
-H 'Content-Type: application/json' \
-d '{
"message": "What is Rukh?"
}'
{
"message": "What is Rukh?",
"model": "mistral",
"sessionId": "f0ea9dc7-03e8-46a7-b3ad-6c3531211f73",
"walletAddress": "0x265E31444C6E279870eB20c15B0547373635840b"
}
{
"message": "What is Rukh?",
"model": "anthropic",
"sessionId": "f0ea9dc7-03e8-46a7-b3ad-6c3531211f73",
"walletAddress": "0x265E31444C6E279870eB20c15B0547373635840b"
}
To use a specific context in your query:
{
"message": "Summarize all information about Rukh",
"model": "anthropic",
"context": "rukh",
"sessionId": "f0ea9dc7-03e8-46a7-b3ad-6c3531211f73"
}
You can upload a file along with your query using multipart/form-data:
curl -X 'POST' \
'http://localhost:3000/ask' \
-H 'accept: application/json' \
-F 'message=Analyze this document for me' \
-F 'model=anthropic' \
-F '[email protected]'
The API will return a response like:
{
"output": "Rukh is a lightweight, developer-friendly toolkit for building AI agents with Web3 integration. It's built with Nest.js (TypeScript) and makes it easy to create, deploy, and scale AI applications. The name 'Rukh' comes from a legendary enormous bird from mythology, also known as the Roc. The toolkit includes features such as quick setup, session management, Web3 integration, modular architecture for LLM integration, token-gated access control, and more.",
"model": "mistral-large-2411",
"network": "arbitrum-sepolia",
"txHash": "0xd96b35d1daefd6dc8368f7a075a1a627df960a541eb30268b1b85cedbae0214a",
"explorerLink": "https://sepolia.arbiscan.io/tx/0xd96b35d1daefd6dc8368f7a075a1a627df960a541eb30268b1b85cedbae0214a",
"sessionId": "bdce1931-b09d-49ef-954b-d20074d11ffa",
"usage": {
"input_tokens": 512,
"output_tokens": 189
}
}
When using Anthropic Claude, the model
field will show claude-3-7-sonnet-20250219
.
Rukh includes a Web Reader API for fetching and processing webpage content:
# Fetch raw HTML
curl -X 'GET' \
'http://localhost:3000/web-reader?url=https://example.com'
# Extract text and links for LLM processing
curl -X 'GET' \
'http://localhost:3000/web-reader/llm?url=https://example.com'
Sign-In with Ethereum authentication is supported:
# Get a challenge
curl -X 'GET' 'http://localhost:3000/siwe/challenge'
# Verify a signature
curl -X 'POST' \
'http://localhost:3000/siwe/verify' \
-H 'Content-Type: application/json' \
-d '{
"address": "0x1234567890123456789012345678901234567890",
"signature": "0xsignature...",
"nonce": "nonce-from-challenge"
}'
Feel free to reach out to Julien through:
- Element: @julienbrg:matrix.org
- Farcaster: julien-
- Telegram: @julienbrg
- Twitter: @julienbrg
- Discord: julienbrg
- LinkedIn: julienberanger