MySQL 使用 SSL 連線(附 Docker 例子)
目錄
檢視是否支援 SSL
使用 OpenSSL 建立 SSL 證照和私鑰
建立 CA 私鑰和 CA 證照
建立伺服器端的 RSA 私鑰和數字證照
建立客戶端的 RSA 私鑰和數字證照
使用工具建立證照與私鑰
SSL 配置
伺服器端配置
客戶端配置
在 Docker 中使能 MySQL SSL 連線
檢視是否支援 SSL
首先在 MySQL 上執行如下命令, 查詢是否 MySQL 支援 SSL:
mysql> SHOW VARIABLES LIKE 'have_ssl';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| have_ssl | YES |
+---------------+-------+
1 row in set (0.02 sec)
當 have_ssl
為 YES
時,
表示此時 MySQL 服務已經支援 SSL 了. 如果是 DESABLE
, 則需要在啟動
MySQL 服務時, 使能 SSL 功能.
使用 OpenSSL 建立 SSL 證照和私鑰
首先我們需要使用 openssl 來建立伺服器端的證照和私鑰. 我使用的 openssl 版本為:
>>> /usr/local/Cellar/openssl/1.0.2j/bin/openssl version
OpenSSL 1.0.2j 26 Sep 2016
新建一個 ~/temp/cert 目錄, 用於存放生成的證照和私鑰
mkdir ~/temp/cert
cd ~/temp/cert
建立 CA 私鑰和 CA 證照
然後, 我們先來生成一個 CA 私鑰:
openssl genrsa 2048 > ca-key.pem
當有了一個 CA 私鑰, 我們接下來就可以使用這個私鑰生成一個新的數字證照:
openssl req -sha1 -new -x509 -nodes -days 3650 -key ca-key.pem > ca-cert.pem
執行這個命令時, 會需要填寫一些問題, 隨便填寫就可以了. 例如:
>>> openssl req -sha1 -new -x509 -nodes -days 3650 -key ca-key.pem > ca-cert.pem
You are about to be asked to enter information that will be incorporated
into your certificate request.
What you are about to enter is what is called a Distinguished Name or a DN.
There are quite a few fields but you can leave some blank
For some fields there will be a default value,
If you enter '.', the field will be left blank.
-----
Country Name (2 letter code) [AU]:CN
State or Province Name (full name) [Some-State]:Beijing
Locality Name (eg, city) []:Beijing
Organization Name (eg, company) [Internet Widgits Pty Ltd]:xys
Organizational Unit Name (eg, section) []:xys
Common Name (e.g. server FQDN or YOUR name) []:xys
Email Address []:yongshun1228@gmail.com
執行上述命令後, 我們就有了一個 CA 私鑰和一個 CA 證照.
建立伺服器端的 RSA 私鑰和數字證照
接著, 我們需要建立伺服器端的私鑰和一個證照請求檔案, 命令如下:
openssl req -sha1 -newkey rsa:2048 -days 3650 -nodes -keyout server-key.pem > server-req.pem
上面這個命令會生成一個新的私鑰(server-key.pem), 同時會使用這個新私鑰來生成一個證照請求檔案(server-req.pem).
上面這個命令同樣需要回答幾個問題, 隨便填寫即可. 不過需要注意的是, A challenge
password
這一項需要為空.
即:
>>> openssl req -sha1 -newkey rsa:2048 -days 3650 -nodes -keyout server-key.pem > server-req.pem
Generating a 2048 bit RSA private key
.................+++
..+++
writing new private key to 'server-key.pem'
-----
You are about to be asked to enter information that will be incorporated
into your certificate request.
What you are about to enter is what is called a Distinguished Name or a DN.
There are quite a few fields but you can leave some blank
For some fields there will be a default value,
If you enter '.', the field will be left blank.
-----
Country Name (2 letter code) [AU]:CN
State or Province Name (full name) [Some-State]:Beijing
Locality Name (eg, city) []:Beijing
Organization Name (eg, company) [Internet Widgits Pty Ltd]:xys
Organizational Unit Name (eg, section) []:xys
Common Name (e.g. server FQDN or YOUR name) []:xys
Email Address []:yongshun1228@gmail.com
Please enter the following 'extra' attributes
to be sent with your certificate request
A challenge password []:
An optional company name []:
下一步, 我們需要將生成的私鑰轉換為 RSA 私鑰檔案格式:
openssl rsa -in server-key.pem -out server-key.pem
最後一步, 我們需要使用原先生成的 CA 證照來生成一個伺服器端的數字證照:
openssl x509 -sha1 -req -in server-req.pem -days 3650 -CA ca-cert.pem -CAkey ca-key.pem -set_serial 01 > server-cert.pem
上面的命令會建立以伺服器端的數字證照檔案.
建立客戶端的 RSA 私鑰和數字證照
和伺服器端所執行的命令類似, 我們也需要為客戶端生成一個私鑰和證照請求檔案, 命令如下:
openssl req -sha1 -newkey rsa:2048 -days 3650 -nodes -keyout client-key.pem > client-req.pem
同樣地, 我們需要將生成的私鑰轉換為 RSA 私鑰檔案格式:
openssl rsa -in client-key.pem -out client-key.pem
最後, 我們也需要為客戶端建立一個數字證照:
openssl x509 -sha1 -req -in client-req.pem -days 3650 -CA ca-cert.pem -CAkey ca-key.pem -set_serial 01 > client-cert.pem
使用工具建立證照與私鑰
前面我們介紹瞭如何使用 OpenSSL 來建立 SSL 連線的私鑰和證照檔案, 現在我們來看一個更簡單的方法.
在 MySQL 5.7 中, 提供了一個名為 mysql_ssl_rsa_setup
的工具,
通過它, 我們可以很方便地建立 SSL 連線所需要的各種檔案:
mkdir ~/temp/cert
cd ~/temp/cert
mysql_ssl_rsa_setup --datadir ./
上面的命令中, --datadir
表示生成的檔案的目錄.
當執行了上述命令後, 也會生成八個檔案:
ca-key.pem
ca.pem
client-cert.pem
client-key.pem
private_key.pem
public_key.pem
server-cert.pem
server-key.pem
這些檔案和我們使用 OpenSSL 所建立的那八個檔案的作用是一樣的, 這裡就不再詳述了.
SSL 配置
在前面的步驟中, 我們已經生成了8個檔案, 分別是:
- ca-cert.pem: CA 證照, 用於生成伺服器端/客戶端的數字證照.
- ca-key.pem: CA 私鑰, 用於生成伺服器端/客戶端的數字證照.
- server-key.pem: 伺服器端的 RSA 私鑰
- server-req.pem: 伺服器端的證照請求檔案, 用於生成伺服器端的數字證照.
- server-cert.pem: 伺服器端的數字證照.
- client-key.pem: 客戶端的 RSA 私鑰
- client-req.pem: 客戶端的證照請求檔案, 用於生成客戶端的數字證照.
- client-cert.pem: 客戶端的數字證照.
接下來我們就需要分別配置伺服器端和客戶端.
伺服器端配置
伺服器端需要用到三個檔案, 分別是: CA 證照
, 伺服器端的
RSA 私鑰
, 伺服器端的數字證照
, 我們需要在 [mysqld]
配置域下新增如下內容:
[mysqld]
ssl-ca=/etc/mysql/ca-cert.pem
ssl-cert=/etc/mysql/server-cert.pem
ssl-key=/etc/mysql/server-key.pem
接著我們還可以更改 bind-address
, 使 MySQL 服務可以接收來自所有
ip 地址的客戶端, 即:
bind-address = *
當配置好後, 我們需要重啟 MySQL 服務, 使能配置.
最後一步, 我們新增一個需要使用 SSL 才可以登入的帳號, 來驗證一下我們所配置的 SSL 是否生效:
GRANT ALL PRIVILEGES ON *.* TO 'ssl_test'@'%' IDENTIFIED BY 'ssl_test' REQUIRE SSL;
FLUSH PRIVILEGES;
當配置好後, 使用 root 登入 MySQL, 執行 show variables like
'%ssl%'
語句會有如下輸出:
mysql> show variables like '%ssl%';
+---------------+-----------------+
| Variable_name | Value |
+---------------+-----------------+
| have_openssl | YES |
| have_ssl | YES |
| ssl_ca | ca.pem |
| ssl_capath | |
| ssl_cert | server-cert.pem |
| ssl_cipher | |
| ssl_crl | |
| ssl_crlpath | |
| ssl_key | server-key.pem |
+---------------+-----------------+
9 rows in set (0.01 sec)
客戶端配置
客戶端配置相對簡單一些. 首先我們需要拷貝 ca-cert.pem
, client-cert.pem
和 client-key.pem
這三個檔案到客戶端主機中,
然後我們可以執行如下命令來使用 SSL 連線 MySQL 服務:
mysql --ssl-ca=/path/to/ca-cert.pem --ssl-cert=/path/to/client-cert.pem --ssl-key=/path/to/client-key.pem -h host_name -u ssl_test -p
除了上述的使用命令列方式配置 SSL 外, 我們也可以使用配置檔案的方式. 即在 ~/.my.cnf
檔案中新增如下內容即可:
[client]
ssl-ca=/path/to/ca-cert.pem
ssl-cert=/path/to/client-cert.pem
ssl-key=/path/to/client-key.pem
當連線成功後, 我們執行如下指令
mysql> \s
--------------
mysql Ver 14.14 Distrib 5.7.17, for Linux (x86_64) using EditLine wrapper
Connection id: 14
Current database:
Current user: ssl_test@172.17.0.4
SSL: Cipher in use is DHE-RSA-AES256-SHA
Current pager: stdout
Using outfile: ''
Using delimiter: ;
Server version: 5.7.17 MySQL Community Server (GPL)
Protocol version: 10
Connection: test_db via TCP/IP
Server characterset: latin1
Db characterset: latin1
Client characterset: latin1
Conn. characterset: latin1
TCP port: 3306
Uptime: 1 hour 2 min 9 sec
Threads: 1 Questions: 23 Slow queries: 0 Opens: 126 Flush tables: 3 Open tables: 0 Queries per second avg: 0.006
--------------
如果輸出中有 SSL: Cipher in use is DHE-RSA-AES256-SHA
之類的資訊,
則表示已經使用 SSL 來連線了.
在 Docker 中使能 MySQL SSL 連線
上面我們簡單介紹了一下如果使能 MySQL SSL 連線, 那麼現在我們使用 Docker 來具體的實戰一把吧!
首先拉取最新的 MySQL 映象:
docker pull mysql
然後需要準備一下掛載到 Docker 容器的目錄結構:
>>> cd ~/temp
>>> tree
.
├── cert
│ ├── ca-key.pem
│ ├── ca.pem
│ ├── client-cert.pem
│ ├── client-key.pem
│ ├── private_key.pem
│ ├── public_key.pem
│ ├── server-cert.pem
│ └── server-key.pem
├── config
│ └── my.cnf
└── db
3 directories, 9 files
在 temp 目錄下有三個子目錄:
cert
目錄用於存放我們先前生成的證照和私鑰資訊;config
目錄用於存放 MySQL 服務的配置檔案db
目錄是用於存放 MySQL 的資料.
下一步我們需要使用如下命令啟動 MySQL 容器:
docker run --rm --name test_db -p 10000:3306 -e MYSQL_ROOT_PASSWORD=root -v /Users/xiongyongshun/temp/db:/var/lib/mysql -v /Users/xiongyongshun/temp/config:/etc/mysql/conf.d -v /Users/xiongyongshun/temp/cert:/etc/mysql/cert mysql:latest
我們在上面的命令中, 我們分別掛載了 cert
, config
, db
這三個宿主機上的目錄到
MySQL 容器中.
啟動了 MySQL 服務後, 可以先使用 root 帳號登入 MySQL, 來檢查 MySQL 服務此時是否已經開啟了 SSL 功能:
docker run -it --link test_db:test_db --rm mysql sh -c 'exec mysql -u root -p -h test_db'
登入成功後, 我們在 MySQL 中執行如下指令:
mysql> show variables like '%ssl%';
+---------------+---------------------------------+
| Variable_name | Value |
+---------------+---------------------------------+
| have_openssl | YES |
| have_ssl | YES |
| ssl_ca | /etc/mysql/cert/ca-cert.pem |
| ssl_capath | |
| ssl_cert | /etc/mysql/cert/server-cert.pem |
| ssl_cipher | |
| ssl_crl | |
| ssl_crlpath | |
| ssl_key | /etc/mysql/cert/server-key.pem |
+---------------+---------------------------------+
9 rows in set (0.01 sec)
有上面的輸出後, 表明此時 MySQL 服務已經使用 SSL 功能了.
接著下一步, 我們按照前面所提到的, 建立一個僅僅可以使用 SSL 登入的帳號, 來檢驗我們的配置是否有效:
GRANT ALL PRIVILEGES ON *.* TO 'ssl_test'@'%' IDENTIFIED BY 'ssl_test' REQUIRE SSL;
FLUSH PRIVILEGES;
上面的命令建立了一個帳號名為 ssl_test
, 密碼為 ssl_test
,
並且不限制登入主機 ip 的帳號.
這些都配置成功後, 我們再啟動一個 MySQL 客戶端容器:
docker run -it --link test_db:test_db --rm -v /Users/xiongyongshun/temp/cert:/etc/mysql/cert mysql sh -c 'exec mysql --ssl-ca=/etc/mysql/cert/ca-cert.pem --ssl-cert=/etc/mysql/cert/client-cert.pem --ssl-key=/etc/mysql/cert/client-key.pem -h test_db -u ssl_test -p'
從上面的這個命令中我們可以看到, 啟動 MySQL 客戶端容器時, 我們掛載了宿主機的 cert
目錄到容器內的 /etc/mysql/cert
目錄,
這樣在容器中就可以訪問到 SSL 私鑰和證照檔案了. 接著我們在 MySQL 客戶端命令列中, 使用 --ssl-ca
, --ssl-cert
, --ssl-key
這三個引數來指定
SSL 連線所需要的 CA 證照, RSA 私鑰和客戶端證照.
登入成功後, 我們執行 \s
命令:
mysql> \s
--------------
mysql Ver 14.14 Distrib 5.7.17, for Linux (x86_64) using EditLine wrapper
Connection id: 5
Current database:
Current user: ssl_test@172.17.0.5
SSL: Cipher in use is DHE-RSA-AES256-SHA
Current pager: stdout
Using outfile: ''
Using delimiter: ;
Server version: 5.7.17 MySQL Community Server (GPL)
Protocol version: 10
Connection: test_db via TCP/IP
Server characterset: latin1
Db characterset: latin1
Client characterset: latin1
Conn. characterset: latin1
TCP port: 3306
Uptime: 6 min 8 sec
Threads: 2 Questions: 10 Slow queries: 0 Opens: 113 Flush tables: 1 Open tables: 106 Queries per second avg: 0.027
--------------
輸出中有 SSL: Cipher in use is DHE-RSA-AES256-SHA
資訊則說明我們確實是使用了
SSL 連線的 MySQL 伺服器.
相關文章
- MySQL SSL連線問題MySql
- MySQL 5.7配置SSL連線MySql
- mysql5.7 ssl加密連線MySql加密
- Docker phpmyadmin 連線mysql容器DockerPHPMySql
- 技術分享 | MySQL : SSL 連線淺析MySql
- spring 簡單的使用 Hikari連線池 和 jdbc連線mysql 的一個簡單例子SpringJDBCMySql單例
- 使用pyMySql 連線mysqlMySql
- 使用mysqlclient庫連線mysqlMySqlclient
- docker中搭建canal監聽mysql例子DockerMySql
- docker安裝mysql8.0.20並遠端連線DockerMySql
- 使用 SSL 加密的 JDBC 連線 SAP HANA 資料庫加密JDBC資料庫
- 使用Docker Context連線遠端節點DockerContext
- Docker的容器使用與連線-WindowDocker
- centos無法建立ssl連線CentOS
- 使用cmd連線mysql資料庫MySql資料庫
- 使用PETAPOCO連線MYSQL資料庫MySql資料庫
- 如何使用Docker生成SSL證書Docker
- 使用 Apache 反向代理來配置 Laravel-Websocket SSL(WSS 連線)ApacheLaravelWeb
- Ubuntu18.04下安裝Docker並配置SSL證書加密遠端連線UbuntuDocker加密
- SSL連線,搭建網路安全道路
- Netty 實現SSL安全連線(wss://)Netty
- 【Docker】使用docker部署 mysqlDockerMySql
- Django使用pymysql連線MySQL資料庫DjangoMySql資料庫
- docker學習5:Docker 容器連線Docker
- 使用 Docker CertBot 獲取 SSL 證書Docker
- C連線MySQLMySql
- IDEA連線MySQLIdeaMySql
- GO 連線 MySQLGoMySql
- JDBC連線mysqlJDBCMySql
- Python 連線 MySQLPythonMySql
- python連線MySQLPythonMySql
- MYSQL語法:左連線、右連線、內連線、全外連線MySql
- MySQL筆記3——內連線/外連線、多表連線MySql筆記
- 檢視使用 MySQL Shell 的連線狀態MySql
- 使用RMySQL包來連線MySQL資料庫MySql資料庫
- mysql使用命令列連線伺服器MySql命令列伺服器
- 允許mysql遠端使用者連線。MySql
- navicat連線遠端伺服器docker的mysql容器時連不上報錯伺服器DockerMySql
- linux下mysql安裝、授權、建立使用者、連線navicat、連線entityLinuxMySql