getCEKSize() / 16); $result = openssl_encrypt($data, $this->getMode(), $k, OPENSSL_RAW_DATA, $iv); if ($result === false) { throw new RuntimeException('Unable to encrypt the content'); } $tag = $this->calculateAuthenticationTag($result, $cek, $iv, $aad, $encoded_protected_header); return $result; } #[Override] public function decryptContent( string $data, string $cek, string $iv, ?string $aad, string $encoded_protected_header, string $tag ): string { if (! $this->isTagValid($data, $cek, $iv, $aad, $encoded_protected_header, $tag)) { throw new RuntimeException('Unable to decrypt or to verify the tag.'); } $k = substr($cek, $this->getCEKSize() / 16); $result = openssl_decrypt($data, $this->getMode(), $k, OPENSSL_RAW_DATA, $iv); if ($result === false) { throw new RuntimeException('Unable to decrypt or to verify the tag.'); } return $result; } #[Override] public function getIVSize(): int { return 128; } protected function calculateAuthenticationTag( string $encrypted_data, string $cek, string $iv, ?string $aad, string $encoded_header ): string { $calculated_aad = $encoded_header; if ($aad !== null) { $calculated_aad .= '.' . Base64UrlSafe::encodeUnpadded($aad); } $mac_key = substr($cek, 0, $this->getCEKSize() / 16); $auth_data_length = strlen($encoded_header); $secured_input = implode('', [ $calculated_aad, $iv, $encrypted_data, pack('N2', ($auth_data_length / 2_147_483_647) * 8, ($auth_data_length % 2_147_483_647) * 8), ]); $hash = hash_hmac($this->getHashAlgorithm(), $secured_input, $mac_key, true); return substr($hash, 0, strlen($hash) / 2); } protected function isTagValid( string $encrypted_data, string $cek, string $iv, ?string $aad, string $encoded_header, string $authentication_tag ): bool { return hash_equals( $authentication_tag, $this->calculateAuthenticationTag($encrypted_data, $cek, $iv, $aad, $encoded_header) ); } abstract protected function getHashAlgorithm(): string; abstract protected function getMode(): string; }