paperscissorsandglue / laravel-encryption-at-rest
Laravel package for encrypting sensitive data at rest and decrypting it when in use - useful for GDPR, HIPAA, and other compliance requirements
Requires
- php: ^8.1
- illuminate/database: ^10.0|^11.0|^12.0
- illuminate/encryption: ^10.0|^11.0|^12.0
- illuminate/support: ^10.0|^11.0|^12.0
Requires (Dev)
- orchestra/testbench: ^8.0|^9.0|^10.0
- phpunit/phpunit: ^10.0|^11.0
This package is auto-updated.
Last update: 2025-04-02 23:53:31 UTC
README
A Laravel package for encrypting sensitive data at rest and automatically decrypting it when in use. Useful for regulatory compliance requirements like GDPR, HIPAA, and other data protection standards. Compatible with Laravel 10, 11, and 12.
Features
- ✅ Encrypt user identifiable data at rest and decrypt while being used
- ✅ Encrypt specific fields within JSON columns
- ✅ Special handling for email addresses with searchable indexes
- ✅ Seamless integration with Laravel's authentication system
- ✅ Compatible with Laravel notifications and other subsystems
- ✅ Command-line tools for migrating existing data
- ✅ Simple trait-based implementation for models
Requirements
- PHP 8.1 or higher
- Laravel 10.x, 11.x, or 12.x
Installation
You can install the package via composer:
composer require paperscissorsandglue/laravel-encryption-at-rest
After installation, publish the configuration file:
php artisan vendor:publish --tag=encryption-at-rest-config
Configuration
In your .env
file, you can optionally set a custom encryption key:
ENCRYPTION_AT_REST_KEY=your-secure-key-here
If not set, the package will use your application key for encryption.
Basic Usage
Add the Encryptable
trait to your model and define which attributes should be encrypted:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Paperscissorsandglue\EncryptionAtRest\Encryptable; class User extends Model { use Encryptable; /** * The attributes that should be encrypted. * * @var array */ protected $encryptable = [ 'email', 'phone', 'address', ]; }
That's it! The specified attributes will be automatically encrypted when saved to the database and decrypted when retrieved.
Encrypting JSON Fields
For JSON columns where you only want to encrypt certain fields within the JSON structure, use the EncryptableJson
trait:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Paperscissorsandglue\EncryptionAtRest\EncryptableJson; class UserProfile extends Model { use EncryptableJson; /** * The attributes that should have encrypted JSON fields. * * @var array */ protected $encryptableJson = [ 'preferences' => ['notification_email', 'backup_phone'], 'settings' => ['api_key', 'personal_token'], ]; /** * The attributes that should be cast. * * @var array */ protected $casts = [ 'preferences' => 'json', 'settings' => 'json', ]; }
With this setup, only the specified fields within your JSON structure will be encrypted while the rest of the JSON remains searchable.
Seamless Encryption/Decryption
This package uses smart dynamic getters and setters to handle all encryption and decryption transparently. All encrypted fields, including email, are automatically:
- Encrypted when saved to the database
- Decrypted when accessed through any means
- Properly handled in all Laravel subsystems
This universal approach means you don't need to write any special code to handle encryption - it just works:
// Regular attribute access $email = $user->email; // Automatically decrypted // Assignment $user->email = 'new@example.com'; // Will be encrypted on save // Laravel notifications work seamlessly $user->notify(new WelcomeNotification()); // Form requests and API responses work correctly return response()->json(['user' => $user]); // Eloquent serialization works properly $array = $user->toArray();
For JSON attributes with encrypted fields, the package also ensures seamless operation:
// If 'api_key' is encrypted within the preferences JSON $apiKey = $user->preferences['api_key']; // Automatically decrypted // Set values that will be encrypted automatically $user->preferences = [ 'api_key' => 'new-secret-key', 'public_setting' => 'not-encrypted' ];
Encrypted Email Authentication
This package provides special support for encrypting the email field while maintaining the ability to authenticate users by email. This is achieved by adding a searchable hash of the email (email_index
) that enables efficient lookup.
Setup Encrypted Email Authentication
- First, publish the migration to add the
email_index
column to your users table:
php artisan vendor:publish --tag=encryption-at-rest-migrations php artisan migrate
- Add the
HasEncryptedEmail
trait to your User model:
<?php namespace App\Models; use Illuminate\Foundation\Auth\User as Authenticatable; use Illuminate\Notifications\Notifiable; use Laravel\Sanctum\HasApiTokens; use Paperscissorsandglue\EncryptionAtRest\HasEncryptedEmail; class User extends Authenticatable { use HasApiTokens, HasEncryptedEmail, Notifiable; // ... existing model code }
- Update your auth configuration in
config/auth.php
to use the encrypted email user provider:
'providers' => [ 'users' => [ 'driver' => 'encrypted-email', 'model' => App\Models\User::class, ], ],
- For existing users, you'll need to regenerate the email index values and encrypt existing emails. Use the provided command:
# Run in dry-run mode first to see what would be changed php artisan encryption:encrypt-emails "App\Models\User" --dry-run # When ready, run the actual encryption (use --chunk=XX to set batch size) php artisan encryption:encrypt-emails "App\Models\User"
Or if you prefer, create a migration:
<?php use Illuminate\Database\Migrations\Migration; use Illuminate\Support\Facades\DB; use App\Models\User; return new class extends Migration { public function up(): void { // Rehash all existing user emails User::all()->each(function ($user) { $user->handleEmailEncryption(); $user->save(); }); } };
Querying Users by Email
With the HasEncryptedEmail
trait, you can still find users by their email:
// Find a user by email $user = User::findByEmail('user@example.com'); // Or use the scope $user = User::whereEmail('user@example.com')->first();
How It Works
-
When a user is created or updated, the email is:
- Encrypted before storage in the
email
column - A deterministic hash is stored in the
email_index
column for searching
- Encrypted before storage in the
-
When a user is retrieved:
- The email is automatically decrypted
- Authentication systems use the
email_index
column for lookups
-
The authentication provider is modified to:
- Look up users by the hashed email index
- Enable all standard Laravel authentication features (login, registration, password reset, etc.)
Manual Encryption/Decryption
You can also use the EncryptionService
directly for custom encryption needs:
use Paperscissorsandglue\EncryptionAtRest\EncryptionService; public function __construct(EncryptionService $encryptionService) { $this->encryptionService = $encryptionService; } public function storeData($data) { $encryptedData = $this->encryptionService->encrypt($data); // Store $encryptedData... } public function retrieveData($encryptedData) { $decryptedData = $this->encryptionService->decrypt($encryptedData); // Use $decryptedData... }
Using the Facade
You can use the provided facade for quick access to encryption functionality:
use Paperscissorsandglue\EncryptionAtRest\Facades\EncryptionAtRest; $encrypted = EncryptionAtRest::encrypt('sensitive data'); $decrypted = EncryptionAtRest::decrypt($encrypted);
CLI Tools
This package includes several command-line tools to help you manage encrypted data.
Encrypting Existing Data
To encrypt data in an existing database table for a model that uses our traits:
php artisan encryption:encrypt-model "App\Models\User"
Options:
--chunk=100
- Process records in chunks (default: 100)--dry-run
- Test the process without making changes--backup=true
- Create a database backup before processing (default: true)--filter="id > 1000"
- Only process records matching SQL where clause
Encrypting Emails Only
For models using the HasEncryptedEmail
trait, you can use a dedicated command to process emails:
php artisan encryption:encrypt-emails "App\Models\User"
Options:
--chunk=100
- Process records in chunks--dry-run
- Test the process without making changes
Decrypting Data
If you need to decrypt data (for example, when migrating away from encryption):
php artisan encryption:decrypt-model "App\Models\User"
Options:
--chunk=100
- Process records in chunks (default: 100)--dry-run
- Test the process without making changes--backup=true
- Create a database backup before processing (default: true)--filter="id > 1000"
- Only process records matching SQL where clause
⚠️ Warning: Decryption permanently removes the encryption protection from your data. Only use this command when absolutely necessary and after creating a backup.
Security Considerations
- All encrypted data is stored using Laravel's built-in encryption features
- The encrypted data cannot be searched or indexed efficiently except for email (which uses a hash-based index)
- Consider using database indexes only on non-encrypted fields
- The email hash is not a security risk as it's a one-way hash, but it does allow deterministic lookup
- Users with the same email will have the same email_index hash, making the email effectively unique in the system
- Always create a database backup before running encryption/decryption commands on production data
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security Vulnerabilities
Please review our security policy on how to report security vulnerabilities.
Credits
License
The MIT License (MIT). Please see License File for more information.