sapmcp es un servidor MCP para operar y consultar sistemas SAP ECC / SAP S/4HANA mediante SAP NetWeaver RFC SDK, sin pyrfc, exponiendo tools, resources y prompts seguros para clientes LLM como Codex, Claude Desktop u otros hosts compatibles con MCP.
El proyecto está pensado para trabajo Basis/DevOps diario: conexión multi-sistema, llamadas RFC controladas, health checks, auditoría JSONL, resources cacheables y playbooks de operación en castellano.
Estado actual: modo lectura por defecto, multi-destination, tools Basis, health check agregado y auditoría local.
- Qué resuelve
- Arquitectura resumida
- Requisitos
- Librería SAP NetWeaver RFC SDK
- Instalación rápida
- Configurar un sistema SAP
- Añadir varios sistemas: DEV/QAS/PRD
- Configurar usuarios y contraseñas
- Seguridad y modo lectura
- Seguridad con LLMs locales
- Enterprise / piloto cliente
- Herramientas MCP
- Resources MCP
- Prompts MCP
- Ejemplos de prompts de demo
- Health check
- Auditoría
- Snapshot offline
- Documentación adicional
- Desarrollo y tests
Con sapmcp, un asistente LLM puede:
- Validar conectividad RFC contra uno o varios sistemas SAP.
- Ejecutar lecturas Basis frecuentes: dumps, syslog, locks, workprocesses, updates, colas RFC, jobs y auditoría de usuario.
- Lanzar un
sap_health_checkcon semáforo globalok/warn/crit/unknown. - Consultar resources cacheables como interfaces RFC, esquemas DDIC, sistema activo y destinos configurados.
- Ejecutar RFCs genéricas bajo una política de seguridad explícita.
- Registrar auditoría local sin contraseñas ni payload sensible.
- Preparar demos reproducibles para Claude Desktop, Codex u otros hosts MCP, con prompts funcionales, Basis y ABAP.
El servidor no incluye un LLM propio. Expone capacidades MCP para que el LLM anfitrión las use.
flowchart LR
C["Cliente MCP / LLM"] --> M["sapmcp FastMCP server"]
M --> T["Tools"]
M --> R["Resources cacheables"]
M --> P["Prompts operativos"]
T --> S["SafetyPolicy + Audit JSONL"]
R --> S
S --> X["SapRFCConnector ctypes"]
X --> N["SAP NetWeaver RFC SDK"]
N --> SAP["SAP ECC / S/4HANA"]
Módulos principales:
| Ruta | Función |
|---|---|
src/sapmcp/server.py |
Registro MCP de tools, resources y prompts. |
src/sapmcp/sap_rfc.py |
Bridge ctypes contra SAP NetWeaver RFC SDK. |
src/sapmcp/config.py |
Variables de entorno, destinos, librería SDK y SafetyPolicy. |
src/sapmcp/basis.py |
Tools Basis read-only. |
src/sapmcp/health.py |
sap_health_check agregado. |
src/sapmcp/resources.py |
Resources cacheables y namespace por destino. |
src/sapmcp/audit.py |
Auditoría JSONL. |
src/sapmcp/prompts.py |
Playbooks MCP en castellano. |
- Python
>=3.10. - SAP NetWeaver RFC SDK instalado localmente.
- Acceso de red desde la máquina donde corre
sapmcpal sistema SAP. - Usuario SAP técnico con permisos RFC de lectura adecuados.
SAP distribuye el SDK bajo licencia. No se incluye en el repositorio y no debe subirse a GitHub.
Debes tener disponible la librería nativa:
| Sistema operativo | Librería esperada |
|---|---|
| macOS | libsapnwrfc.dylib |
| Linux | libsapnwrfc.so |
| Windows | sapnwrfc.dll |
Ejemplos de rutas habituales:
/usr/local/sap/nwrfcsdk/lib
/opt/sap/nwrfcsdk/lib
~/nwrfcsdk/lib
C:\nwrfcsdk\lib
Guía completa:
docs/instalacion-sap-nwrfc-sdk.md— descarga oficial, instalación, Docker, verificación y motivos por los que el SDK no se redistribuye con el repo.
Esta es la pieza crítica del proyecto. sapmcp evita pyrfc, pero necesita el SAP NetWeaver RFC SDK instalado en la máquina donde corre el servidor MCP.
Página oficial de SAP:
SAP indica que la información de descarga de la versión 7.50 está en la SAP Note 2573790:
Normalmente hace falta un S-user con autorización de descarga de software. Por licencia/compliance, el SDK no se vendorizá en este repositorio. Instálalo localmente y configura:
SAP_NWRFC_LIB_DIR=/opt/sap/nwrfcsdk/libo en macOS:
SAP_NWRFC_LIB_DIR=/usr/local/sap/nwrfcsdk/libcd /Users/eduardoariasbravo/Developer/sapmcp
python3 -m venv .venv
source .venv/bin/activate
pip install -e .
cp .env.example .envSi quieres usar keyring para no guardar passwords en .env:
pip install -e '.[keyring]'En macOS/Linux puede hacer falta exportar la ruta del SDK antes de arrancar el servidor:
export DYLD_LIBRARY_PATH=/usr/local/sap/nwrfcsdk/lib:$DYLD_LIBRARY_PATH # macOS
export LD_LIBRARY_PATH=/usr/local/sap/nwrfcsdk/lib:$LD_LIBRARY_PATH # LinuxArranque local:
source .venv/bin/activate
sapmcpO directamente:
python -m sapmcp.serverLa configuración base usa variables clásicas SAP_*. Este bloque define el destino lógico default y preserva compatibilidad con despliegues previos.
SAP_ASHOST=host.sap.local
SAP_SYSNR=00
SAP_CLIENT=100
SAP_USER=RFC_USER
SAP_PASS=********
SAP_LANG=ES
SAP_NWRFC_LIB_DIR=/usr/local/sap/nwrfcsdk/libAlternativa con message server / logon group:
SAP_MSHOST=message-server.sap.local
SAP_R3NAME=S4D
SAP_GROUP=PUBLIC
SAP_CLIENT=100
SAP_USER=RFC_USER
SAP_PASS=********
SAP_LANG=ESSAProuter, si aplica:
SAP_SAPROUTER=/H/router.example.local/H/SNC, si aplica:
SAP_SNC_MODE=1
SAP_SNC_LIB=/path/to/libsapcrypto.dylib
SAP_SNC_QOP=8
SAP_SNC_MYNAME=p:CN=client, OU=...
SAP_SNC_PARTNERNAME=p:CN=server, OU=...sapmcp soporta múltiples destinos nombrados. Cada tool que abre conexión SAP acepta destination opcional.
SAPMCP_DESTINATIONS=DEV,QAS,PRD
SAPMCP_DEFAULT_DESTINATION=DEVEjemplo completo:
# Destino clásico retrocompatible: default
SAP_ASHOST=classic.sap.local
SAP_SYSNR=00
SAP_CLIENT=100
SAP_USER=RFC_CLASSIC
SAP_PASS=********
SAP_LANG=ES
# Destino DEV
SAP_DEV_ASHOST=dev.sap.local
SAP_DEV_SYSNR=01
SAP_DEV_CLIENT=110
SAP_DEV_USER=RFC_DEV
SAP_DEV_PASS=********
SAP_DEV_LANG=ES
# Destino QAS
SAP_QAS_ASHOST=qas.sap.local
SAP_QAS_SYSNR=02
SAP_QAS_CLIENT=120
SAP_QAS_USER=RFC_QAS
SAP_QAS_PASS=********
SAP_QAS_LANG=ES
# Destino PRD vía message server
SAP_PRD_MSHOST=msg-prd.sap.local
SAP_PRD_R3NAME=PRD
SAP_PRD_GROUP=PUBLIC
SAP_PRD_CLIENT=100
SAP_PRD_USER=RFC_PRD
SAP_PRD_PASS=********
SAP_PRD_LANG=ESUso desde MCP:
{
"tool": "sap_health_check",
"arguments": {"destination": "QAS", "profile": "standard"}
}Si omites destination, se usa:
SAPMCP_DEFAULT_DESTINATION, si está definido.- Si no, el destino clásico
defaultcon variablesSAP_*.
Crea en SAP un usuario técnico RFC por entorno, por ejemplo:
| Entorno | Usuario sugerido |
|---|---|
| DEV | RFC_SAPMCP_DEV |
| QAS | RFC_SAPMCP_QAS |
| PRD | RFC_SAPMCP_PRD |
Recomendaciones:
- Tipo de usuario: técnico/comunicación según política interna.
- No usar usuarios personales para operación diaria automatizada.
- Dar solo permisos read-only necesarios.
- Separar usuarios por entorno.
- En PRD, aplicar autorización más restrictiva y registrar responsable.
Permisos típicos a coordinar con Basis/Security:
S_RFCpara RFCs/BAPIs usadas.- Permisos de lectura DDIC/tablas para
RFC_READ_TABLEcuando proceda. - Permisos XBP para
BAPI_XBP_JOB_SELECTsi se usan jobs. - Permisos de administración/monitorización necesarios para
TH_*, syslog, enqueue, dumps, updates, según release y política.
SAP_DEV_USER=RFC_SAPMCP_DEV
SAP_DEV_PASS=********Instala el extra:
pip install -e '.[keyring]'Guarda password genérica para el usuario clásico:
export SAP_USER=RFC_USER
sapmcp-credentials setActiva keyring:
SAPMCP_USE_KEYRING=truePara destinos nombrados, el código busca primero service="sapmcp", user="DESTINO:USUARIO", y si no existe cae a user="USUARIO".
Ejemplo para guardar una password específica de DEV:
python - <<'PY'
import getpass
import keyring
keyring.set_password("sapmcp", "DEV:RFC_SAPMCP_DEV", getpass.getpass("Password DEV: "))
PYPor defecto, sapmcp arranca en modo conservador:
SAPMCP_READ_ONLY=true
SAPMCP_ALLOWED_RFC=
SAPMCP_ALLOW_DANGEROUS=false
SAPMCP_MAX_ROWS=200
SAPMCP_RFC_TIMEOUT=60Comportamiento:
- RFCs conocidas como lectura se permiten.
- RFCs con nombres peligrosos (
*CREATE*,*CHANGE*,*DELETE*,*UPDATE*,*POST*, etc.) se bloquean salvo confirmación doble. RFC_READ_TABLEse limita conSAPMCP_MAX_ROWS.SAPMCP_RFC_TIMEOUTcontrola el timeout deRfcInvoke.
Allowlist opcional:
SAPMCP_ALLOWED_RFC=RFC_PING,STFC_CONNECTION,RFC_READ_TABLE,RFC_GET_*,BAPI_*_GET*,Z_MI_RFC_LECTURAPara cambios reales —no recomendado salvo entorno controlado— se exige:
SAPMCP_READ_ONLY=false
SAPMCP_ALLOW_DANGEROUS=truey además la tool debe pasar confirm_dangerous=true.
Para sistemas SAP reales o datos de cliente, consulta la guía específica:
docs/seguridad-y-modelos-locales.md— consideraciones de seguridad, privacidad, modelo de control desapmcpy recomendaciones de modelos locales/open-weight.
La recomendación general es clara: aunque sapmcp opere en modo lectura, los datos SAP devueltos por las tools pueden ser sensibles. Para entornos reales, usa LLMs locales, on-prem o plataformas cloud privadas aprobadas; no envíes resultados SAP a LLMs públicos no autorizados.
El Sprint 2 añade documentación y estructura para presentar sapmcp a SAP Basis, Security, arquitectura y dirección sin cambiar la lógica crítica del servidor:
| Documento / artefacto | Uso |
|---|---|
docs/matriz-autorizaciones-sap.md |
Matriz por RFC/BAPI con riesgo, objeto de autorización probable y recomendación DEV/QAS/PRD. |
docs/rol-pfcg-recomendado.md |
Estrategia de roles Z_SAPMCP_READ_DEV, Z_SAPMCP_READ_QAS y Z_SAPMCP_READ_PRD. |
docs/politica-destinos.md |
Modelo recomendado de política por destino y convención futura de variables. |
docs/gobierno-llm-sap.md |
Gobierno LLM, compliance, trazabilidad, aprobación humana y prompt injection. |
docs/matriz-compatibilidad.md |
Estado validado/esperado/pendiente por plataforma SAP. |
sap_zrfc/README.md |
Estrategia ABAP companion para Z-RFCs read-only tipadas. |
docs/docker.md |
Ejecución Docker/demo local sin redistribuir el SAP NetWeaver RFC SDK. |
Recomendación enterprise: usar RFC_READ_TABLE solo en DEV/QAS controlado; en PRD preferir RFCs estándar certificadas o Z-RFCs read-only revisadas por ABAP/Security, con roles PFCG mínimos y auditoría.
| Tool | Uso |
|---|---|
sap_config_status(destination=None) |
Configuración activa sanitizada y destinos disponibles. |
sap_safety_check(function_name) |
Clasifica una RFC frente a la política de seguridad. |
sap_resources_invalidate(prefix=None) |
Limpia cache de resources. |
sap_audit_tail(n=50) |
Lee las últimas entradas de auditoría del día. |
sap_catalog_search(kind, pattern) |
Busca en snapshot offline sin tocar SAP. |
sap_prompt_protocol(prompt) |
Protocolo genérico para traducir una petición humana a pasos SAP seguros. |
| Tool | Uso |
|---|---|
sap_ping(destination=None) |
Prueba RFC_PING. |
sap_describe_rfc(function_name, destination=None) |
Describe interfaz con RFC_GET_FUNCTION_INTERFACE. |
sap_read_table(table_name, fields=None, where=None, rowcount=None, ..., destination=None) |
Lee tabla/vista vía RFC_READ_TABLE. |
sap_rfc_call(function_name, ..., destination=None) |
Llamada RFC genérica con import params, tablas y salidas explícitas. |
sap_search_rfc(prefix, limit=50, destination=None) |
Busca módulos RFC por prefijo en TFDIR. |
sap_read_table(where=...)se conserva por compatibilidad como modo avanzado. Las tools internas de Basis/catalogo construyenOPTIONScon helpers seguros para campos/literales; para uso normal evita WHERE libre y prefiere tools específicas o filtros construidos por el servidor.
| Tool | Transacción mental | Uso |
|---|---|---|
sap_get_short_dumps(date_from=None, date_to=None, user=None, destination=None) |
ST22 | Dumps ABAP por fecha/usuario. |
sap_get_syslog(date_from=None, date_to=None, severity=None, destination=None) |
SM21 | Syslog con fallback de RFC estándar. |
sap_get_locks(table=None, user=None, destination=None) |
SM12 | Locks activos. |
sap_get_workprocesses(server=None, destination=None) |
SM50/SM66 | Workprocesses por servidor o todos. |
sap_get_update_requests(status=None, user=None, destination=None) |
SM13 | Updates pendientes/erróneos. |
sap_get_rfc_queue(queue_type="trfc", destination=None) |
SM58/qRFC | tRFC, qRFC out, qRFC in. |
sap_get_jobs(top_n=50, status=None, since_days=1, destination=None) |
SM37 | Jobs por estado y ventana. |
sap_get_user_audit(user, destination=None) |
SU01/SUIM | Perfiles, roles, bloqueo, vigencia, SAP_ALL y fallos de logon. |
| Tool | Uso |
|---|---|
sap_health_check(destination=None, profile="standard") |
Dashboard JSON con semáforo y métricas agregadas. |
Resources cacheables. Admiten URI legacy y URI con destino.
La cache es in-memory y thread-safe: usa single-flight por clave para que, ante misses concurrentes del mismo resource, solo una llamada SAP/RFC cargue el dato y el resto de callers reutilice ese resultado.
| URI | TTL | Uso |
|---|---|---|
sap://destinations |
infinito | Lista destinos configurados, sin contraseñas ni rutas SDK. |
sap://system/info |
60s | Info del destino por defecto. |
sap://{destination}/system/info |
60s | Ping funcional STFC_CONNECTION + SAP params sanitizados. |
sap://policy/allowlist |
infinito | Política de seguridad activa. |
sap://{destination}/policy/allowlist |
infinito | Política namespaced por destino. |
sap://function/{name}/interface |
600s | Interfaz RFC cacheada. |
sap://{destination}/function/{name}/interface |
600s | Interfaz RFC por destino. |
sap://table/{name}/schema |
600s | Esquema DDIC. |
sap://{destination}/table/{name}/schema |
600s | Esquema DDIC por destino. |
sap://catalog/rfc?prefix=... |
300s | Catálogo RFC por prefijo. |
sap://{destination}/catalog/rfc?prefix=... |
300s | Catálogo RFC por destino. |
sap://catalog/snapshot |
archivo | Snapshot offline local. |
Ejemplos:
read_resource("sap://destinations")
read_resource("sap://DEV/system/info")
read_resource("sap://QAS/function/BAPI_USER_GET_DETAIL/interface")
read_resource("sap://DEV/table/T000/schema")
Los prompts son playbooks; no ejecutan acciones por sí mismos.
| Prompt | Uso |
|---|---|
basis_triage_sistema |
Triage inicial de sistema. |
inspeccionar_pedido_venta |
Inspección read-only de pedido SD. |
seguimiento_idoc |
Seguimiento read-only de IDoc. |
revisar_jobs_largos |
Revisión de jobs largos. |
informe_sociedad |
Informe financiero básico por sociedad/ejercicio. |
pre_change_check |
Checklist humano antes de cualquier RFC peligrosa. |
health_check_response |
Redacta en español una respuesta humana a partir del JSON de sap_health_check. |
El repositorio incluye una guía oficial de prompts listos para copiar y pegar:
docs/ejemplos-prompts.md— prompts endurecidos para demos funcionales, Basis, ABAP y management.
La guía está pensada para demostrar en directo la propuesta de valor de sapmcp:
- SAP como interfaz conversacional mediante MCP.
- Operación segura en modo
SAPMCP_READ_ONLY=true. - Cruce de tablas y RFCs estándar sin depender de
pyrfc. - Preflight, fallbacks y límites de lectura para evitar demos frágiles.
sap_health_check orquesta varias lecturas Basis. En perfiles standard y deep, usa paralelización con ThreadPoolExecutor(max_workers=4), pero cada check abre su propia conexión RFC.
| Perfil | Tope global | Checks |
|---|---|---|
quick |
2s | Ping, locks, workprocesses básicos. |
standard |
15s | quick + dumps 24h, updates, tRFC, jobs cancelados 24h. |
deep |
45s | standard + usuarios activos, T000, servidores, muestra SNAP/syslog si autorizado. |
Umbrales por defecto:
| Check | Warn | Crit | Variables |
|---|---|---|---|
| dumps 24h | 5 | 20 | SAPMCP_HC_DUMPS_WARN, SAPMCP_HC_DUMPS_CRIT |
| locks total | 50 | 200 | SAPMCP_HC_LOCKS_WARN, SAPMCP_HC_LOCKS_CRIT |
| updates pendientes | 1 | 10 | SAPMCP_HC_UPDATE_PENDING_WARN, SAPMCP_HC_UPDATE_PENDING_CRIT |
| updates ERR | 1 | 1 | SAPMCP_HC_UPDATE_ERR_WARN, SAPMCP_HC_UPDATE_ERR_CRIT |
| tRFC pendiente | 20 | 100 | SAPMCP_HC_TRFC_WARN, SAPMCP_HC_TRFC_CRIT |
| jobs abortados 24h | 1 | 5 | SAPMCP_HC_JOBS_ABORTED_WARN, SAPMCP_HC_JOBS_ABORTED_CRIT |
| workprocess PRIV | 1 | 2 | SAPMCP_HC_WP_PRIV_WARN, SAPMCP_HC_WP_PRIV_CRIT |
| workprocess stopped | 1 | 1 | SAPMCP_HC_WP_STOPPED_WARN, SAPMCP_HC_WP_STOPPED_CRIT |
Ejemplo:
{
"tool": "sap_health_check",
"arguments": {"destination": "DEV", "profile": "standard"}
}Salida abreviada:
{
"destination": "DEV",
"sid": "S4D",
"mandt": "100",
"profile": "standard",
"verdict": "warn",
"checks": [
{"name": "ping", "status": "ok", "value": {"RESPTEXT": "ok"}, "threshold": null, "duration_ms": 42.1, "error": null},
{"name": "jobs_aborted_24h", "status": "warn", "value": 1, "threshold": {"warn": 1, "crit": 5}, "duration_ms": 120.4, "error": null}
],
"summary": "DEV: verdict=warn; crit=0, warn=1, unknown=0. Revisar: jobs_aborted_24h."
}Las tools que tocan SAP se auditan en JSONL:
~/.sapmcp/audit-YYYYMMDD.jsonl
Cada línea incluye:
ts, tool, function, destination, params_hash, rc, duration_ms,
sid, mandt, user, dangerous, confirmed, error_key, error_message
No se guarda payload completo ni contraseñas. params_hash se calcula sobre parámetros redactados.
Consulta desde MCP:
{
"tool": "sap_audit_tail",
"arguments": {"n": 50}
}Puedes generar un catálogo local comprimido para búsquedas sin tocar SAP:
sapmcp-snapshot --page-size 500Genera en ~/.sapmcp/ un fichero tipo:
catalog-{SID}.json.gz
Búsqueda offline:
{
"tool": "sap_catalog_search",
"arguments": {"kind": "rfc", "pattern": "BAPI_USER"}
}Ejemplo genérico:
{
"mcpServers": {
"sapmcp": {
"command": "/Users/eduardoariasbravo/Developer/sapmcp/.venv/bin/sapmcp",
"cwd": "/Users/eduardoariasbravo/Developer/sapmcp"
}
}
}Si el cliente no carga variables de entorno del shell, define env explícito o usa .env en el cwd del proyecto.
docs/operation.md— manual básico de instalación, configuración y operación desde cero.docs/instalacion-sap-nwrfc-sdk.md— guía específica para descargar, instalar y verificar el SAP NetWeaver RFC SDK.docs/architecture.md— arquitectura técnica.docs/ejemplos-prompts.md— ejemplos oficiales de prompts para demos con Claude/Codex y otros hosts MCP.docs/seguridad-y-modelos-locales.md— seguridad operativa y recomendaciones de modelos locales para sapmcp.docs/matriz-autorizaciones-sap.md— matriz enterprise de RFC/BAPI, riesgos y autorizaciones SAP probables.docs/rol-pfcg-recomendado.md— propuesta de roles PFCG por entorno y proceso de construcción con trazas.docs/politica-destinos.md— modelo recomendado DEV/QAS/PRD y convención futura de políticas por destino.docs/gobierno-llm-sap.md— gobierno LLM, AI compliance, trazabilidad y mitigación de prompt injection.docs/matriz-compatibilidad.md— compatibilidad validada/esperada/pendiente por plataforma SAP.docs/docker.md— Docker/demo local con SDK SAP montado externamente.sap_zrfc/README.md— estrategia y plantillas ABAP companion para Z-RFCs read-only.docs/sapmcp-abap-docker-validation-2026-05-06.md— validación contra ABAP Docker.docs/sapmcp-abap-docker-validation-fase8-20260506.md— validación ampliada de multi-destination, Basis, health check, SafetyPolicy y snapshot.
Instalar editable:
source .venv/bin/activate
pip install -e .Ejecutar tests:
pytest -q
# o
.venv/bin/pytest -qResultado esperado actual:
68 passed
Eduardo Arias Bravo, Orjiva, Mayo 2026