Documenta el EP de ingesta y los webhooks salientes
api-docs/README.md: método/URL, auth Bearer, esquema del cuerpo (items foto/texto, zona, momento, flags), respuesta y errores (401/404/422), ejemplos curl de los 5 casos, y los payloads de los 3 webhooks salientes (perfil, entrega WhatsApp, arranque WhatsApp) con cómo el flujo externo devuelve las "después" por el mismo EP. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
163
mvp/b2c/api-docs/README.md
Normal file
163
mvp/b2c/api-docs/README.md
Normal file
@@ -0,0 +1,163 @@
|
||||
# API — Ingesta del perfil del lead
|
||||
|
||||
Endpoint único y asíncrono para enriquecer el perfil de un lead del funnel B2C. Cada llamada puede
|
||||
traer **imágenes, imagen+texto o solo texto**, etiquetado por **zona** y, en fotos, por **momento**
|
||||
(antes/después). Lo usan tanto el formulario web como los flujos externos (agente de llamada, bot de
|
||||
WhatsApp, generador de renders).
|
||||
|
||||
## Endpoint
|
||||
|
||||
```
|
||||
POST /api/leads/:id/ingesta
|
||||
```
|
||||
|
||||
- `:id` = UUID del lead (el que crea el funnel al capturar nombre/teléfono/email).
|
||||
- **Auth:** header `Authorization: Bearer <FUNNEL_API_KEY>`. Sin él, o con clave incorrecta → `401`.
|
||||
- **Content-Type:** `application/json`.
|
||||
|
||||
## Cuerpo (JSON)
|
||||
|
||||
| Campo | Tipo | Notas |
|
||||
| --- | --- | --- |
|
||||
| `items` | array | Lista de items `foto` o `texto`. Por defecto `[]`. |
|
||||
| `perfilCompleto` | boolean | Opcional. `true` señala al flujo externo que genere renders/agente. |
|
||||
| `finalizar` | boolean | Opcional. `true` construye el PDF y lo entrega (email + señal WhatsApp). |
|
||||
|
||||
Debe llegar **al menos un item o un flag**; una llamada totalmente vacía → `422`.
|
||||
|
||||
### Item `foto`
|
||||
|
||||
| Campo | Tipo | Notas |
|
||||
| --- | --- | --- |
|
||||
| `tipo` | `"foto"` | Obligatorio. |
|
||||
| `imagen` | string | Obligatorio. Data URI (`data:image/...;base64,...`) o URL `http(s)`. |
|
||||
| `zona` | enum | Opcional: `cocina`,`bano`,`salon`,`comedor`,`integral`,`otro`. Si falta, se asume el tipo de reforma del lead. |
|
||||
| `momento` | `"antes"`\|`"despues"` | Por defecto `"antes"`. Los renders del flujo externo se mandan como `despues`. |
|
||||
| `orden` | number | Opcional. Si falta, continúa el máximo actual del lead. |
|
||||
|
||||
### Item `texto`
|
||||
|
||||
| Campo | Tipo | Notas |
|
||||
| --- | --- | --- |
|
||||
| `tipo` | `"texto"` | Obligatorio. |
|
||||
| `texto` | string | Obligatorio, no vacío. Ej. `"suelo premium"`. |
|
||||
| `zona` | enum | Opcional (mismo enum que en `foto`). |
|
||||
|
||||
## Respuesta
|
||||
|
||||
`200`:
|
||||
|
||||
```json
|
||||
{
|
||||
"ok": true,
|
||||
"fotos": 1,
|
||||
"notas": 1,
|
||||
"perfilSenalado": false,
|
||||
"finalizado": null
|
||||
}
|
||||
```
|
||||
|
||||
- `fotos` / `notas`: cuántos items de cada tipo se guardaron.
|
||||
- `perfilSenalado`: `true` si `perfilCompleto` disparó el webhook y respondió ok.
|
||||
- `finalizado`: `null` si no se pidió `finalizar`; si sí,
|
||||
`{ ok, emailEnviado, whatsappSenal }`.
|
||||
|
||||
### Códigos de error
|
||||
|
||||
| Código | Cuándo |
|
||||
| --- | --- |
|
||||
| `401` | Falta `Authorization: Bearer`, o la clave no coincide con `FUNNEL_API_KEY`. |
|
||||
| `404` | El lead `:id` no existe. |
|
||||
| `422` | JSON inválido, item mal formado, o llamada vacía (sin items ni flag). |
|
||||
|
||||
## Ejemplos (curl)
|
||||
|
||||
Sustituye `LEAD`, `KEY` y el host según tu entorno.
|
||||
|
||||
```bash
|
||||
# 1) Solo texto: añade un dato a la zona baño
|
||||
curl -X POST "$HOST/api/leads/$LEAD/ingesta" \
|
||||
-H "Authorization: Bearer $KEY" -H "Content-Type: application/json" \
|
||||
-d '{"items":[{"tipo":"texto","zona":"bano","texto":"suelo premium"}]}'
|
||||
|
||||
# 2) Solo imagen (antes)
|
||||
curl -X POST "$HOST/api/leads/$LEAD/ingesta" \
|
||||
-H "Authorization: Bearer $KEY" -H "Content-Type: application/json" \
|
||||
-d '{"items":[{"tipo":"foto","zona":"bano","imagen":"https://cdn/ejemplo/bano-antes.jpg"}]}'
|
||||
|
||||
# 3) Imagen + texto en la misma llamada
|
||||
curl -X POST "$HOST/api/leads/$LEAD/ingesta" \
|
||||
-H "Authorization: Bearer $KEY" -H "Content-Type: application/json" \
|
||||
-d '{"items":[
|
||||
{"tipo":"foto","zona":"cocina","imagen":"data:image/jpeg;base64,..."},
|
||||
{"tipo":"texto","zona":"cocina","texto":"encimera de cuarzo"}
|
||||
]}'
|
||||
|
||||
# 4) Perfil completo: que el flujo externo genere renders / corra el agente
|
||||
curl -X POST "$HOST/api/leads/$LEAD/ingesta" \
|
||||
-H "Authorization: Bearer $KEY" -H "Content-Type: application/json" \
|
||||
-d '{"perfilCompleto":true}'
|
||||
|
||||
# 5) Devolver renders "después" y finalizar (genera PDF + email + señal WhatsApp)
|
||||
curl -X POST "$HOST/api/leads/$LEAD/ingesta" \
|
||||
-H "Authorization: Bearer $KEY" -H "Content-Type: application/json" \
|
||||
-d '{"items":[{"tipo":"foto","zona":"cocina","momento":"despues","imagen":"https://cdn/render-cocina.jpg"}],"finalizar":true}'
|
||||
```
|
||||
|
||||
## Webhooks salientes (hacia el flujo externo)
|
||||
|
||||
La app **emite** estas señales; el flujo externo (n8n/generador) las recibe y, cuando toca,
|
||||
devuelve los resultados llamando de nuevo a este mismo EP (las "después" con `finalizar:true`).
|
||||
Cada webhook es opcional: sin su URL en el entorno, la señal simplemente no se manda.
|
||||
|
||||
### `PERFIL_WEBHOOK_URL` — perfil completo
|
||||
|
||||
Disparado por `perfilCompleto:true`. Payload:
|
||||
|
||||
```json
|
||||
{
|
||||
"leadId": "uuid",
|
||||
"cliente": { "nombre": "...", "telefono": "...", "email": "...", "provincia": "..." },
|
||||
"reforma": { "tipo": "cocina", "m2Suelo": 12, "calidad": "media",
|
||||
"estructural": false, "urgencia": "media", "presupuestoTarget": 800000 },
|
||||
"empresa": { "tenantId": "uuid", "nombre": "Reformas Ejemplo" },
|
||||
"zonas": [
|
||||
{ "zona": "cocina", "notas": ["encimera de cuarzo"],
|
||||
"fotos": { "antes": ["url", "..."], "despues": [] } }
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
> El flujo externo genera los renders y los devuelve como items `foto` con `momento:"despues"`
|
||||
> por `POST /api/leads/:id/ingesta`, y cierra con `finalizar:true`.
|
||||
|
||||
### `WHATSAPP_WEBHOOK_URL` — entrega del PDF
|
||||
|
||||
Disparado por `finalizar:true`. Payload:
|
||||
|
||||
```json
|
||||
{
|
||||
"leadId": "uuid",
|
||||
"telefono": "+34...",
|
||||
"nombre": "...",
|
||||
"empresa": "Reformas Ejemplo",
|
||||
"pdfBase64": "JVBERi0xLj...",
|
||||
"filename": "presupuesto-nombre.pdf"
|
||||
}
|
||||
```
|
||||
|
||||
### `WHATSAPP_START_WEBHOOK_URL` — arranque de conversación
|
||||
|
||||
Disparado cuando el lead elige continuar por WhatsApp en el funnel. Payload:
|
||||
|
||||
```json
|
||||
{ "leadId": "uuid", "telefono": "+34...", "nombre": "...", "empresa": "Reformas Ejemplo" }
|
||||
```
|
||||
|
||||
## Notas
|
||||
|
||||
- **Storage:** las imágenes se guardan tal cual se reciben (data URI o URL) en `lead_fotos.url`;
|
||||
las notas en `lead_notas`. Ver `mvp/b2c/db-schema/`.
|
||||
- **Email:** la entrega por email es real vía SMTP (`SMTP_*` + `EMAIL_FROM`). Sin configurar,
|
||||
`finalizado.emailEnviado` será `false` y el evento queda marcado como simulado.
|
||||
- Variables de entorno: ver `mvp/b2c/.env.example`.
|
||||
Reference in New Issue
Block a user