<?php namespace App\Util\HttpSignatures; use Psr\Http\Message\RequestInterface; class Verification { /** @var RequestInterface */ private $message; /** @var KeyStoreInterface */ private $keyStore; /** @var array */ private $_parameters; /** * @param RequestInterface $message * @param KeyStoreInterface $keyStore */ public function __construct($message, KeyStoreInterface $keyStore) { $this->message = $message; $this->keyStore = $keyStore; } /** * @return bool */ public function isValid() { return $this->hasSignatureHeader() && $this->signatureMatches(); } /** * @return bool */ private function signatureMatches() { try { $key = $this->key(); switch ($key->getType()) { case 'secret': $random = random_bytes(32); $expectedResult = hash_hmac( 'sha256', $this->expectedSignatureBase64(), $random, true ); $providedResult = hash_hmac( 'sha256', $this->providedSignatureBase64(), $random, true ); return $expectedResult === $providedResult; case 'asymmetric': $signedString = new SigningString( $this->headerList(), $this->message ); $hashAlgo = explode('-', $this->parameter('algorithm'))[1]; $algorithm = new RsaAlgorithm($hashAlgo); $result = $algorithm->verify( $signedString->string(), $this->parameter('signature'), $key->getVerifyingKey()); return $result; default: throw new Exception("Unknown key type '".$key->getType()."', cannot verify"); } } catch (SignatureParseException $e) { return false; } catch (KeyStoreException $e) { return false; } catch (SignedHeaderNotPresentException $e) { return false; } } /** * @return string */ private function expectedSignatureBase64() { return base64_encode($this->expectedSignature()->string()); } /** * @return Signature */ private function expectedSignature() { return new Signature( $this->message, $this->key(), $this->algorithm(), $this->headerList() ); } /** * @return string */ private function providedSignatureBase64() { return $this->parameter('signature'); } /** * @return Key */ private function key() { return $this->keyStore->fetch($this->parameter('keyId')); } /** * @return HmacAlgorithm */ private function algorithm() { return Algorithm::create($this->parameter('algorithm')); } /** * @return HeaderList */ private function headerList() { return HeaderList::fromString($this->parameter('headers')); } /** * @param string $name * * @return string * * @throws Exception */ private function parameter($name) { $parameters = $this->parameters(); if (!isset($parameters[$name])) { throw new Exception("Signature parameters does not contain '$name'"); } return $parameters[$name]; } /** * @return array */ private function parameters() { if (!isset($this->_parameters)) { $parser = new SignatureParametersParser($this->signatureHeader()); $this->_parameters = $parser->parse(); } return $this->_parameters; } /** * @return bool */ private function hasSignatureHeader() { return $this->message->hasHeader('Signature') || $this->message->hasHeader('Authorization'); } /** * @return string * * @throws Exception */ private function signatureHeader() { if ($signature = $this->fetchHeader('Signature')) { return $signature; } elseif ($authorization = $this->fetchHeader('Authorization')) { return substr($authorization, strlen('Signature ')); } else { throw new Exception('HTTP message has no Signature or Authorization header'); } } /** * @param $name * * @return string|null */ private function fetchHeader($name) { // grab the most recently set header. $header = $this->message->getHeader($name); return end($header); } }