Hasan Orchestrator
An MCP server that orchestrates multiple devices (mobile, desktop, laptop) via REST API, allowing a local voice assistant (Hermes) to execute actions across devices through dynamic capability routing.
README
Hasan Orchestrator
Orchestrateur MCP multi-devices en Python/FastAPI faisant le lien entre Hermes (assistant vocal local) et plusieurs appareils (mobile, desktop, laptop) via une API REST avec polling.
Hermes (localhost) → MCP localhost:8643
↓
Orchestrateur FastAPI (localhost:8080)
↓ HTTPS + Bearer token
┌───────────┼───────────┐
"phone" "desk" "laptop"
(polling) (polling) (polling)
Les devices initient toujours la connexion (register, heartbeat, long polling). L'orchestrateur ne contacte jamais directement un device.
1. Installation
cd ~/hasan-orchestrator
./install.sh
Le script :
- Crée un environnement virtuel Python (
venv/) - Installe les dépendances (
requirements.txt) - Crée
~/.hasan-orchestrator/pour la base SQLite (registry.db) - Génère
ORCHESTRATOR_ADMIN_KEYdans.envsi absent - Installe et active le service systemd utilisateur
hasan-orchestrator.service
Pour une installation manuelle :
python3 -m venv venv
source venv/bin/activate
pip install -r requirements.txt
cp .env.example .env
# éditer .env et définir ORCHESTRATOR_ADMIN_KEY (openssl rand -hex 32)
python main.py
L'API REST écoute par défaut sur http://127.0.0.1:8080.
2. Configuration Hermes
Ajouter dans ~/.hermes/config.yaml :
Option A — MCP en stdio (recommandé)
mcp:
servers:
- name: hasan-orchestrator
type: stdio
command: /chemin/vers/hasan-orchestrator/venv/bin/python
args: ["/chemin/vers/hasan-orchestrator/mcp_server.py"]
description: "Contrôle multi-devices (phone, desk, laptop)"
Le serveur MCP en stdio doit être lancé alors que l'orchestrateur FastAPI tourne déjà (il s'appuie sur l'API interne
/internal/*pour créer les commandes et lire les résultats).
Option B — MCP exposé via HTTP
mcp:
servers:
- name: hasan-orchestrator
url: "http://localhost:8643/mcp"
auth:
type: bearer
token: "${ORCHESTRATOR_ADMIN_KEY}"
3. Connecter un nouveau device
Chaque device (agent mobile/desktop/laptop) doit :
-
Calculer un
device_hashunique et immuable — un SHA256 généré une seule fois (par exemple à partir d'un identifiant matériel + sel aléatoire) et conservé localement. -
S'enregistrer :
curl -X POST http://localhost:8080/register \
-H "Content-Type: application/json" \
-d '{
"device_name": "phone",
"device_hash": "a3f2c8...",
"device_type": "mobile_agent",
"version": "1.0.0",
"capabilities": {
"send_sms": {"enabled": true, "auth_required": false},
"make_call": {"enabled": true, "auth_required": true},
"screenshot": {"enabled": true, "auth_required": false},
"get_battery": {"enabled": true, "auth_required": false}
}
}'
Réponse :
{
"status": "registered",
"session_token": "tok_...",
"heartbeat_interval": 30,
"polling_interval": 30,
"server_time": "..."
}
-
Conserver
session_tokenet l'utiliser dans le headerAuthorization: Bearer tok_...pour tous les appels suivants. -
Envoyer un heartbeat toutes les
heartbeat_intervalsecondes :
curl -X POST http://localhost:8080/heartbeat \
-H "Authorization: Bearer tok_..." \
-H "Content-Type: application/json" \
-d '{
"device_hash": "a3f2c8...",
"capabilities_version": "<hash renvoyé/calculé>",
"network": {"ip": "192.168.1.42", "transport": "https", "nat": true, "carrier": "WiFi"}
}'
Si capabilities_refresh_needed: true est retourné, refaire un /register
complet (les capabilities ont été modifiées côté orchestrateur).
- Boucler sur
GET /commands(long polling, jusqu'à 55s) pour recevoir les commandes à exécuter, puis poster le résultat sur/results.
Un device sans heartbeat depuis 2 × heartbeat_interval est marqué offline.
4. Tools MCP disponibles
L'orchestrateur n'a aucune connaissance hardcodée des capabilities. Les 3 tools ci-dessous sont les seuls exposés à Hermes :
| Tool | Description |
|---|---|
device_list() |
Liste les devices online et leurs capabilities activées |
device_info(device_name) |
Détails complets d'un device (registry, capabilities, statut) |
exec_action(action, params, device_name?) |
Exécute n'importe quelle action sur un device |
Principe : le device déclare ses propres capabilities à l'enregistrement.
L'orchestrateur les stocke et les route dynamiquement via exec_action.
Si device_name n'est pas précisé, l'orchestrateur choisit automatiquement
l'unique device online disposant de la capability demandée. S'il y en a
plusieurs, il demande de préciser.
Exemples d'appels exec_action
exec_action("send_sms", {"numero": "0612345678", "message": "hello"}, "phone")exec_action("record_audio", {"duration": 10}, "phone")exec_action("open_file", {"path": "/home/user/doc.pdf"}, "desk")exec_action("get_battery", {}, "phone")exec_action("set_volume", {"level": 50}, "desk")
5. Exemples de commandes vocales pour Hermes
- "Liste mes appareils"
- "Prends un screenshot du desk"
- "Envoie un SMS au 06 12 34 56 78 depuis le phone : je serai en retard"
- "Quel est le niveau de batterie du phone ?"
- "Lance Spotify sur le laptop"
- "Mets le volume à 50 sur le desk"
- "Ouvre le fichier rapport.pdf sur le laptop"
- "Active la capability run_terminal sur desk"
- "Désactive get_location sur phone"
6. Gestion des capabilities depuis l'orchestrateur (admin)
Modifier (merge) les capabilities d'un device :
curl -X PATCH http://localhost:8080/api/devices/phone/capabilities \
-H "Authorization: Bearer ${ORCHESTRATOR_ADMIN_KEY}" \
-H "Content-Type: application/json" \
-d '{"run_terminal": {"enabled": true, "auth_required": true}}'
Ajouter/modifier une seule capability :
curl -X POST http://localhost:8080/api/devices/desk/capabilities/run_terminal \
-H "Authorization: Bearer ${ORCHESTRATOR_ADMIN_KEY}" \
-H "Content-Type: application/json" \
-d '{"enabled": true, "auth_required": true}'
Supprimer une capability :
curl -X DELETE http://localhost:8080/api/devices/phone/capabilities/get_location \
-H "Authorization: Bearer ${ORCHESTRATOR_ADMIN_KEY}"
Au prochain heartbeat du device concerné, capabilities_refresh_needed: true
sera retourné et le device devra refaire un /register complet.
7. Endpoints admin (Hermes)
| Endpoint | Description |
|---|---|
GET /api/devices |
Liste tous les devices et leur statut |
GET /api/devices/{device_name} |
Détails d'un device |
GET /api/commands/{command_id} |
Statut d'une commande |
PATCH /api/devices/{device_name}/capabilities |
Merge de capabilities |
POST /api/devices/{device_name}/capabilities/{capability_name} |
Ajoute/modifie une capability |
DELETE /api/devices/{device_name}/capabilities/{capability_name} |
Supprime une capability |
Tous nécessitent Authorization: Bearer ${ORCHESTRATOR_ADMIN_KEY}.
8. Ajouter un nouveau type d'appareil
- Choisir un
device_typeparmimobile_agent,desktop_agent,laptop_agent(ou étendre l'enumDeviceTypedansmodels.pypour un nouveau type, ex.tablet_agent). - Implémenter côté agent :
/register,/heartbeat, boucleGET /commandsPOST /results(+/confirmsi des capabilitiesauth_required).
- Définir les
capabilitiespertinentes pour ce type d'appareil. Aucune modification de l'orchestrateur n'est nécessaire — les actions inconnues sont routées automatiquement viaexec_action. - Enregistrer le device : il apparaîtra automatiquement dans
device_list()et sera routable par nom ou par capability.
9. Logs
journalctl --user -u hasan-orchestrator -f
Niveau de log configurable via LOG_LEVEL dans .env (DEBUG, INFO,
ERROR).
10. Architecture des fichiers
hasan-orchestrator/
main.py # FastAPI : endpoints REST devices + admin
registry.py # Gestion SQLite des devices (registry persistant)
command_queue.py # File de commandes en mémoire avec TTL
mcp_server.py # Serveur MCP (stdio) exposé à Hermes
auth.py # Validation des Bearer tokens (admin + sessions device)
models.py # Schémas Pydantic
config.py # Configuration (.env)
install.sh # Installation + service systemd
requirements.txt
.env.example
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.