# 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](https://en.wikipedia.org/wiki/HMAC)) con [SHA-256](https://en.wikipedia.org/wiki/SHA-2). Para protegerte contra [ataques de degradación](https://en.wikipedia.org/wiki/Downgrade_attack), debes ignorar todos los esquemas que no sean `v1`.

## Verificación de firmas con nuestras bibliotecas oficiales[​](#verificación-de-firmas-con-nuestras-bibliotecas-oficiales "Enlace directo al 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.

<!-- -->

* Node.js
* PHP
* Python

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`);

});
```

server.php

```
<?php

require_once(__DIR__ . '/vendor/autoload.php');



use Monei\\Model\\PaymentStatus;

use Monei\\ApiException;



// Replace YOUR_API_KEY with your actual MONEI API key

$monei = new Monei\\MoneiClient('YOUR_API_KEY');



// Parse raw body for signature verification

$rawBody = file_get_contents('php://input');

$signature = $_SERVER['HTTP_MONEI_SIGNATURE'] ?? '';



try {

    // Verify the signature

    // verifySignature returns the validated Payment object or throws ApiException

    $payment = $monei->verifySignature($rawBody, $signature);



    // Optional: Log the received payment status

    error_log('Webhook received for Payment ID: ' . $payment->getId() . ', Status: ' . $payment->getStatus());



    // Update your order status based on the payment status

    if ($payment->getStatus() === PaymentStatus::SUCCEEDED) {

        // Payment successful - fulfill the order

        error_log('Payment ' . $payment->getId() . ' succeeded. Fulfilling order...');

        // Update your database, send confirmation email, etc.

    } else if ($payment->getStatus() === PaymentStatus::FAILED) {

        // Payment failed - notify the customer

        error_log('Payment ' . $payment->getId() . ' failed. Notifying customer...');

        // Log the failure, update your database, etc.

    } else if ($payment->getStatus() === PaymentStatus::AUTHORIZED) {

        // Payment is authorized but not yet captured

        error_log('Payment ' . $payment->getId() . ' authorized. Capture if needed.');

        // You can capture it later using $monei->payments->capture(...)

    } else if ($payment->getStatus() === PaymentStatus::CANCELED) {

        // Payment was canceled

        error_log('Payment ' . $payment->getId() . ' was canceled.');

        // Update your database accordingly

    } else {

        // Handle other potential statuses if necessary

        error_log('Unhandled payment status: ' . $payment->getStatus() . ' for Payment ' . $payment->getId());

    }



    // Acknowledge receipt of the webhook with a 200 OK status

    http_response_code(200);

    header('Content-Type: application/json'); // Ensure JSON header

    echo json_encode(['received' => true]);



} catch (ApiException $e) {

    // Handle signature verification failure (ApiException specifically)

    error_log('Invalid webhook signature: ' . $e->getMessage());

    http_response_code(401); // Respond with 401 Unauthorized

    header('Content-Type: application/json'); // Ensure JSON header

    echo json_encode(['error' => 'Invalid signature']);

    exit(); // Stop script execution after sending error response

} catch (Exception $e) {

    // Handle any other unexpected errors during processing

    error_log('Webhook processing error: ' . $e->getMessage());

    http_response_code(500); // Respond with 500 Internal Server Error

    header('Content-Type: application/json');

    echo json_encode(['error' => 'Internal server error']);

    exit();

}

?>
```

server.py

```
import os

import Monei

from Monei.errors import SignatureVerificationError

from flask import Flask, request, abort, jsonify



# Replace YOUR_API_KEY with your actual MONEI API key

monei_client = Monei.MoneiClient(api_key="YOUR_API_KEY")



app = Flask(__name__)



# Update route and function name

@app.route('/checkout/callback', methods=['POST'])

def callback():

    signature = request.headers.get('MONEI-Signature') # Use correct header key



    # Check if signature header exists (optional but good practice)

    if not signature:

        print("MONEI-Signature header missing")

        return jsonify({'error': 'Missing MONEI-Signature header'}), 400



    try:

        # Verify the signature using raw request data (bytes)

        payment = monei_client.verify_signature(request.data, signature)



        # Optional: Log the received payment status

        payment_id = payment.get('id')

        payment_status = payment.get('status')

        print(f"Webhook received for Payment ID: {payment_id}, Status: {payment_status}")



        # Update your order status based on the payment status

        if payment_status == 'SUCCEEDED':

            # Payment successful - fulfill the order

            print(f"Payment {payment_id} succeeded. Fulfilling order...")

            # Update your database, send confirmation email, etc.

            pass # Placeholder from user snippet

        elif payment_status == 'FAILED':

            # Payment failed - notify the customer

            print(f"Payment {payment_id} failed. Notifying customer...")

            # Log the failure, update your database, etc.

            pass # Placeholder from user snippet

        elif payment_status == 'AUTHORIZED':

            # Payment is authorized but not yet captured

            print(f"Payment {payment_id} authorized. Capture if needed.")

            # You can capture it later

            pass # Placeholder from user snippet

        elif payment_status == 'CANCELED':

            # Payment was canceled

            print(f"Payment {payment_id} was canceled.")

            pass # Placeholder from user snippet

        else:

             print(f"Unhandled payment status: {payment_status} for Payment {payment_id}")



        # Acknowledge receipt of the webhook

        return jsonify({'received': True}), 200



    except SignatureVerificationError as e: # Catch specific signature error

        print(f"Invalid webhook signature: {e}")

        # Return 401 Unauthorized as requested

        return jsonify({'error': 'Invalid signature'}), 401

    except Exception as e: # Catch other potential errors during processing

        print(f"Webhook processing error: {e}")

        # Return 500 Internal Server Error for other issues

        return jsonify({'error': 'Internal server error'}), 500



if __name__ == '__main__':

    app.run(port=3000)
```

## Verificación manual de firmas[​](#verificación-manual-de-firmas "Enlace directo al Verificación manual de firmas")

### Paso 1: Extrae la marca de tiempo y la firma de la cabecera[​](#paso-1-extrae-la-marca-de-tiempo-y-la-firma-de-la-cabecera "Enlace directo al 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`[​](#paso-2-prepara-la-cadena-signed_payload "Enlace directo al 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[​](#paso-3-determina-la-firma-esperada "Enlace directo al 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](https://dashboard.monei.com/settings/api).

### Paso 4: Compara las firmas[​](#paso-4-compara-las-firmas "Enlace directo al 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.
