vamischenko / decorators
PSR-7 stream decorators for WhatsApp media encryption and decryption
Requires
- php: ^8.1
- ext-hash: *
- ext-openssl: *
- guzzlehttp/psr7: ^2.0
- psr/http-message: ^1.1 || ^2.0
Requires (Dev)
- phpunit/phpunit: ^10.0
README
PSR-7 stream decorators for encrypting and decrypting WhatsApp media files using the WhatsApp AES-256-CBC algorithm.
Requirements
- PHP 8.1+
- ext-openssl
- ext-hash
- guzzlehttp/psr7 ^2.0
Installation
composer require vamischenko/decorators
Architecture
The encryption pipeline is built from composable PSR-7 stream decorators, inspired by jeskew/php-encrypted-streams (Apache 2.0). Since that package requires guzzlehttp/psr7 ~1.0 and is no longer maintained, its core classes were adapted and included directly.
source
→ AesEncryptingStream (AES-256-CBC, cipherKey, iv)
→ AppendStream([iv, ciphertext]) ← iv prepended so HMAC covers it
→ HashingStream (HMAC-SHA256, macKey)
Encryption is true streaming — no full-file buffering. Data is processed block by block as the stream is read.
Algorithm
- Expand the 32-byte
mediaKeyto 112 bytes using HKDF with SHA-256 (RFC 5869) - Split into
iv(16 bytes),cipherKey(32),macKey(32),refKey(32) - Encrypt with AES-256-CBC + PKCS7 padding
- Compute HMAC-SHA256 over
iv + ciphertext, truncate to 10 bytes - Output:
[ciphertext][mac]
Media-type-specific HKDF info strings:
| Type | Info string |
|---|---|
| IMAGE | WhatsApp Image Keys |
| VIDEO | WhatsApp Video Keys |
| AUDIO | WhatsApp Audio Keys |
| DOCUMENT | WhatsApp Document Keys |
Usage
Encryption
use GuzzleHttp\Psr7\Utils; use Vamischenko\Decorators\KeyExpander; use Vamischenko\Decorators\MediaKey; use Vamischenko\Decorators\MediaType; use Vamischenko\Decorators\Sidecar\SidecarContext; use Vamischenko\Decorators\Stream\EncryptingStream; $mediaKey = MediaKey::generate(); // or MediaKey::fromBinary($existingKey) $expanded = (new KeyExpander())->expand($mediaKey, MediaType::IMAGE); $source = Utils::streamFor(fopen('photo.jpg', 'rb')); $sidecar = new SidecarContext($expanded->macKey); // optional, for VIDEO/AUDIO $stream = new EncryptingStream($source, $expanded, $sidecar); // Read the encrypted stream — data is processed incrementally, not buffered file_put_contents('photo.jpg.enc', (string) $stream); // For streamable media, retrieve the sidecar after reading the full stream $sidecarBytes = $sidecar->getSidecar();
Decryption
use GuzzleHttp\Psr7\Utils; use Vamischenko\Decorators\KeyExpander; use Vamischenko\Decorators\MediaKey; use Vamischenko\Decorators\MediaType; use Vamischenko\Decorators\Stream\DecryptingStream; $mediaKey = MediaKey::fromBinary($keyBytes); $expanded = (new KeyExpander())->expand($mediaKey, MediaType::IMAGE); $source = Utils::streamFor(fopen('photo.jpg.enc', 'rb')); $stream = new DecryptingStream($source, $expanded); // MAC is verified before any bytes are returned $plaintext = (string) $stream;
Exceptions
InvalidMediaKeyException— thrown when the key is not exactly 32 bytesMacVerificationException— thrown when the HMAC does not match (corrupt or tampered data)
Security
- MAC verification uses constant-time
hash_equals()to prevent timing attacks - Integrity is verified before any plaintext is returned
- Key derivation follows RFC 5869 (HKDF) with WhatsApp-specific info strings
Memory behaviour
| Stream type | Encryption | Decryption |
|---|---|---|
| Seekable (file) | Incremental — O(block) memory | Incremental — O(block) memory (MAC read via seek, then streaming AesDecryptingStream) |
| Non-seekable (HTTP) | Incremental — O(block) memory | Full ciphertext buffered (MAC is at the tail), then plaintext delivered incrementally — no second copy |
For memory-constrained environments with non-seekable sources, wrap the response body in a temp-file stream before passing it to DecryptingStream.
Sidecar
The sidecar enables random-offset decryption for VIDEO and AUDIO streams, allowing players to seek without downloading the full file.
It is generated during encryption with no additional reads from the source stream. Each 10-byte entry is the HMAC-SHA256 of the [n*64K, (n+1)*64K+16] slice of the logical combined buffer iv + ciphertext + mac, truncated to 10 bytes.
Running tests
composer install vendor/bin/phpunit