添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
相关文章推荐
小眼睛的手套  ·  android ...·  5 月前    · 

SSL 加密概述

SSL(Secure Sockets Layer)认证是客户端到云数据库服务器端的认证,对用户和服务器进行认证。开通 SSL 加密,可获取 CA 证书,将 CA 证书上传在服务端。在客户端访问数据库时,将激活 SSL 协议,在客户端和数据库服务端之间建立一条 SSL 安全通道,实现数据信息加密传输,防止数据在传输过程中被截取、篡改、窃听,保证双方传递信息的安全性。

SSL 协议要求建立在可靠的传输层协议(TCP)之上,其优势在于它是与应用层协议独立无关的,高层的应用层协议(例如:HTTP、FTP、TELNET 等)能透明地建立于 SSL 协议之上。SSL 协议在应用层协议通信之前就已经完成加密算法、通信密钥的协商及服务器认证工作,在此之后应用层协议所传送的数据都会被加密,从而保证通信的私密性。

背景

使用非加密方式连接数据库时,在网络中传输的所有信息都是明文,因此存在被非法用户窃听、篡改、冒充的三大风险;而 SSL 协议是为解决这三大风险而设计的,理论上可达到:

  • 信息是加密传播,第三方无法窃听。
  • 具有校验机制,一旦被篡改,通信双方会立刻发现。
  • 配备身份证书,防止身份被冒充。
  • 云数据库 MySQL 支持通过开启 SSL 加密来增强链路安全性,并支持下载和安装 SSL CA 证书到需要的应用服务。

    SSL 加密不保护数据本身,是确保来往于数据库和服务器之间的流量安全,在传输层对网络连接进行加密,能够提升通信数据的安全性和完整性,但会同时增加网络连接响应时间。

  • 实例版本为 MySQL 5.6/5.7/8.0。
  • 实例架构为双节点/三节点。
  • 实例引擎为 InooDB/RocksDB。
  • 开启 SSL 加密

    1. 登录 MySQL 控制台 ,在实例列表,单击实例 ID 或操作列的 管理 ,进入实例管理页面。
    2. 在实例管理页面的 数据安全 页下,选择 SSL 页。
    3. 此功能状态默认为未打开,将开关调为开启,然后单击 确定 ,开启 SSL 加密。
      • 主实例开启 SSL 窗口如下:

        开启 SSL 过程中,会重启您的数据库实例以加载 SSL 证书,请确保业务具备重连机制。

      • RO 实例开启 SSL 界面如下:

        配置 RO 实例 SSL 功能会同步配置所属 RO 组中其他 RO 实例。

      • 单击 下载 ,下载 SSL CA 证书。
        下载的文件为压缩包(TencentDB-CA-Chain.zip),包含如下三个文件:
        • p7b 文件:用于 Windows 系统中导入 CA 证书。
        • jks 文件:Java 中的 truststore 证书存储文件,密码统一为 tencentdb,用于 Java 程序中导入 CA 证书链。
        • pem 文件:用于其他系统或应用中导入 CA 证书。
        • 配置 SSL CA 证书

          开启 SSL 加密后,使用客户端连接云数据库时需要配置 SSL CA 证书。以下以 Navicat 为例,为您介绍 SSL CA 证书安装方法。其它应用或者客户端请参见对应产品的使用说明。

          云数据库 MySQL 每开启或关闭一次 SSL 加密,其证书就会新生成。

        • 打开 Navicat。
        • 在对应数据库上单击鼠标右键,选择 编辑连接
        • 选择 SSL 页签,选择.pem 格式 CA 证书的路径。完成下图设置后单击 确定

          如果出现 connection is being used 报错,可能由于之前的会话未断开,请关闭 Navicat 后重试。

        • 双击对应数据库,测试能否正常连接。
        • 关闭 SSL 加密

          1. 登录 MySQL 控制台 ,在实例列表,单击实例 ID 或操作列的 管理 ,进入实例管理页面。
          2. 在实例管理页面的 数据安全 页下,选择 SSL 页。
          3. 单击已开通前面的开关按钮,在弹出的提示框中单击 确定

            关闭 SSL 过程中,会重启您的数据库实例以卸载 SSL 证书,请确保业务具备重连机制。

            连接开启 SSL 连接加密的实例

            使用 SSL 连接加密的方式连接数据库 SQL 如下:

            mysql -P <端口号> -h <IP 地址>  -u <用户名> -p<密码> --ssl-ca<ca证书>

            常用程序连接开启 SSL 的实例的代码示例

            • PHP

              $conn = mysqli_init();
              mysqli_ssl_set($conn,NULL,NULL, "<下载的证书路径>", NULL, NULL);
              mysqli_real_connect($conn, '<数据库访问地址>', '<数据库访问用户名>', '<数据库访问密码>', '<指定访问数据库>', <访问端口>, MYSQLI_CLIENT_SSL);
              if (mysqli_connect_errno($conn)) {
              die('Failed to connect to MySQL: '.mysqli_connect_error());
              
            • PHP (Using PDO)

              $options = array(
                  PDO::MYSQL_ATTR_SSL_CA => '<下载的证书路径>'
              $db = new PDO('mysql:host=<数据库访问地址>;port=<访问端口>;dbname=<指定访问数据库>', '<数据库访问用户名>', '<数据库访问密码>', $options);
            • Java (MySQL Connector for Java)

              # generate truststore and keystore in code
              String importCert = " -import "+
                  " -alias mysqlServerCACert "+
                  " -file " + ssl_ca +
                  " -keystore truststore "+
                  " -trustcacerts " +
                  " -storepass password -noprompt ";
              String genKey = " -genkey -keyalg rsa " +
                  " -alias mysqlClientCertificate -keystore keystore " +
                  " -storepass password123 -keypass password " +
                  " -dname CN=MS ";
              sun.security.tools.keytool.Main.main(importCert.trim().split("\\s+"));
              sun.security.tools.keytool.Main.main(genKey.trim().split("\\s+"));
              # use the generated keystore and truststore
              System.setProperty("javax.net.ssl.keyStore","<下载的证书路径>");
              System.setProperty("javax.net.ssl.keyStorePassword","tencentdb");
              System.setProperty("javax.net.ssl.trustStore","<下载的证书路径>");
              System.setProperty("javax.net.ssl.trustStorePassword","tencentdb");
              url = String.format("jdbc:mysql://%s/%s?serverTimezone=UTC&useSSL=true", '<数据库访问地址>', '<指定访问数据库>');
              properties.setProperty("user", '<数据库访问用户名>');
              properties.setProperty("password", '<数据库访问密码>');
              conn = DriverManager.getConnection(url, properties);
            • .NET (MySqlConnector)

              var builder = new MySqlConnectionStringBuilder
                  Server = "<数据库访问地址>",
                  UserID = "<数据库访问用户名>",
                  Password = "<数据库访问密码>",
                  Database = "<指定访问数据库>",
                  SslMode = MySqlSslMode.VerifyCA,
                  SslCa = "<下载的证书>",
              using (var connection = new MySqlConnection(builder.ConnectionString))
                  connection.Open();
              
            • Python (MySQLConnector Python)

              try:
                  conn = mysql.connector.connect(user='<数据库访问用户名>',
                                                 password='<数据库访问密码>',
                                                 database='<指定访问数据库>',
                                                 host='<数据库访问地址>',
                                                 ssl_ca='<下载的证书路径>')
              except mysql.connector.Error as err:
                  print(err)
            • Python (PyMySQL)

              conn = pymysql.connect(user='<数据库访问用户名>',
                                     password='<数据库访问密码>',
                                     database='<指定访问数据库>',
                                     host='<数据库访问地址>',
                                     ssl={'ca': '<下载的证书路径>'})
            • Django (PyMySQL)

              DATABASES = {
                  'default': {
                      'ENGINE': 'django.db.backends.mysql',
                      'NAME': '<指定访问数据库>',
                      'USER': '<数据库访问用户名>',
                      'PASSWORD': '<数据库访问密码>',
                      'HOST': '<数据库访问地址>',
                      'PORT': '<访问端口>',
                      'OPTIONS': {
                          'ssl': {'ca': '<下载的证书路径>'}
              
            • Node.js

              var fs = require('fs');
              var mysql = require('mysql');
              const serverCa = [fs.readFileSync("<下载的证书路径>", "utf8")];
              var conn=mysql.createConnection({
                  host:"<数据库访问地址>",
                  user:"<数据库访问用户名>",
                  password:"<数据库访问密码>",
                  database:"<指定访问数据库>",
                  port:<访问端口>,
                  ssl: {
                      rejectUnauthorized: true,
                      ca: serverCa
              conn.connect(function(err) {
                if (err) throw err;
              
            • Golang

              rootCertPool := x509.NewCertPool()
              pem, _ := ioutil.ReadFile("<下载的证书路径>")
              if ok := rootCertPool.AppendCertsFromPEM(pem); !ok {
                  log.Fatal("Failed to append PEM.")
              mysql.RegisterTLSConfig("custom", &tls.Config{RootCAs: rootCertPool})
              var connectionString string
              connectionString = fmt.Sprintf("%s:%s@tcp(%s:<访问端口>)/%s?allowNativePasswords=true&tls=custom","<数据库访问用户名>" , "<数据库访问密码>", "<数据库访问地址>", '<指定访问数据库>')
              db, _ := sql.Open("mysql", connectionString)
            • Ruby

              client = Mysql2::Client.new(
                      :host     => '<数据库访问地址>',
                      :username => '<数据库访问用户名>',
                      :password => '<数据库访问密码>',
                      :database => '<指定访问数据库>',
                      :sslca => '<下载的证书路径>'