给Python中的MySQLdb模块添加超时功能的教程


Posted in Python onMay 05, 2015

使用Python操作MySQL数据库的时候常使用MySQLdb这个模块。

今天在开发的过程发现MySQLdb.connect有些参数没法设置。通过这个页面我们可以看到在connect的时候,可以设置的option和client_flags和MySQL c api相比差不少。

一个很重要的参数 MYSQL_OPT_READ_TIMEOUT没法设置,这个参数如果不设置,极致状况MySQL处于hang住,自动切换IP漂移,客户端无法重连到新MySQL。

给MySQLdb加Option很简单,只要修改_mysql.c这个把Python对象映射到MySQL操作的文件,添加参数,再加一段mysql_option即可。

下面是修改后的git diff 文件

?View Code BASH

diff --git a/_mysql.c b/_mysql.c
index d42cc54..61a9b34 100644
--- a/_mysql.c
+++ b/_mysql.c
@@ -489,9 +489,10 @@ _mysql_ConnectionObject_Initialize(
"named_pipe", "init_command",
"read_default_file", "read_default_group",
"client_flag", "ssl",
-                 "local_infile",
+                 "local_infile", "read_timeout",
NULL } ;
int connect_timeout = 0;
+    int read_timeout = 0;
int compress = -1, named_pipe = -1, local_infile = -1;
char *init_command=NULL,
*read_default_file=NULL,
@@ -500,7 +501,7 @@ _mysql_ConnectionObject_Initialize(
self->converter = NULL;
self->open = 0;
check_server_init(-1);
-    if (!PyArg_ParseTupleAndKeywords(args, kwargs, "|ssssisOiiisssiOi:connect",
+    if (!PyArg_ParseTupleAndKeywords(args, kwargs, "|ssssisOiiisssiOii:connect",
kwlist,
&host, &user, &passwd, &db,
&port, &unix_socket, &conv,
@@ -509,7 +510,8 @@ _mysql_ConnectionObject_Initialize(
&init_command, &read_default_file,
&read_default_group,
&client_flag, &ssl,
-                    &local_infile /* DO NOT PATCH FOR RECONNECT, IDIOTS
+                    &local_infile, &read_timeout
+                    /* DO NOT PATCH FOR RECONNECT, IDIOTS
IF YOU DO THIS, I WILL NOT SUPPORT YOUR PACKAGES. */
))
return -1;
@@ -540,6 +542,12 @@ _mysql_ConnectionObject_Initialize(
mysql_options(&(self->connection), MYSQL_OPT_CONNECT_TIMEOUT,
(char *)&timeout);
}
+
+    if (read_timeout) {
+        unsigned int timeout = read_timeout;
+        mysql_options(&(self->connection), MYSQL_OPT_READ_TIMEOUT, (char *)&timeout);
+    }
+
if (compress != -1) {
mysql_options(&(self->connection), MYSQL_OPT_COMPRESS, 0);
client_flag |= CLIENT_COMPRESS;

代码修改完毕,python setup.py install 即可,如果出现mysql_config找不到的问题。你还要修改setup_posix.py文件。

hoterran@hoterran-laptop:~/Projects/MySQL-python-1.2.3$ git diff setup_posix.py
diff --git a/setup_posix.py b/setup_posix.py
index 86432f5..f4f08f1 100644
--- a/setup_posix.py
+++ b/setup_posix.py
@@ -23,7 +23,7 @@ def mysql_config(what):
     if ret/256 > 1:
       raise EnvironmentError("%s not found" % (mysql_config.path,))
   return data
-mysql_config.path = "mysql_config"
+mysql_config.path = "/usr/local/mysql/bin/mysql_config"

 def get_config():
   import os, sys

编译通过,我们来试试添加的read_timeout这个参数。

conn = MySQLdb.connect(host = DB_SERVER,user = DB_USERNAME,passwd = DB_PASSWORD,db = DB_NAME, port=int(DB_PORT), client_flag = 2, read_timeout = 10)

然后执行语句前,你试着把mysql用gdb hang住10s后,python就会异常抛错

OperationalError: (2013, 'Lost connection to MySQL server during query')
 >/home/hoterran/Projects/dbaas/trunk/dbtest.py(18)()
 >mydb.execute_sql(conn, sql)
(Pdb)
--Return--
> /home/hoterran/Projects/dbaas/trunk/dbtest.py(18)()->None
> mydb.execute_sql(conn, sql)
(Pdb)
OperationalError: (2013, 'Lost connection to MySQL server during query')
> <string>(1)<module>()->None

Python 相关文章推荐
Python实现的生成自我描述脚本分享(很有意思的程序)
Jul 18 Python
Python选择排序、冒泡排序、合并排序代码实例
Apr 10 Python
Python文件的读写和异常代码示例
Oct 31 Python
Python金融数据可视化汇总
Nov 17 Python
Python+tkinter使用40行代码实现计算器功能
Jan 30 Python
python如何创建TCP服务端和客户端
Aug 26 Python
django基于cors解决跨域请求问题详解
Aug 06 Python
PyTorch的Optimizer训练工具的实现
Aug 18 Python
jenkins配置python脚本定时任务过程图解
Oct 29 Python
python内置模块collections知识点总结
Dec 19 Python
pytorch:torch.mm()和torch.matmul()的使用
Dec 27 Python
如何解决pycharm调试报错的问题
Aug 06 Python
用Python实现一个简单的多线程TCP服务器的教程
May 05 #Python
简单介绍Python中的try和finally和with方法
May 05 #Python
python中的闭包用法实例详解
May 05 #Python
Python闭包实现计数器的方法
May 05 #Python
深入探究Python中变量的拷贝和作用域问题
May 05 #Python
Python使用metaclass实现Singleton模式的方法
May 05 #Python
python中查看变量内存地址的方法
May 05 #Python
You might like
新的一年,新的期待:DC在2020年的四部动画电影
2020/01/01 欧美动漫
php使用Header函数,PHP_AUTH_PW和PHP_AUTH_USER做用户验证
2016/05/04 PHP
Yii2.0预定义的别名功能小结
2016/07/04 PHP
JQuery+JS实现仿百度搜索结果中关键字变色效果
2011/08/02 Javascript
JS链式调用的实现方法
2013/03/07 Javascript
Jquery实现的tab效果可以指定默认显示第几页
2013/10/16 Javascript
JavaScript 动态加载脚本和样式的方法
2015/04/13 Javascript
页面内容排序插件jSort使用方法
2015/10/10 Javascript
深入理解javascript作用域第二篇之词法作用域和动态作用域
2016/07/24 Javascript
Zepto实现密码的隐藏/显示
2017/04/07 Javascript
JS实现线性表的顺序表示方法示例【经典数据结构】
2017/04/11 Javascript
vue修改vue项目运行端口号的方法
2017/08/04 Javascript
node.js调用C++函数的方法示例
2018/09/21 Javascript
node.js的Express服务器基本使用教程
2019/01/09 Javascript
js获取form表单中name属性的值
2019/02/27 Javascript
JavaScript展开操作符(Spread operator)详解
2019/07/20 Javascript
浅谈Vue为什么不能检测数组变动
2019/10/14 Javascript
vue实现路由不变的情况下,刷新页面操作示例
2020/02/02 Javascript
Jquery高级应用Deferred对象原理及使用实例
2020/05/28 jQuery
[00:57]深扒TI7聊天轮盘语音出处5
2017/05/11 DOTA
[56:57]LGD vs VP 2019DOTA2国际邀请赛淘汰赛 胜者组赛BO3 第一场 8.20.mp4
2019/08/22 DOTA
Python中函数的用法实例教程
2014/09/08 Python
Python中static相关知识小结
2018/01/02 Python
Php多进程实现代码
2018/05/07 Python
HTML5中使用json对象的实例代码
2018/09/10 HTML / CSS
韩国三大免税店之一:THE GRAND 中文免税店
2016/07/21 全球购物
希尔顿酒店中国网站:Hilton中国
2017/03/11 全球购物
英国医生在线预约:Top Doctors
2019/10/30 全球购物
亚洲颇具影响力的男性在线购物零售商:His
2019/11/24 全球购物
美国优质宠物用品购买网站:Muttropolis
2020/02/17 全球购物
公司节能减排倡议书
2014/05/14 职场文书
导盲犬小Q观后感
2015/06/11 职场文书
反邪教学习心得体会
2016/01/15 职场文书
2019秋季运动会口号
2019/06/25 职场文书
2019年聘任书的写作格式及范文!
2019/07/03 职场文书
Python中requests做接口测试的方法
2021/05/30 Python