{
  "$type": "site.standard.document",
  "bskyPostRef": {
    "cid": "bafyreigahz7ijvxb43hixicnrgevbak62h7mrfyy3rcka277unmz6nmc74",
    "uri": "at://did:plc:25rdn5elo5izoxrmtis34zuk/app.bsky.feed.post/3mogkayz6bcx2"
  },
  "coverImage": {
    "$type": "blob",
    "ref": {
      "$link": "bafkreid3qorptyfjgkzzsypxiot2o6djexvsiatwn5urbyzgsskpi2ysuu"
    },
    "mimeType": "image/webp",
    "size": 91752
  },
  "path": "/edgardo_genini/es-como-organizar-el-arbol-de-apis-rest-para-que-sobreviva-al-tiempo-y-a-los-cambios-de-4e6g",
  "publishedAt": "2026-06-16T19:29:10.000Z",
  "site": "https://dev.to",
  "tags": [
    "api",
    "architecture",
    "spanish",
    "systemdesign",
    "También disponible en inglés",
    "En catálogo"
  ],
  "textContent": "También disponible en inglés\n\n> **Antes de diseñar un endpoint, clasificá su naturaleza. Recién después elegí su ubicación.**\n\n##  El problema silencioso\n\nExisten muchos artículos que explican cómo escribir una API REST. Aquí, en cambio, nos vamos a centrar en una pregunta previa: ¿cómo decidir **dónde debe vivir un endpoint** antes de escribirlo?\n\nCuando ese criterio no existe, la ubicación de cada nuevo endpoint termina dependiendo de la intuición, del precedente o de una negociación entre desarrolladores. El resultado suele ser un árbol de APIs que funciona al principio, pero que se vuelve cada vez más difícil de navegar y mantener.\n\nLos síntomas son conocidos:\n\n  * Endpoints casi iguales, creados por equipos distintos.\n  * Frontends que deben ensamblar múltiples llamadas para una sola pantalla.\n  * URLs que reflejan el organigrama y no la actividad principal de la organización.\n  * Convenciones implícitas que solo conocen quienes llevan años en el proyecto.\n\n\n\nNo suele haber un gran error arquitectónico. Hay cientos de pequeñas decisiones inconsistentes.\n\n##  La idea central\n\nEste artículo propone un criterio de clasificación **previo al diseño del endpoint**. La idea es sencilla: antes de pensar la URL, hay que entender qué tipo de necesidad estamos resolviendo.\n\nLa tesis que atraviesa todo el texto es:\n\n> **Antes de diseñar un endpoint, hay que clasificar su naturaleza. Recién después decidir dónde debe vivir.**\n\nNo se trata de una convención de nombres ni de una discusión más sobre REST. Se trata de un marco de decisión que puede aplicarse independientemente del estilo arquitectónico.\n\n##  Seis clasificaciones posibles\n\nEl framework establece seis categorías. Cada una responde a una naturaleza distinta de endpoint.\n\nClasificación | Ubicación sugerida\n---|---\nDominio Core | `/{dominio}/{recurso}`\nVista | `/vistas/{nombre}`\nIntegración |  `/integraciones/externos/...` y `/integraciones/webhooks/...`\nProcesos | `/procesos/{nombre}`\nConfiguración | `/configuracion/{tabla}`\nSistema | `/sistema/{operacion}`\n\nComo regla general, cuando un requerimiento parece encajar en más de una clasificación, conviene priorizar la más específica y restrictiva. **Estas seis categorías son el mecanismo mediante el cual resolvemos el problema de la falta de criterio.**\n\n##  Dominios Core: la fuente de verdad\n\nLos **Dominios Core** representan la fuente de verdad de la actividad principal. Contienen operaciones atómicas sobre sus entidades fundamentales. Habitualmente se corresponden con los conceptos centrales que maneja la organización: clientes, pedidos, productos, cuentas, etc.\n\nEjemplo:\n\n\n    GET /api/v1/ventas/pedidos/{id}\n    POST /api/v1/ventas/pedidos\n\n\nAquí las operaciones son atómicas, afectan a una sola entidad y respetan el principio de que el método HTTP define la acción.\n\n##  Vistas: desacoplar al consumidor\n\nLas Vistas no existen únicamente para simplificar consultas. Su propósito más valioso es **desacoplar al consumidor de la estructura interna de los dominios core**.\n\nUna pantalla, un reporte o un sistema externo no deberían necesitar conocer cómo están organizados internamente los dominios. Las Vistas actúan como una capa de desacople: si mañana un dominio se divide o se reorganiza, el consumidor puede seguir utilizando la misma Vista mientras el backend adapta su implementación.\n\nEjemplo:\n\n\n    GET /api/v1/vistas/resumen-ventas-por-cliente\n\n\nEse resumen puede combinar datos de pedidos, pagos y envíos. El frontend recibe exactamente lo que necesita, sin ensamblar llamadas ni conocer la estructura interna.\n\n##  Procesos: cuando el objetivo es ejecutar, no operar sobre una entidad\n\nLa categoría **Procesos** agrupa endpoints cuya naturaleza no es operar sobre una entidad del dominio core, sino **ejecutar una secuencia coordinada de acciones**. Puede tratarse de:\n\n  * Un cierre mensual\n  * Una conciliación bancaria\n  * Una reversión o compensación de una operación compleja\n  * Un recálculo de saldos\n  * Una liquidación\n  * Una sincronización de padrones\n  * Un proceso batch pesado\n  * Cualquier otra rutina que no encaje naturalmente en un CRUD de entidad\n\n\n\nEl nombre \"procesos\" es intencionalmente amplio. No asume masividad ni cantidad de entidades afectadas. Solo asume que el endpoint existe para **poner en marcha un proceso** , no para leer o modificar un recurso atómico.\n\nEjemplos:\n\n\n    POST /api/v1/procesos/cierre-mensual\n    POST /api/v1/procesos/revertir-operacion/{id}\n    POST /api/v1/procesos/conciliacion-bancaria\n\n\nLa pregunta que define esta categoría es:\n\n> **¿El endpoint existe para operar sobre una entidad del dominio core, o para ejecutar un proceso?**\n\nSi la respuesta es \"proceso\", su lugar es `/procesos`.\n\n##  Un ejemplo práctico: importar clientes\n\nSupongamos que aparece el siguiente requerimiento:\n\n> _\"El sistema debe permitir importar clientes desde un archivo Excel.\"_\n\nSin un criterio claro, distintos equipos podrían proponer ubicaciones diferentes para el endpoint:\n\n\n\n    POST /clientes/importacion\n    POST /clientes/importar\n    POST /admin/importacion-clientes\n    POST /batch/clientes/importacion\n\n\nTodas las alternativas parecen razonables, pero responden a intuiciones distintas. El resultado es que, con el tiempo, el árbol de APIs termina creciendo de forma inconsistente.\n\nAplicando el flujo de decisión de este framework:\n\n  * **¿Interactúa con un sistema externo?** → No.\n  * **¿Es una lectura que compone múltiples dominios?** → No.\n  * **¿El objetivo es operar sobre una entidad del dominio core?** → No. No se está creando, modificando ni consultando un cliente de forma atómica. El endpoint no recibe la representación de un recurso Cliente para operarlo de forma atómica; recibe un archivo binario (Excel) que requiere parsing, validaciones de lote y una lógica de coordinación pesada.\n  * **¿El objetivo es ejecutar un proceso?** → Sí.\n\n\n\nPor lo tanto, la clasificación correcta es **Procesos** , y una ubicación coherente sería:\n\n\n\n    POST /procesos/importacion-clientes\n\n\nLo importante es observar que la decisión **no depende de la entidad involucrada** , sino de la **naturaleza del endpoint**. Aunque el proceso trabaje sobre clientes, el endpoint existe para poner en marcha una importación, no para operar sobre el recurso `clientes`.\n\nEste ejemplo ilustra el objetivo principal del framework: que dos arquitectos distintos, enfrentados al mismo requerimiento, lleguen a la misma decisión siguiendo el mismo razonamiento, en lugar de depender de preferencias personales o convenciones implícitas.\n\n##  Integraciones, Configuración y Sistema\n\n  * **Integraciones externas** : para invocar sistemas externos (cliente activo). Ejemplo: `/integraciones/externos/salesforce/contactos`. El contrato lo define el tercero, y eso determina tanto la ubicación como la política de versionado: no es el ciclo de vida interno del sistema el que gobierna esta interfaz.\n  * **Webhooks** : para recibir notificaciones desde sistemas externos. Ejemplo: `/integraciones/webhooks/paypal/pago-completado`. La misma lógica aplica en sentido inverso: es el tercero quien impone la URL que debe exponerse, razón por la cual forzarlo dentro de `/api/v1/` sería una convención vacía.\n  * **Configuración** : para tablas paramétricas o globales. Ejemplo: `/configuracion/impuestos`. Estos datos existen como carril separado porque son transversales a varios dominios y no tienen un dueño de dominio claro; meterlos dentro de cualquier dominio core sería una decisión arbitraria.\n  * **Sistema** : para operaciones técnicas del propio API. Ejemplo: `/sistema/health`. La regla de inclusión es la ausencia de semántica de negocio: si el endpoint tiene sentido para el negocio, no pertenece acá.\n\n\n\n##  ¿De dónde salen los dominios? APQC y TM Forum\n\nDefinir los dominios de alto nivel suele ser un desafío. Inventar una taxonomía propia para cada organización puede llevar a discusiones interminables y a que el árbol termine reflejando el organigrama en lugar de la actividad real.\n\nPara evitarlo, el framework se apoya en **taxonomías maduras y ampliamente probadas** :\n\n  * **APQC PCF** (Process Classification Framework): ideal para organizaciones con operaciones físicas, logística, manufactura y procesos corporativos tradicionales.\n  * **TM Forum eTOM** (enhanced Telecom Operations Map): pensado para organizaciones de servicios digitales, telecomunicaciones, plataformas y suscripciones.\n\n\n\nApoyarse en estos modelos no es un detalle secundario. Es una de las mayores fortalezas del enfoque, porque reduce discusiones de nomenclatura y hace que el árbol refleje la realidad operativa.\n\nA modo de ilustración, algunos ejemplos de cómo se traducen estas taxonomías a URLs concretas:\n\nTaxonomía | Ejemplos de dominios y recursos\n---|---\n**APQC** |  `/finanzas/cuentas-contables`\n`/logistica/inventarios`\n`/comercial/oportunidades`\n**TM Forum** |  `/cliente/abonados`\n`/operaciones/facturacion`\n`/recursos/licencias`\n\n> _En catálogo se encuentran los árboles de contexto detallados basados en APQC PCF y TM Forum eTOM, con una lista mucho más extensa de dominios y recursos._\n\nRegla: APQC y TM Forum solo se utilizan para estructurar los dominios core. Las demás categorías son transversales y no siguen ninguna taxonomía de negocio, porque su naturaleza no es operar sobre entidades del dominio.\n\n##  Flujo de decisión\n\nPara aplicar la clasificación en el día a día, conviene seguir un flujo de decisión práctico:\n\n\n    ¿Interactúa con un sistema externo?\n    │\n    ├── Sí → ¿Es una llamada activa o un webhook?\n    │         ├── Llamada activa → /integraciones/externos/{sistema}\n    │         └── Webhook → /integraciones/webhooks/{origen}/{evento}\n    │\n    └── No → ¿Es una lectura que compone múltiples dominios?\n             │\n             ├── Sí → /vistas/{nombre}\n             │\n             └── No → ¿El objetivo es operar sobre una entidad del dominio core?\n                      │\n                      ├── Sí → /{dominio}/{recurso}\n                      │\n                      └── No → ¿El objetivo es ejecutar un proceso?\n                               │\n                               ├── Sí → /procesos/{nombre}\n                               │\n                               └── No → ¿Es configuración global?\n                                        │\n                                        ├── Sí → /configuracion/{tabla}\n                                        │\n                                        └── No → /sistema/{operacion}\n\n\nEste árbol no es un dogma. Es una herramienta práctica que ayuda a mantener consistencia cuando distintos equipos o personas diseñan nuevos endpoints.\n\n##  Algunas convenciones útiles\n\nMás allá de la clasificación, hay buenas prácticas que conviene adoptar:\n\n  * Usar `kebab-case` en toda la URL (ejemplo: `procesos/cierre-mensual`).\n  * Evitar verbos en la ruta; la acción la define el método HTTP.\n  * Mantener una semántica uniforme para los identificadores.\n\n\n\n**Sobre el prefijo`/api`**\n\nEl prefijo `/api` se utiliza para separar los endpoints de la interfaz programática de otros recursos del servidor (archivos estáticos, páginas administrativas, etc.). No es obligatorio, pero es una práctica ampliamente extendida que permite aplicar políticas específicas (autenticación, rate limiting, logging) sobre todo el árbol de la API de forma consistente. En este artículo se asume su uso, pero cada organización puede decidir si lo incluye o no.\n\n**Sobre el versionado**\n\nLa convención `/api/v1/` al inicio de la URL es útil cuando todos los endpoints de una categoría comparten el mismo ciclo de vida. Esto aplica naturalmente a **Dominios Core** y **Vistas**.\n\nSin embargo, otras categorías responden a contratos cuyo propietario y evolución son externos al sistema. Por ejemplo:\n\n  * Un webhook de PayPal o Stripe impone la URL que debe exponerse. No tiene sentido forzarlo dentro de `/api/v1/`.\n  * Una integración activa con Salesforce puede necesitar su propio versionado, por ejemplo `/integraciones/externos/salesforce/v2/...`, independiente del versionado global.\n\n\n\n**Regla general** : el prefijo `/api/v1/` no debe aplicarse mecánicamente a todas las categorías. Cada clasificación puede definir su política de versionado según su naturaleza. Para integraciones y webhooks, la estrategia de versionado se deriva del contrato que gobierna esa interfaz, no necesariamente del versionado global de la API.\n\n##  Para cerrar\n\nUna buena arquitectura de APIs no consiste en encontrar un lugar para cada endpoint. Consiste en tener un criterio que haga evidente cuál es ese lugar **antes de escribirlo**.\n\nCuando dos arquitectos distintos llegan a la misma decisión siguiendo el mismo razonamiento, el árbol deja de depender de la intuición y pasa a formar parte de la arquitectura del sistema. Las discusiones sobre dónde colocar un nuevo endpoint se vuelven rápidas, predecibles y fundamentadas.\n\n_Este trabajo pretende ser un punto de partida. Más que establecer reglas inmutables, busca proponer un criterio de clasificación que pueda enriquecerse con la experiencia y el debate de la comunidad. Ningún framework sobrevive sin evolución, y este tampoco._\n\n**Nota final** : la propuesta de las seis clasificaciones y el flujo de decisión es abierta. Si tu organización encuentra que una categoría adicional o una variante funciona mejor, el principio fundamental —clasificar antes de diseñar— sigue siendo lo que realmente importa.",
  "title": "[ES] Cómo organizar el árbol de APIs REST para que sobreviva al tiempo (y a los cambios de organigrama)"
}