数据库的Connection-Control插件

    作者:潇湘隐者更新于: 2021-06-29 12:44:14

    MySQL的Connection-Control插件。数据库中的数据是为众多用户所 共享其信息而建立的,已经摆脱了具体 程序的限制和制约。不同的用户可以按各自的用法使用数据库中的数据;多个用户可以同时共享数据库中的数据资源,即不同的用户可以同时存取数据库中的同一个数据。数据共享性不仅满足了各用户对信息内容的要求,同时也满足了各用户之间信息通信的要求。

    OracleSQL Server数据库中,可以设置一些复杂的账号密码策略,例如在失败登录超过N次即可锁定账号,那么在MySQL中能否也有这样的功能呢?

    数据库的Connection-Control插件_数据库管理系统_数据分析_MySQL数据库_课课家

    本文转载自微信公众号「DBA闲思杂想录」,作者潇湘隐者 。转载本文请联系DBA闲思杂想录公众号。 

    在Oracle和SQL Server数据库中,可以设置一些复杂的账号密码策略,例如在失败登录超过N次即可锁定账号,那么在MySQL中能否也有这样的功能呢?答案是MySQL也有类似这样的功能,只不过在MySQL中是在其登录失败超过阈值后,它是延迟响应时间,而不是锁定账号,在MySQL 5.6.35以后提供了Connection-Control插件用来控制客户端在登录操作连续失败一定次数后的响应延迟。该插件可有效的防止客户端暴力登录的风险(攻击)。该插件包含以下2个组件

    • CONNECTION_CONTROL:用来控制登录失败的次数及延迟响应时间
    • CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS:该表将登录失败的操作记录至INFORMATION_SCHEMA库中

    下面我们介绍一下Connection-Control Plugins,下面实验的版本为MySQL 8.0.18。

    首先查看系统变量plugin_dir,找到插件(Plugins)所在的路径,如下所示

    1. mysql> select version() from dual; 
    2. +-----------+ 
    3. | version() | 
    4. +-----------+ 
    5. | 8.0.18    | 
    6. +-----------+ 
    7. 1 row in set (0.00 sec) 
    8.   
    9.   
    10. mysql> show variables like 'plugin_dir'
    11. +---------------+--------------------------+ 
    12. | Variable_name | Value                    | 
    13. +---------------+--------------------------+ 
    14. | plugin_dir    | /usr/lib64/mysql/plugin/ | 
    15. +---------------+--------------------------+ 
    16. 1 row in set (0.01 sec) 
    17.   
    18.   
    19.   
    20. [root@KerryDB ~]# cd /usr/lib64/mysql/plugin/ 
    21. [root@KerryDB plugin]# ls -lrt 
    22. total 76388 
    23. -rwxr-xr-x. 1 root root   106696 Sep 20  2019 rewrite_example.so 
    24. -rwxr-xr-x. 1 root root   104432 Sep 20  2019 mysql_no_login.so 
    25. -rwxr-xr-x. 1 root root   111464 Sep 20  2019 mypluglib.so 
    26. -rwxr-xr-x. 1 root root   106648 Sep 20  2019 auth_socket.so 
    27. -rwxr-xr-x. 1 root root   163560 Sep 20  2019 adt_null.so 
    28. -rwxr-xr-x. 1 root root   409032 Sep 20  2019 validate_password.so 
    29. -rwxr-xr-x. 1 root root  9054776 Sep 20  2019 libpluginmecab.so 
    30. -rwxr-xr-x. 1 root root   344696 Sep 20  2019 authentication_ldap_sasl_client.so 
    31. -rwxr-xr-x. 1 root root  1145832 Sep 20  2019 rewriter.so 
    32. -rwxr-xr-x. 1 root root   625944 Sep 20  2019 ha_example.so 
    33. -rwxr-xr-x. 1 root root   388992 Sep 20  2019 semisync_slave.so 
    34. -rwxr-xr-x. 1 root root    32368 Sep 20  2019 component_log_sink_json.so 
    35. -rwxr-xr-x. 1 root root   235200 Sep 20  2019 component_audit_api_message_emit.so 
    36. -rwxr-xr-x. 1 root root   494720 Sep 20  2019 keyring_udf.so 
    37. -rwxr-xr-x. 1 root root   149280 Sep 20  2019 component_log_sink_syseventlog.so 
    38. -rwxr-xr-x. 1 root root  1446024 Sep 20  2019 semisync_master.so 
    39. -rwxr-xr-x. 1 root root  2277480 Sep 20  2019 mysql_clone.so 
    40. -rwxr-xr-x. 1 root root  1231376 Sep 20  2019 libmemcached.so 
    41. -rwxr-xr-x. 1 root root   454096 Sep 20  2019 component_mysqlbackup.so 
    42. -rwxr-xr-x. 1 root root   193336 Sep 20  2019 component_log_filter_dragnet.so 
    43. -rwxr-xr-x. 1 root root  1177352 Sep 20  2019 ha_mock.so 
    44. -rwxr-xr-x. 1 root root    83936 Sep 20  2019 locking_service.so 
    45. -rwxr-xr-x. 1 root root  1269784 Sep 20  2019 connection_control.so 
    46. -rwxr-xr-x. 1 root root  1280936 Sep 20  2019 innodb_engine.so 
    47. -rwxr-xr-x. 1 root root   442304 Sep 20  2019 component_validate_password.so 
    48. -rwxr-xr-x. 1 root root  1206024 Sep 20  2019 version_token.so 
    49. -rwxr-xr-x. 1 root root  2338880 Sep 20  2019 keyring_file.so 
    50. -rwxr-xr-x. 1 root root  2031912 Sep 20  2019 ddl_rewriter.so 
    51. -rwxr-xr-x. 1 root root 49246400 Sep 20  2019 group_replication.so 
    52. drwxr-xr-x. 2 root root     4096 Nov  6  2019 debug 

    安装插件

    1. mysql> INSTALL PLUGIN CONNECTION_CONTROL SONAME 'connection_control.so'
    2. Query OK, 0 rows affected (0.02 sec) 
    3. mysql> INSTALL PLUGIN CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS SONAME 'connection_control.so'
    4. Query OK, 0 rows affected (0.00 sec) 

    检查是否安装成功

    1. mysql> show plugins; 
    2.   
    3. mysql> SELECT PLUGIN_NAME, PLUGIN_LIBRARY, PLUGIN_STATUS, LOAD_OPTION  
    4.     -> FROM INFORMATION_SCHEMA.PLUGINS  
    5.     -> WHERE PLUGIN_LIBRARY = 'CONNECTION_CONTROL.SO'
    6. +------------------------------------------+-----------------------+---------------+-------------+ 
    7. | PLUGIN_NAME                              | PLUGIN_LIBRARY        | PLUGIN_STATUS | LOAD_OPTION | 
    8. +------------------------------------------+-----------------------+---------------+-------------+ 
    9. | CONNECTION_CONTROL                       | connection_control.so | ACTIVE        | FORCE       | 
    10. | CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS | connection_control.so | ACTIVE        | FORCE       | 
    11. +------------------------------------------+-----------------------+---------------+-------------+ 
    12. rows in set (0.00 sec) 
    13.   
    14. mysql>  

    设置系统变量

    1. mysql> show variables like 'connection_control%'
    2. +-------------------------------------------------+------------+ 
    3. | Variable_name                                   | Value      | 
    4. +-------------------------------------------------+------------+ 
    5. | connection_control_failed_connections_threshold | 3          | 
    6. | connection_control_max_connection_delay         | 2147483647 | 
    7. | connection_control_min_connection_delay         | 1000       | 
    8. +-------------------------------------------------+------------+ 
    • connection_control_failed_connections_threshold #登陆失败次数限制,默认值为3
    • connection_control_max_connection_delay #限制重试时间最大值,单位为毫秒( milliseconds),默认值2147483647
    • connection_control_min_connection_delay #限制重试时间最小值,单位为毫秒( milliseconds),默认值为1000毫秒,也就是1秒

    注意事项:

    1:connection_control_min_connection_delay的值必须小于connection_control_max_connection_delay,connection_control_max_connection_delay不能小于connection_control_min_connection_delay的值。

    1. mysql> set global connection_control_min_connection_delay=60000; 
    2.  
    3. Query OK, 0 rows affected (0.00 sec) 

    注意,命令方式设置全局系统变量在服务器重启后丢失,所以最好的方式在参数文件my.cnf设置全局系统变量

    -- 配置文件增加以下配置

    1. [mysqld]  
    2.   
    3. plugin-load-add = connection_control.so          #不是必须 
    4.   
    5. connection-control = FORCE                       #不是必须 
    6.   
    7. connection-control-failed-login-attempts = FORCE #不是必须 
    8.   
    9. connection_control_min_connection_delay = 60000  
    10.   
    11. connection_control_max_connection_delay = 1800000  
    12.   
    13. connection_control_failed_connections_threshold = 3 

    三次连续输错密码后,就会在第四次输入密码后挂起

    1. [root@lnx02 ~]# mysql -h 10.20.57.24 -u test -p 
    2. Enter password:  
    3. ERROR 1045 (28000): Access denied for user 'test'@'192.168.27.180' (using password: YES) 
    4. [root@lnx02 ~]# mysql -h 10.20.57.24 -u test -p 
    5. Enter password:  
    6. ERROR 1045 (28000): Access denied for user 'test'@'192.168.27.180' (using password: YES) 
    7. [root@lnx02 ~]# mysql -h 10.20.57.24 -u test -p 
    8. Enter password:  
    9. ERROR 1045 (28000): Access denied for user 'test'@'192.168.27.180' (using password: YES) 
    10. [root@lnx02 ~]# mysql -h 10.20.57.24 -u test -p 
    11. Enter password:  

    注意,MySQL服务重启过后,INFORMATION_SCHEMA.CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS中的数据全部前空。

    • The CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS plugin must be activated for this table to be available, and the CONNECTION_CONTROL plugin must be activated or the table contents will always be empty. See Section 6.4.2, “The Connection-Control Plugins”.

    必须激活CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS插件才能使用该表CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS,并且要么激活CONNECTION_CONTROL插件,要么该表的内容始终为空。请参见第6.4.2节“连接控制插件”。

    • The table contains rows only for clients that have had one or more consecutive failed connection attempts without a subsequent successful attempt. When a client connects successfully, its failed-connection count is reset to zero and the server removes any row corresponding to the client.

    该表仅包含已进行一次或多次连续失败连接尝试而没有随后成功尝试的客户端的行。当客户端成功连接时,其失败连接计数将重置为零,并且服务器将删除与该客户端对应的任何行。

    • Assigning a value to the connection_control_failed_connections_threshold system variable at runtime resets all accumulated failed-connection counters to zero, which causes the table to become empty.

    在运行时为connection_control_failed_connections_threshold系统变量分配一个值会将所有累积的失败连接计数器重置为零,这将导致表变空。

    解除账号延迟响应限制

    方法1:重启MySQL实例

    方法2:调整系统变量connection_control_failed_connections_threshold的值。

    1. mysql> SELECT * FROM  
    2.     -> INFORMATION_SCHEMA.CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS; 
    3. +-------------------+-----------------+ 
    4. | USERHOST          | FAILED_ATTEMPTS | 
    5. +-------------------+-----------------+ 
    6. 'test'@'192.168%' |               5 | 
    7. +-------------------+-----------------+ 
    8. 1 row in set (0.00 sec) 
    9.   
    10.   
    11. mysql>  
    12. mysql> set global connection_control_failed_connections_threshold=2; 
    13. Query OK, 0 rows affected (0.00 sec) 
    14.   
    15. mysql> SELECT * FROM  
    16.     -> INFORMATION_SCHEMA.CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS; 
    17. Empty set (0.00 sec) 

    3:卸载插件plugin

    1. mysql> UNINSTALL PLUGIN CONNECTION_CONTROL_FAILED_LOGIN_ATTEMPTS; 
    2.  
    3. mysql> UNINSTALL PLUGIN CONNECTION_CONTROL; 

    注意事项:

    关于Connection-Control的实现原理,淘宝数据库内核月报有篇文章分析过代码,介绍过实现原理,Connection-Control插件虽然可以防止恶意暴力破解MySQL账户,但是可能会浪费MySQL的资源;

    • 比如如果短时间内有大量的恶意攻击,该插件虽然可以防止破解MySQL账户,但是会消耗主机资源(每一个连接创建一个线程);
    • 如果这里使用了线程池,虽然可以避免消耗主机资源,但是等线程池中的线程被消耗光,再有新连接来就会拒绝服务。
      数据库是一个 单位或是一个应用领域的通用数据处理系统,它存储的是属于企业和事业部门、 团体和个人的有关数据的 集合。数据库中的数据是从全局观点出发建立的,按一定的 数据模型进行组织、描述和存储。其结构基于数据间的自然联系,从而可提供一切必要的存取 路径,且数据不再针对某一应用,而是面向全组织,具有整体的结构化特征。

课课家教育

未登录