Your IP : 216.73.217.77


Current Path : /home/users/unlimited/www/whatsapp-crm/vendor/resend/resend-php/src/
Upload File :
Current File : /home/users/unlimited/www/whatsapp-crm/vendor/resend/resend-php/src/WebhookSignature.php

<?php

namespace Resend;

use Resend\Exceptions\WebhookSignatureVerificationException;

final class WebhookSignature
{
    public static function verify(string $payload, array $headers, string $secret, ?int $tolerance = null): bool
    {
        $secret = static::getSecret($secret);

        if (isset($headers['svix-id']) && isset($headers['svix-timestamp']) && isset($headers['svix-signature'])) {
            $messageId = $headers['svix-id'];
            $messageTimestamp = $headers['svix-timestamp'];
            $messageSignature = $headers['svix-signature'];
        } else {
            throw new WebhookSignatureVerificationException('Missing required headers');
        }

        $timestamp = static::verifyTimestamp($messageTimestamp, $tolerance);
        $signature = static::sign($secret, $messageId, $timestamp, $payload);

        $expectedSignature = explode(',', $signature, 2)[1];
        $passedSignatures = explode(' ', $messageSignature);

        $signatureFound = false;

        foreach ($passedSignatures as $versionedSignature) {
            $signatureParts = explode(',', $versionedSignature, 2);

            $version = $signatureParts[0];
            if ($version !== 'v1') {
                continue;
            }

            $passedSignature = $signatureParts[1];
            if (hash_equals($expectedSignature, $passedSignature)) {
                $signatureFound = true;

                break;
            }
        }

        if (! $signatureFound) {
            throw new WebhookSignatureVerificationException('No signatures found matching the expected signature');
        }

        return true;
    }

    protected static function sign(string $secret, string $messageId, int $timestamp, string $payload): string
    {
        $toSign = "{$messageId}.{$timestamp}.{$payload}";
        $hash = hash_hmac('sha256', $toSign, $secret);
        $signature = base64_encode(pack('H*', $hash));

        return "v1,{$signature}";
    }

    /**
     * Determine the correct secret from the user provided secret string.
     */
    protected static function getSecret(string $secret): string
    {
        $prefix = 'whsec_';
        if (substr($secret, 0, strlen($prefix)) === $prefix) {
            $secret = substr($secret, strlen($prefix));
        }

        return base64_decode($secret);
    }

    /**
     * Verify the given timestamp with the set tolerance.
     */
    protected static function verifyTimestamp(string $timestamp, int $tolerance = 300)
    {
        if (! is_numeric($timestamp)) {
            throw new WebhookSignatureVerificationException('Invalid timestamp');
        }

        $now = time();
        $timestamp = (int) $timestamp;

        if ($timestamp < ($now - $tolerance)) {
            throw new WebhookSignatureVerificationException('Message timestamp too old');
        }

        if ($timestamp > ($now + $tolerance)) {
            throw new WebhookSignatureVerificationException('Message timestamp too new');
        }

        return $timestamp;
    }
}