Saltar al contenido principal

Verificar firma

La cabecera MONEI-Signature incluida en cada solicitud firmada contiene una marca de tiempo y una o más firmas. La marca de tiempo va precedida del prefijo t=, y cada firma va precedida de un esquema. Los esquemas comienzan con v, seguido de un número entero. Actualmente, el único esquema de firma en producción válido es v1.

MONEI-Signature: t=1492774577,v1=5257a869e7ecebeda32affa62cdca3fa51cad7e77a0e56ff536d0ce8e108d8bd

MONEI genera firmas utilizando un código de autenticación de mensajes basado en hash (HMAC) con SHA-256. Para protegerte contra ataques de degradación, debes ignorar todos los esquemas que no sean v1.

Verificación de firmas con nuestras bibliotecas oficiales

Usa una de nuestras bibliotecas oficiales para verificar las firmas. La verificación se realiza proporcionando el payload de la solicitud y la cabecera MONEI-Signature. Si la verificación falla, MONEI devuelve un error.

server.js
import express from 'express';
import {Monei, PaymentStatus} from '@monei-js/node-sdk';

// Set your api key. Remember to switch to your live api key in production!
// See your api key here: https://dashboard.monei.com/settings/api
const monei = new Monei('YOUR_API_KEY');

// This example uses Express to receive webhooks
const app = express();

// Endpoint to handle MONEI webhooks
app.post('/checkout/callback', express.raw({type: 'application/json'}), (req, res) => {
const signature = req.headers['MONEI-Signature'];

try {
// Verify the signature using the raw request body and the header
const payment = monei.verifySignature(req.body.toString(), signature);

// Optional: Log the received payment status
console.log(`Webhook received for Payment ID: ${payment.id}, Status: ${payment.status}`);

// Update your order status based on the payment status
if (payment.status === PaymentStatus.SUCCEEDED) {
// Payment successful - fulfill the order
console.log(`Payment ${payment.id} succeeded. Fulfilling order...`);
// Update your database, send confirmation email, etc.
} else if (payment.status === PaymentStatus.FAILED) {
// Payment failed - notify the customer
console.log(`Payment ${payment.id} failed. Notifying customer...`);
// Log the failure, update your database, etc.
} else if (payment.status === PaymentStatus.AUTHORIZED) {
// Payment is authorized but not yet captured
console.log(`Payment ${payment.id} authorized. Capture if needed.`);
// You might want to capture it later using monei.payments.capture()
} else if (payment.status === PaymentStatus.CANCELED) {
// Payment was canceled by the user or system
console.log(`Payment ${payment.id} was canceled.`);
// Update your database accordingly
} else {
// Handle other potential statuses if necessary
console.log(`Unhandled payment status: ${payment.status} for Payment ${payment.id}`);
}

// Acknowledge receipt of the webhook with a 200 OK status
res.status(200).json({received: true});
} catch (error) {
// Handle signature verification failure
console.error('Invalid webhook signature:', error.message);
// Respond with 401 Unauthorized if the signature is invalid
res.status(401).json({error: 'Invalid signature'});
}
});

// Start the server
app.listen(3000, () => {
console.log(`Server listening on port 3000`);
});

Verificación manual de firmas

Paso 1: Extrae la marca de tiempo y la firma de la cabecera

Divide la cabecera usando el carácter , como separador para obtener una lista de elementos. Luego divide cada elemento usando el carácter = como separador para obtener un par prefijo-valor.

El valor del prefijo t corresponde a la marca de tiempo, y v1 corresponde a la firma (o firmas). Puedes descartar todos los demás elementos.

Paso 2: Prepara la cadena signed_payload

La cadena signed_payload se crea concatenando:

  • La marca de tiempo (como cadena de texto)
  • El carácter .
  • El payload JSON real (es decir, el cuerpo de la solicitud)

Paso 3: Determina la firma esperada

Calcula un HMAC con la función hash SHA256. Usa la clave de API de tu cuenta como clave y la cadena signed_payload como mensaje.

Puedes obtener la contraseña de tu cuenta en MONEI Dashboard → Ajustes → API.

Paso 4: Compara las firmas

Compara la firma de la cabecera con la firma esperada. Para una coincidencia exacta, calcula la diferencia entre la marca de tiempo actual y la marca de tiempo recibida, y decide si la diferencia está dentro de tu margen de tolerancia.

Para protegerte contra ataques de temporización, usa una comparación de cadenas en tiempo constante para comparar la firma esperada con la firma recibida.