## 概述
Sodium crypto library是一個現代化的,易于使用的軟件庫,用于加密,解密,簽名,密碼散列等。
Sodium 出現的目的也是為了代替 Mcrypt 這個原來的加密擴展。在 PHP7.2 之后,Mcrypt 已經被移除,在 PHP7.1 時就已經被標記為過時。不過,Sodium 擴展的應用也并不是很多,大部分情況下我們都會使用 OpenSSL 來進行加密操作,同時,Sodium 擴展提供的函數也非常多
Sodium 擴展在 PHP7.2 后是跟隨 PHP 源碼一起發布的,只需要在編譯的時候加上`--with-sodium`即可安裝成功。如果是 PHP7.2 之前的版本,需要單獨安裝這個擴展。同時,操作系統中也需要安裝 `libsodium-devel` 庫
## 安裝
```
sudo pecl install -f libsodium
```
以下表示安裝成功
```
Build process completed successfully
Installing '/usr/local/php-7.4/lib/php/extensions/no-debug-non-zts-20190902/sodium.so'
install ok: channel://pecl.php.net/libsodium-2.0.23
configuration option "php_ini" is not set to php.ini location
You should add "extension=sodium.so" to php.ini
```
安裝錯誤
```
checking for libsodium... configure: error: Please install libsodium - See https://github.com/jedisct1/libsodium
ERROR: `/tmp/pear/temp/libsodium/configure --with-php-config=/usr/local/php-7.4/bin/php-config' failed
```
請安裝擴展系統擴展庫`libsodium-dev`
```
sudo apt install libsodium-dev
Reading package lists... Done
Building dependency tree
Reading state information... Done
The following NEW packages will be installed:
libsodium-dev
0 upgraded, 1 newly installed, 0 to remove and 254 not upgraded.
Need to get 160 kB of archives.
After this operation, 760 kB of additional disk space will be used.
Get:1 http://mirrors.cloud.aliyuncs.com/ubuntu bionic/main amd64 libsodium-dev amd64 1.0.16-2 [160 kB]
Fetched 160 kB in 0s (1,514 kB/s)
Selecting previously unselected package libsodium-dev:amd64.
(Reading database ... 161808 files and directories currently installed.)
Preparing to unpack .../libsodium-dev_1.0.16-2_amd64.deb ...
Unpacking libsodium-dev:amd64 (1.0.16-2) ...
Setting up libsodium-dev:amd64 (1.0.16-2) ...
```
修改`php.ini`
```
extension=sodium.so
```
查看模塊
```
php -m |grep sodium
sodium
```
## 簡單使用
**加密**
```
<?php
/**
* @desc libsodium
* @author Tinywan(ShaoBo Wan)
* @date 2024/8/14 22:45
*/
require '../vendor/autoload.php';
$secretKey = sodium_crypto_secretbox_keygen();
$message = 'Sensitive 開源技術小棧';
$nonce = random_bytes(SODIUM_CRYPTO_SECRETBOX_NONCEBYTES);
$encryptedMessage = sodium_crypto_secretbox($message, $nonce, $secretKey);
var_dump($encryptedMessage);
```
輸出
```
string(44) "??|}3?¢?[zm?? n?\?V?ˉC"
```
**解密**
```
$decryptedMessage = sodium_crypto_secretbox_open($encryptedMessage, $nonce, $secretKey);
var_dump($decryptedMessage);
```
輸出
```
string(28) "Sensitive 開源技術小棧"
```
## 隨機數據
在使用密碼學時,經常需要隨機字節或整數用于各種目的(加密密鑰,隨機數等)。具體來說,您需要使用密碼學安全隨機數生成器(CSPRNG).一個通用的隨機數生成器是不夠的。
### 隨機字節
如果你需要一個由隨機字節組成的字符串,你可以使用`\Sodium\randombytes_buf()`。
```
$string = \Sodium\randombytes_buf($num_bytes);
```
如果將`$num_bytes`設置為32,則`$string`將是一個32字節的字符串,每個字節將是0到255之間的隨機值的字符表示。
### 隨機整數
如果你需要一個在0和一個特定上限之間的均勻分布的隨機整數,你可以使用`\Sodium\randombytes_uniform()`。
例如,如果您需要一個介于1和100之間的數字:
```
$int = \Sodium\randombytes_uniform(100) + 1;
```
請注意,在上面的例子中,`$int`的可能值范圍從1到100,因為`\Sodium\randombytes_uniform`將返回0到99之間的隨機整數。100**不**包括在`\Sodium\randombytes_uniform(100)`的可能輸出值中。
### 隨機16位整數
返回一個介于0和65535(含)之間的整數,服從均勻分布。
```
$tcp_port = \Sodium\randombytes_random16();
```
## 對稱加密算法 AEAD_AES_256_GCM
對數據進行 AES-256-GCM 加密和解密。可以使用 `openssl` 和 `sodium` 擴展來實現加密,它們都支持 `AES-256-GCM` 算法,下面將給出兩種擴展的代碼示例。
微信支付最新的 V3 版本接口,微信返回的報文中,如果涉及敏感信息,是需要基于 AEAD_AES_256_GCM 進行解密的。在官方文檔中,也提供了 PHP 對應的解密方式,其中使用的就是 Sodium 擴展庫中的函數。
如何解密證書和回調報文:https://pay.weixin.qq.com/docs/merchant/development/interface-rules/certificate-callback-decryption.html#_2-%E8%A7%A3%E5%AF%86
### Sodium 擴展實現
```php
<?php
/**
* @desc 對稱加密算法 AES-256-GCM 代碼示例
* @author Tinywan(ShaoBo Wan)
* @date 2024/8/14 9:15
*/
declare(strict_types=1);
/**
* @desc 加密
* @param string $data
* @param string $keygen
* @param string $aad
* @return array
* @throws SodiumException
* @throws \Random\RandomException
* @author Tinywan(ShaoBo Wan)
*/
function aes256gcm_encrypt(string $data, string $keygen, string $aad = ''): array
{
$iv = random_bytes(SODIUM_CRYPTO_AEAD_AES256GCM_NPUBBYTES);
$encrypt = sodium_crypto_aead_aes256gcm_encrypt($data, $aad, $iv, $keygen);
return [
'iv' => sodium_bin2base64($iv, SODIUM_BASE64_VARIANT_ORIGINAL),
'aad' => sodium_bin2base64($aad, SODIUM_BASE64_VARIANT_ORIGINAL),
'cipher_text' => sodium_bin2base64($encrypt, SODIUM_BASE64_VARIANT_ORIGINAL),
];
}
/**
* @desc 解密
* @param array $secretData
* @param string $keygen
* @return false|string
* @throws SodiumException
* @author Tinywan(ShaoBo Wan)
*/
function aes256gcm_decrypt(array $secretData, string $keygen)
{
$iv = sodium_base642bin($secretData['iv'], SODIUM_BASE64_VARIANT_ORIGINAL);
$aad = sodium_base642bin($secretData['aad'], SODIUM_BASE64_VARIANT_ORIGINAL);
$cipherText = sodium_base642bin($secretData['cipher_text'], SODIUM_BASE64_VARIANT_ORIGINAL);
return sodium_crypto_aead_aes256gcm_decrypt($cipherText, $aad, $iv, $keygen);
}
/**
* @desc 主函數
* @return void
* @throws SodiumException
* @throws \Random\RandomException
* @author Tinywan(ShaoBo Wan)
*/
function main()
{
if (!sodium_crypto_aead_aes256gcm_is_available()) {
exit('Not support AES-256-GCM');
}
// 生成AES-256-GCM的密鑰
$keygen = sodium_crypto_aead_aes256gcm_keygen();
// 加密原文內容
$data = '開源技術小棧';
$secretData = aes256gcm_encrypt($data, $keygen, 'tinywan');
// 解密
$plainText = aes256gcm_decrypt($secretData, $keygen);
echo '[x] 加密原文:' .$data.PHP_EOL;
echo '[x] 密文數據:' . json_encode($secretData) .PHP_EOL;
echo '[x] 解密原文:' .$plainText.PHP_EOL;
}
main();
```
執行結果
```ts
[x] 加密原文:開源技術小棧
[x] 密文數據:{"iv":"KGMwg3PKlihp46qo","aad":"dGlueXdhbg==","cipher_text":"6oRb78FP33byDBhlBc0g6e7DYH0k3Oiethup+4YeWPN4Xw=="}
[x] 解密原文:開源技術小棧
```
### Openssl擴展實現
```php
<?php
/**
* @desc openssl.php
* @author Tinywan(ShaoBo Wan)
* @date 2024/8/14 9:22
*/
declare(strict_types=1);
/**
* @desc 加密
* @param string $data
* @param string $keygen
* @param string $aad
* @return array
* @throws \Random\RandomException
* @author Tinywan(ShaoBo Wan)
*/
function aes256gcm_encrypt(string $data, string $keygen, string $aad = ''): array
{
$cipher = 'aes-256-gcm';
$ivLen = openssl_cipher_iv_length($cipher);
$iv = random_bytes($ivLen);
$encrypt = openssl_encrypt($data, $cipher, $keygen, OPENSSL_RAW_DATA, $iv, $tag, $aad);
return [
'iv' => base64_encode($iv),
'tag' => base64_encode($tag),
'aad' => base64_encode($aad),
'cipher_text' => base64_encode($encrypt),
];
}
/**
* @desc 解密
* @param array $secretData
* @param string $keygen
* @return false|string
* @author Tinywan(ShaoBo Wan)
*/
function aes256gcm_decrypt(array $secretData, string $keygen)
{
$iv = base64_decode($secretData['iv']);
$tag = base64_decode($secretData['tag']);
$aad = base64_decode($secretData['aad']);
$cipherText = base64_decode($secretData['cipher_text']);
$cipher = 'aes-256-gcm';
return openssl_decrypt($cipherText, $cipher, $keygen, OPENSSL_RAW_DATA, $iv, $tag, $aad);
}
/**
* @desc 主函數
* @return void
* @throws SodiumException
* @throws \Random\RandomException
* @author Tinywan(ShaoBo Wan)
*/
function main()
{
// AES-256-GCM需要32字節的密鑰
$keygen = bin2hex(random_bytes(16));
// 加密
$data = '開源技術小棧';
$secretData = aes256gcm_encrypt($data, $keygen, 'tinywan');
// 解密
$plainText = aes256gcm_decrypt($secretData, $keygen);
echo '[x] 加密原文:' . $data . PHP_EOL;
echo '[x] 密文數據:' . json_encode($secretData) . PHP_EOL;
echo '[x] 解密原文:' . $plainText . PHP_EOL;
}
main();
```
執行結果
```ts
[x] 加密原文:開源技術小棧
[x] 密文數據:{"iv":"CUit1bBOIOOM4Oaa","tag":"hlyP0MpykCNekBd1Kkobdw==","aad":"dGlueXdhbg==","cipher_text":"ToKc2+rVh8udQa\/bUtKodoC+"}
[x] 解密原文:開源技術小棧
```
### AAD參數(Additional Authenticated Data)
在上面的代碼示例中,可以看到在加密的時候有一個`$aad`參數,如果在加密的時候使用了這個參數,那么在解密時也需要使用同樣的AAD值才能成功解密。
> 什么時候會需要用到AAD呢?
用戶`Tinywan`在微信公眾號上寫了一篇私密文章,微信公眾號使用 `AES-256-GCM` 加密了這篇文章,然后存儲到了數據庫里,`AAD`的取值是`Tinywan`的用戶`ID`。
后續如果`Tinywan`去查看這篇文章,微信公眾號會使用`Tinywan`的用戶`ID(AAD值)`去解密,因為解密的`AAD`值與加密的`AAD`值相同,所以可以成功解密。
然后有一個黑客`ShaoBoWan`,他也向微信公眾號發出解密這篇文章的請求,微信公眾號就會使用`ShaoBoWan`的用戶`ID`作為`AAD`去解密這篇文章,但因為`AAD`值錯誤,肯定是解密不了的,所以這時候`AAD`的作用就體現了出來。
## 其他
https://paragonie.com/book/pecl-libsodium
https://cloud.tencent.com/developer/information/%E5%9C%A8PHP%E5%87%BD%E6%95%B0%E4%B8%AD%E4%BD%BF%E7%94%A8Sodium%E5%BA%93-article
https://paragonie.com/book/pecl-libsodium/read/00-intro.md#what-is-libsodium
- 設計模式系列
- 工廠方法模式
- 序言
- Windows程序注冊為服務的工具WinSW
- 基礎
- 安裝
- 開發規范
- 目錄結構
- 配置
- 快速入門
- 架構
- 請求流程
- 架構總覽
- URL訪問
- 容器和依賴注入
- 中間件
- 事件
- 代碼層結構
- 四個層次
- 路由
- 控制器
- 請求
- 響應
- 數據庫
- MySQL實時同步數據到ES解決方案
- 阿里云DTS數據MySQL同步至Elasticsearch實戰
- PHP中的MySQL連接池
- PHP異步非阻塞MySQL客戶端連接池
- 模型
- 視圖
- 注解
- @SpringBootApplication(exclude={DataSourceAutoConfiguration.calss})
- @EnableFeignClients(basePackages = "com.wotu.feign")
- @EnableAspectJAutoProxy
- @EnableDiscoveryClient
- 錯誤和日志
- 異常處理
- 日志處理
- 調試
- 驗證
- 驗證器
- 驗證規則
- 擴展庫
- 附錄
- Spring框架知識體系詳解
- Maven
- Maven和Composer
- 構建Maven項目
- 實操課程
- 01.初識SpringBoot
- 第1章 Java Web發展史與學習Java的方法
- 第2章 環境與常見問題踩坑
- 第3章 springboot的路由與控制器
- 02.Java編程思想深度理論知識
- 第1章 Java編程思想總體
- 第2章 英雄聯盟的小案例理解Java中最為抽象的概念
- 第3章 徹底理解IOC、DI與DIP
- 03.Spring與SpringBoot理論篇
- 第1章 Spring與SpringBoot導學
- 第2章 Spring IOC的核心機制:實例化與注入
- 第3章 SpringBoot基本配置原理
- 04.SprinBoot的條件注解與配置
- 第1章 conditonal 條件注解
- 第2章 SpringBoot自動裝配解析
- 05.Java異常深度剖析
- 第1章 Java異常分類剖析與自定義異常
- 第2章 自動配置Url前綴
- 06.參數校驗機制與LomBok工具集的使用
- 第1章 LomBok工具集的使用
- 第2章 參數校驗機制以及自定義校驗
- 07.項目分層設計與JPA技術
- 第1章 項目分層原則與層與層的松耦合原則
- 第2章 數據庫設計、實體關系與查詢方案探討
- 第3章 JPA的關聯關系與規則查詢
- 08.ORM的概念與思維
- 第1章 ORM的概念與思維
- 第2章 Banner等相關業務
- 第3章 再談數據庫設計技巧與VO層對象的技巧
- 09.JPA的多種查詢規則
- 第1章 DozerBeanMapper的使用
- 第2章 詳解SKU的規格設計
- 第3章 通用泛型Converter
- 10.令牌與權限
- 第1章 通用泛型類與java泛型的思考
- 常見問題
- 微服務
- demo
- PHP中Self、Static和parent的區別
- Swoole-Cli
- 為什么要使用現代化PHP框架?
- 公眾號
- 一鍵部署微信公眾號Markdown編輯器(支持適配和主題設計)
- Autodesigner 2.0發布
- Luya 一個現代化PHP開發框架
- PHPZip - 創建、讀取和管理 ZIP 文件的簡單庫
- 吊打Golang的PHP界天花板webman壓測對比
- 簡潔而強大的 YAML 解析庫
- 推薦一個革命性的PHP測試框架:Kahlan
- ServBay下一代Web開發環境
- 基于Websocket和Canvas實現多人協作實時共享白板
- Apipost預執行腳本如何調用外部PHP語言
- 認證和授權的安全令牌 Bearer Token
- Laradock PHP 的 Docker 完整本地開發環境
- 高效接口防抖策略,確保數據安全,避免重復提交的終極解決方案!
- TIOBE 6月榜單:PHP穩步前行,編程語言生態的微妙變化
- Aho-Corasick字符串匹配算法的實現
- Redis鍵空間通知 Keyspace Notification 事件訂閱
- ServBay如何啟用并運行Webman項目
- 使用mpdf實現導出pdf文件功能
- Medoo 輕量級PHP數據庫框架
- 在PHP中編寫和運行單元測試
- 9 PHP運行時基準性能測試
- QR碼生成器在PHP中的源代碼
- 使用Gogs極易搭建的自助Git服務
- Gitea
- webman如何記錄SQL到日志?
- Sentry PHP: 實時監測并處理PHP應用程序中的錯誤
- Swoole v6 Alpha 版本已發布
- Proxypin
- Rust實現的Redis內存數據庫發布
- PHP 8.4.0 Alpha 1 測試版本發布
- 121
- Golang + Vue 開發的開源輕量 Linux 服務器運維管理面板
- 內網穿透 FRP VS Tailscale
- 新一代開源代碼托管平臺Gitea
- 微服務系列
- Nacos云原生配置中心介紹與使用
- 輕量級的開源高性能事件庫libevent
- 國密算法
- 國密算法(商用密碼)
- GmSSL 支持國密SM2/SM3/SM4/SM9/SSL 密碼工具箱
- GmSSL PHP 使用
- 數據庫
- SQLite數據庫的Web管理工具
- 阿里巴巴MySQL數據庫強制規范
- PHP
- PHP安全測試秘密武器 PHPGGC
- 使用declare(strict_types=1)來獲得更健壯的PHP代碼
- PHP中的魔術常量
- OSS 直傳阿里騰訊示例
- PHP源碼編譯安裝APCu擴展實現數據緩存
- BI性能DuckDB數據管理系統
- 為什么別人可以是架構師!而我卻不是?
- 密碼還在用 MD5 加鹽?不如試試 password_hash
- Elasticsearch 在電商領域的應用與實踐
- Cron 定時任務入門
- 如何動態設置定時任務!而不是寫死在Linux Crontab
- Elasticsearch的四種查詢方式,你知道多少?
- Meilisearch vs Elasticsearch
- OpenSearch vs Elasticsearch
- Emlog 輕量級開源博客及建站系統
- 現代化PHP原生協程引擎 PRipple
- 使用Zephir編寫C擴展將PHP源代碼編譯加密
- 如何將PHP源代碼編譯加密,同時保證代碼能正常的運行
- 為什么選擇Zephir給PHP編寫動態擴展庫?
- 使用 PHP + XlsWriter實現百萬級數據導入導出
- Rust編寫PHP擴展
- 阿里云盤開放平臺對接進行文件同步
- 如何構建自己的PHP靜態可執行文件
- IM后端架構
- RESTful設計方法和規范
- PHP編譯器BPC 7.3 發布,成功編譯ThinkPHP8
- 高性能的配置管理擴展 Yaconf
- PHP實現雪花算法庫 Snowflake
- PHP官方現代化核心加密庫Sodium
- pie
- 現代化、精簡、非阻塞PHP標準庫PSL
- PHP泛型和集合
- 手把手教你正確使用 Composer包管理
- JWT雙令牌認證實現無感Token自動續期
- 最先進PHP大模型深度學習庫TransformersPHP
- PHP如何啟用 FFI 擴展
- PHP超集語言PXP
- 低延遲雙向實時事件通信 Socket.IO
- PHP OOP中的繼承和多態
- 強大的現代PHP高級調試工具Kint
- PHP基金會
- 基于webman+vue3高質量中后臺框架SaiAdmin
- 開源免費的定時任務管理系統:Gocron
- 簡單強大OCR工具EasyOCR在PHP中使用
- PHP代碼抽象語法樹工具PHP AST Viewer
- MySQL數據庫管理工具PHPMyAdmin
- Rust編寫的一款高性能多人代碼編輯器Zed
- 超高性能PHP框架Workerman v5.0.0-beta.8 發布
- 高并發系列
- 入門介紹及安裝
- Lua腳本開發 Hello World
- 執行流程與階段詳解
- Nginx Lua API 接口開發
- Lua模塊開發
- OpenResty 高性能的正式原因
- 記一次查找 lua-resty-mysql 庫 insert_id 的 bug
- 包管理工具OPM和LuaRocks使用
- 異步非阻塞HTTP客戶端庫 lua-resty-http
- Nginx 內置綁定變量
- Redis協程網絡庫 lua-resty-redis
- 動態HTML渲染庫 lua-testy-template
- 單獨的
- StackBlitz在線開發環境
- AI
- 基礎概念
- 12312
- 基礎鏡像的坑
- 利用phpy實現 PHP 編寫 Vision Transformer (ViT) 模型
- 語義化版本 2.0.0