Skip to main content

¿Cómo funcionan los webhooks de cobros?

Cuando cambia el estado de una sesión de cobro, Tu Pana envía un POST firmado a tu endpoint registrado. El sistema de webhooks es el mismo que usan el resto de las APIs — si ya tienes uno configurado, solo necesitas suscribirte a los nuevos eventos. Para registrar o gestionar tus endpoints de webhook, consulta la sección de Webhooks.

Eventos disponibles

EventoCuándo se emite
payment_request.paidEl pago fue confirmado exitosamente. Único evento que se dispara en modo cobro directo.
payment_request.document_issuedEl DTE fue emitido (automáticamente o vía PATCH /v1/payment-requests/{id}/ con status=issued). No se dispara en modo cobro directo.
payment_request.expiredLa sesión venció sin pagarse.

Estructura del payload

Todos los eventos de cobro siguen la estructura estándar de webhooks de Tu Pana:
{
  "event": "payment_request.paid",
  "created_at": "2026-04-15T14:03:45Z",
  "data": {
    "payment_request_id": 1042,
    "status": "paid",
    "external_id": "booking_abc123",
    "amount": 50000,
    "target_dte_type": "39",
    "payment_url": "https://www.tupana.ai/pagos/tok_xyz",
    "issued_document_id": 9871,
    "metadata": {},
    "document_ids": []
  }
}

Campos de data

CampoDescripción
payment_request_idID numérico de la sesión de cobro.
statusEstado resultante (paid, expired).
external_idTu ID propio para esta sesión. null si no enviaste uno.
amountMonto del cobro en CLP.
target_dte_typeTipo de DTE configurado para la sesión. null en modo cobro directo.
payment_urlURL de pago de la sesión.
issued_document_idID del DTE emitido. null si el DTE aún no fue emitido o si la sesión es modo cobro directo.
metadataDatos adicionales que enviaste al crear la sesión.
document_idsIDs de los documentos que esta sesión cobró (modo cobro directo). Lista vacía en modo emisión.

Ejemplo en modo cobro directo

{
  "event": "payment_request.paid",
  "created_at": "2026-04-15T14:03:45Z",
  "data": {
    "payment_request_id": 1099,
    "status": "paid",
    "external_id": "collection_session_001",
    "amount": 75000,
    "target_dte_type": null,
    "payment_url": "https://www.tupana.ai/pagos/tok_def",
    "issued_document_id": null,
    "metadata": { "campaign": "cobranza-abril" },
    "document_ids": [9871, 9872]
  }
}

Headers

Cada notificación incluye los siguientes headers:
Content-Type: application/json
X-Pana-Event-Type: payment_request.paid
X-Pana-Event-Id: <uuid>
X-Pana-Timestamp: 2026-04-15T14:03:45Z
X-Pana-Delivery: <delivery-uuid>
X-Pana-Signature: <firma-hmac>

Verificar la firma

Verifica que el webhook proviene realmente de Tu Pana usando el secret que recibiste al registrar el endpoint.
import hmac
import hashlib
import json

def verify_signature(payload, signature, secret):
    expected = hmac.new(
        secret.encode(),
        json.dumps(payload, sort_keys=True).encode(),
        hashlib.sha256
    ).hexdigest()

    return hmac.compare_digest(expected, signature)

# En tu endpoint:
signature = request.headers.get('X-Pana-Signature')
is_valid = verify_signature(request.json, signature, os.environ['TUPANA_WEBHOOK_SECRET'])

if not is_valid:
    return {'error': 'Invalid signature'}, 401
const crypto = require('crypto');

function verifySignature(payload, signature, secret) {
  const expected = crypto
    .createHmac('sha256', secret)
    .update(JSON.stringify(payload))
    .digest('hex');

  return crypto.timingSafeEqual(
    Buffer.from(expected),
    Buffer.from(signature)
  );
}

const signature = req.headers['x-pana-signature'];
const isValid = verifySignature(req.body, signature, process.env.TUPANA_WEBHOOK_SECRET);

if (!isValid) {
  return res.status(401).json({ error: 'Invalid signature' });
}

Reintentos

Tu endpoint debe responder con 2xx en menos de 120 segundos. Si no responde o devuelve error, Tu Pana reintenta con backoff:
IntentoTiempo de espera
21 minuto después
35 minutos después
415 minutos después
51 hora después
66 horas después
Cada entrega tiene un X-Pana-Delivery único. Si recibes el mismo ID dos veces, es un reintento del mismo evento — tu endpoint debe ser idempotente.
Usa el evento webhook.test disponible en el panel de Tu Pana para verificar que tu endpoint está recibiendo correctamente antes de ir a producción.