xepozz / php-age
Installs: 0
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/xepozz/php-age
Requires
- php: ^8.1
- ext-sodium: *
Requires (Dev)
- phpstan/phpstan: ^2.1
- phpunit/phpunit: ^13
This package is auto-updated.
Last update: 2026-02-22 11:36:57 UTC
README
A pure PHP implementation of the age file encryption format, compatible with the age specification (v1).
Built entirely on PHP's ext-sodium (libsodium) — no external binary dependencies.
Features
- X25519 recipients — public-key encryption using Curve25519
- Scrypt passphrases — password-based encryption with configurable work factor
- Multiple recipients — encrypt a file for several public keys at once
- ASCII armor — PEM-style encoding/decoding
- Key generation — generate identity/recipient key pairs
- Cross-compatible — interoperable with age (Go), rage (Rust), typage (TypeScript), and other spec-compliant implementations
Requirements
- PHP 8.1+
ext-sodium(bundled with PHP since 7.2)
Installation
composer require xepozz/php-age
Usage
Encrypt with a recipient (public key)
use Xepozz\PhpAge\Encrypter; use Xepozz\PhpAge\Decrypter; $e = new Encrypter(); $e->addRecipient('age1tgyuvdlmpejqsdf847hevurz9szk7vf3j7ytfyqecgzvphvu2d8qrtaxl6'); $ciphertext = $e->encrypt('hello, world!'); $d = new Decrypter(); $d->addIdentity('AGE-SECRET-KEY-1RKH0DGHQ0FU6VLXX2VW6Y3W2TKK7KR4J36N9SNDXK75JHCJ3N6JQNZJF5J'); $plaintext = $d->decrypt($ciphertext); // "hello, world!"
Encrypt with a passphrase
use Xepozz\PhpAge\Encrypter; use Xepozz\PhpAge\Decrypter; $e = new Encrypter(); $e->setPassphrase('my-secret-passphrase'); $ciphertext = $e->encrypt('hello, world!'); $d = new Decrypter(); $d->addPassphrase('my-secret-passphrase'); $plaintext = $d->decrypt($ciphertext); // "hello, world!"
Generate a key pair
use Xepozz\PhpAge\Age; $identity = Age::generateIdentity(); // AGE-SECRET-KEY-1... $recipient = Age::identityToRecipient($identity); // age1...
Multiple recipients
use Xepozz\PhpAge\Encrypter; $e = new Encrypter(); $e->addRecipient('age1...'); $e->addRecipient('age1...'); $ciphertext = $e->encrypt('secret data');
ASCII armor
use Xepozz\PhpAge\Armor; $encoded = Armor::encode($binaryData); // -----BEGIN AGE ENCRYPTED FILE----- // ...base64... // -----END AGE ENCRYPTED FILE----- $decoded = Armor::decode($encoded);
Custom scrypt work factor
use Xepozz\PhpAge\Encrypter; $e = new Encrypter(); $e->setScryptWorkFactor(18); // default is 18 (2^18 = 262144 iterations) $e->setPassphrase('my-passphrase'); $ciphertext = $e->encrypt('data');
Testing
Run the full test suite:
vendor/bin/phpunit
Run fast tests only (excludes heavy scrypt RFC vectors):
vendor/bin/phpunit --exclude-group=slow
Run with code coverage:
XDEBUG_MODE=coverage vendor/bin/phpunit --exclude-group=slow --coverage-text
Code coverage
Code Coverage Report:
Summary:
Classes: 100.00% (13/13)
Methods: 100.00% (47/47)
Lines: 100.00% (465/465)
Xepozz\PhpAge\Age .................. Lines: 100.00%
Xepozz\PhpAge\Armor ............... Lines: 100.00%
Xepozz\PhpAge\Bech32 .............. Lines: 100.00%
Xepozz\PhpAge\Decrypter ........... Lines: 100.00%
Xepozz\PhpAge\Encrypter ........... Lines: 100.00%
Xepozz\PhpAge\Header .............. Lines: 100.00%
Xepozz\PhpAge\Scrypt .............. Lines: 100.00%
Xepozz\PhpAge\ScryptIdentity ...... Lines: 100.00%
Xepozz\PhpAge\ScryptRecipient ..... Lines: 100.00%
Xepozz\PhpAge\Stanza .............. Lines: 100.00%
Xepozz\PhpAge\Stream .............. Lines: 100.00%
Xepozz\PhpAge\X25519Identity ...... Lines: 100.00%
Xepozz\PhpAge\X25519Recipient ..... Lines: 100.00%
150 tests, 236 assertions — runs in ~0.4 seconds (excluding slow RFC scrypt vectors).
Specification and references
- age specification (v1) — the format this library implements
- age (Go) — the original reference implementation by Filippo Valsorda
- typage (TypeScript) — TypeScript implementation, used as the reference for this PHP port
- RFC 7914 — scrypt key derivation function
- RFC 7748 — X25519 key agreement
- RFC 8439 — ChaCha20-Poly1305 AEAD
License
MIT