1、前言
mysql 是完全网络化的跨平台关系型数据库系统,同时是具有客户机/服务器体系结构的分布式数据库管理系统。它具有功能强、使用简便、管理方便、运行速度快、安全可靠性强等优点,用户可利用许多语言编写访问mysql 数据库的程序,特别是与php更是黄金组合,运用十分广泛。
由于mysql是多平台的数据库,它的默认配置要考虑各种情况下都能适用,所以在我们自己的使用环境下应该进行进一步的安全加固。作为一个mysql的系统管理员,我们有责任维护mysql数据库系统的数据安全性和完整性。
mysql数据库的安全配置必须从两个方面入手,系统内部安全和外部网络安全,另外我们还将简单介绍编程时要注意的一些问题以及一些小窍门。
2、系统内部安全
首先简单介绍一下mysql数据库目录结构。mysql安装好,运行了mysql_db_install脚本以后就会建立数据目录和初始化数据库。如果我们用mysql源码包安装,而且安装目录是/usr/local/mysql,那么数据目录一般会是/usr/local/mysql/var。数据库系统由一系列数据库组成,每个数据库包含一系列数据库表。mysql是用数据库名在数据目录建立建立一个数据库目录,各数据库表分别以数据库表名作为文件名,扩展名分别为myd、myi、frm的三个文件放到数据库目录中。
mysql的授权表给数据库的访问提供了灵活的权限控制,但是如果本地用户拥有对库文件的读权限的话,攻击者只需把数据库目录打包拷走,然后拷到自己本机的数据目录下就能访问窃取的数据库。所以mysql所在的主机的安全性是最首要的问题,如果主机不安全,被攻击者控制,那么mysql的安全性也无从谈起。其次就是数据目录和数据文件的安全性,也就是权限设置问题。
从mysql主站一些老的binary发行版来看,3.21.xx版本中数据目录的属性是775,这样非常危险,任何本地用户都可以读数据目录,所以数据库文件很不安全。3.22.xx版本中数据目录的属性是770,这种属性也有些危险,本地的同组用户既能读也能写,所以数据文件也不安全。3.23.xx版本数据目录的属性是700,这样就比较好,只有启动数据库的用户可以读写数据库文件,保证了本地数据文件的安全。
如果启动mysql数据库的用户是mysql,那么象如下的目录和文件的是安全的,请注意数据目录及下面的属性:
shell>ls -l /usr/local/mysql
total 40
drwxrwxr-x 2 root root 4096 feb 27 20:07 bin
drwxrwxr-x 3 root root 4096 feb 27 20:07 include
drwxrwxr-x 2 root root 4096 feb 27 20:07 info
drwxrwxr-x 3 root root 4096 feb 27 20:07 lib
drwxrwxr-x 2 root root 4096 feb 27 20:07 libexec
drwxrwxr-x 3 root root 4096 feb 27 20:07 man
drwxrwxr-x 6 root root 4096 feb 27 20:07 mysql-test
drwxrwxr-x 3 root root 4096 feb 27 20:07 share
drwxrwxr-x 7 root root 4096 feb 27 20:07 sql-bench
drwx—— 4 mysql mysql 4096 feb 27 20:07 var
shell>ls -l /usr/local/mysql/var
total 8
drwx—— 2 mysql mysql 4096 feb 27 20:08 mysql
drwx—— 2 mysql mysql 4096 feb 27 20:08 test
shell>ls -l /usr/local/mysql/var/mysql
total 104
-rw——- 1 mysql mysql 0 feb 27 20:08 columns_priv.myd
-rw——- 1 mysql mysql 1024 feb 27 20:08 columns_priv.myi
-rw——- 1 mysql mysql 8778 feb 27 20:08 columns_priv.frm
-rw——- 1 mysql mysql 302 feb 27 20:08 db.myd
-rw——- 1 mysql mysql 3072 feb 27 20:08 db.myi
-rw——- 1 mysql mysql 8982 feb 27 20:08 db.frm
-rw——- 1 mysql mysql 0 feb 27 20:08 func.myd
-rw——- 1 mysql mysql 1024 feb 27 20:08 func.myi
-rw——- 1 mysql mysql 8641 feb 27 20:08 func.frm
-rw——- 1 mysql mysql 0 feb 27 20:08 host.myd
-rw——- 1 mysql mysql 1024 feb 27 20:08 host.myi
-rw——- 1 mysql mysql 8958 feb 27 20:08 host.frm
-rw——- 1 mysql mysql 0 feb 27 20:08 tables_priv.myd
-rw——- 1 mysql mysql 1024 feb 27 20:08 tables_priv.myi
-rw——- 1 mysql mysql 8877 feb 27 20:08 tables_priv.frm
-rw——- 1 mysql mysql 428 feb 27 20:08 user.myd
-rw——- 1 mysql mysql 2048 feb 27 20:08 user.myi
-rw——- 1 mysql mysql 9148 feb 27 20:08 user.frm
如果这些文件的属主及属性不是这样,请用以下两个命令修正之:
shell>chown -r mysql.mysql /usr/local/mysql/var
shell>chmod -r go-rwx /usr/local/mysql/var
用root用户启动远程服务一直是安全大忌,因为如果服务程序出现问题,远程攻击者极有可能获得主机的完全控制权。mysql从3.23.15版本开始时作了小小的改动,默认安装后服务要用mysql用户来启动,不允许root用户启动。如果非要用root用户来启动,必须加上–user=root的参数(./safe_mysqld –user=root &)。因为mysql中有load data infile和select … into outfile的sql语句,如果是root用户启动了mysql服务器,那么,数据库用户就拥有了root用户的写权限。不过mysql还是做了一些限制的,比如load data infile只能读全局可读的文件,select … into outfile不能覆盖已经存在的文件。
本地的日志文件也不能忽视,包括shell的日志和mysql自己的日志。有些用户在本地登陆或备份数据库的时候为了图方便,有时会在命令行参数里直接带了数据库的密码,如:
shell>/usr/local/mysql/bin/mysqldump -uroot -ptest test>test.sql
shell>/usr/local/mysql/bin/mysql -uroot -ptest
这些命令会被shell记录在历史文件里,比如bash会写入用户目录的.bash_history文件,如果这些文件不慎被读,那么数据库的密码就会泄漏。用户登陆数据库后执行的sql命令也会被mysql记录在用户目录的.mysql_history文件里。如果数据库用户用sql语句修改了数据库密码,也会因.mysql_history文件而泄漏。所以我们在shell登陆及备份的时候不要在-p后直接加密码,而是在提示后再输入数据库密码。
另外这两个文件我们也应该不让它记录我们的操作,以防万一。
shell>rm .bash_history .mysql_history
shell>ln -s /dev/null .bash_history
shell>ln -s /dev/null .mysql_history
上门这两条命令把这两个文件链接到/dev/null,那么我们的操作就不会被记录到这两个文件里了。
3、外部网络安全
mysql数据库安装好以后,unix平台的user表是这样的:
mysql> use mysql;
database changed
mysql> select host,user,password,select_priv,grant_priv from user;
+———–+——+———-+————-+————+
| host | user | password | select_priv | grant_priv |
+———–+——+———-+————-+————+
| localhost | root | | y | y |
| redhat | root | | y | y |
| localhost | | | n | n |
| redhat | | | n | n |
+———–+——+———-+————-+————+
4 rows in set (0.00 sec)
windows平台的user表是这样的:
mysql> use mysql;
database changed
mysql> select host,user,password,select_priv,grant_priv from user;
+———–+——+———-+————-+————+
| host | user | password | select_priv | grant_priv |
+———–+——+———-+————-+————+
| localhost | root | | y | y |
| % | root | | y | y |
| localhost | | | y | y |
| % | | | n | n |
+———–+——+———-+————-+————+
4 rows in set (0.00 sec)
我们先来看unix平台的user表。其中redhat只是我试验机的机器名,所以实际上unix平台的mysql默认只允许本机才能连接数据库。但是缺省root用户口令是空,所以当务之急是给root用户加上口令。给数据库用户加口令有三种方法:
1)在shell提示符下用mysqladmin命令来改root用户口令:
shell>mysqladmin -uroot password test
这样,mysql数据库root用户的口令就被改成test了。(test只是举例,我们实际使用的口令一定不能使用这种易猜的弱口令)
2)用set password修改口令:
mysql> set password for root@localhost=password(test);
这时root用户的口令就被改成test了。
3)直接修改user表的root用户口令:
mysql> use mysql;
mysql> update user set password=password(test) where user=root;
mysql> flush privileges;
这样,mysql数据库root用户的口令也被改成test了。其中最后一句命令flush privileges的意思是强制刷新内存授权表,否则用的还是缓冲中的口令,这时非法用户还可以用root用户及空口令登陆,直到重启mysql服务器。
我们还看到user为空的匿名用户,虽然它在unix平台下没什么权限,但为了安全起见我们应该删除它:
mysql> delete from user where user=;
windows版本mysql的user表有很大不同,我们看到host字段除了localhost还有是%。这里%的意思是允许任意的主机连接mysql服务器,这是非常不安全的,给攻击者造成可乘之机,我们必须删除host字段为%的记录:
mysql>delete from user where host=%;
默认root用户的空密码也是必须修改,三种修改方法和unix平台一样。
我们注意到host字段为localhost的匿名用户拥有所有的权限!就是说本地用户用空的用户名和空的口令登陆mysql数据库服务器可以得到最高的权限!所以匿名用户必须删除!
mysql> delete from user where user=;
对user表操作以后不要忘了用flush privileges来强制刷新内存授权表,这样才能生效。
默认安装的windows版mysql存在的不安全因素太多,我们在安装后一定要进一步配置!
mysql的5个授权表:user, db, host, tables_priv和columns_priv提供非常灵活的安全机制,从mysql 3.22.11开始引入了两条语句grant和revoke来创建和删除用户权限,可以方便的限制哪个用户可以连接服务器,从哪里连接以及连接后可以做什么操作。作为mysql管理员,我们必须了解授权表的意义以及如何用grant和revoke来创建用户、授权和撤权、删除用户。
在3.22.11版本以前的mysql授权机制不完善,和新版本也有较大的不同,建议升级到最新版本的mysql。(本书的操作例子是以mysql 3.23.49
为样本)我们先来了解授权表的结构。
1)mysql授权表的结构与内容:
mysql> desc user;
+—————–+—————–+——+—–+———+——-+
| field | type | null | key | default | extra |
+—————–+—————–+——+—–+———+——-+
| host | char(60) binary | | pri | | |
| user | char(16) binary | | pri | | |
| password | char(16) binary | | | | |
| select_priv | enum(n,y) | | | n | |
| insert_priv | enum(n,y) | | | n | |
| update_priv | enum(n,y) | | | n | |
| delete_priv | enum(n,y) | | | n | |
| create_priv | enum(n,y) | | | n | |
| drop_priv | enum(n,y) | | | n | |
| reload_priv | enum(n,y) | | | n | |
| shutdown_priv | enum(n,y) | | | n | |
| process_priv | enum(n,y) | | | n | |
| file_priv | enum(n,y) | | | n | |
| grant_priv | enum(n,y) | | | n | |
| references_priv | enum(n,y) | | | n | |
| index_priv | enum(n,y) | | | n | |
| alter_priv | enum(n,y) | | | n | |
+—————–+—————–+——+—–+———+——-+
17 rows in set (0.01 sec)
user表是5个授权表中最重要的一个,列出可以连接服务器的用户及其加密口令,并且它指定他们有哪种全局(超级用户)权限。在user表启用的任何权限均是全局权限,并适用于所有数据库。所以我们不能给任何用户访问mysql.user表的权限!
权限说明:
+———–+————-+———————————————————————–+
| 权限指定符| 列名 |权限操作 |
+———–+————-+———————————————————————–+
| select | select_priv | 允许对表的访问,不对数据表进行访问的select语句不受影响,比如select 1+1|
+———–+————-+———————————————————————–+
| insert | insert_priv | 允许对表用insert语句进行写入操作。 |
+———–+————-+———————————————————————–+
| update | update_priv | 允许用update语句修改表中现有记录。 |
+———–+————-+———————————————————————–+
| delete | delete_priv | 允许用delete语句删除表中现有记录。 |
+———–+————-+———————————————————————–+
| create | create_priv | 允许建立新的数据库和表。 |
+———–+————-+———————————————————————–+
| drop | drop_priv | 允许删除现有的数据库和表。 |
+———–+————-+———————————————————————–+
| index | index_priv | 允许创建、修改或删除索引。 |
+———–+————-+———————————————————————–+
| alter | alter_priv | 允许用alter语句修改表结构。 |
+———–+————-+———————————————————————–+
| grant | grant_priv | 允许将自己拥有的权限授予其它用户,包括grant。 |
+———–+————-+———————————————————————–+
| reload | reload | 允许重载授权表,刷新服务器等命令。 |
+———–+————-+———————————————————————–+
| shutdown | shudown_priv| 允许用mysqladmin shutdown命令关闭mysql服务器。该权限比较危险, |
| | | 不应该随便授予。 |
+———–+————-+———————————————————————–+
| process | process_priv| 允许查看和终止mysql服务器正在运行的线程(进程)以及正在执行的查询语句 |
| | | ,包括执行修改密码的查询语句。该权限比较危险,不应该随便授予。 |
+———–+————-+———————————————————————–+
| file | file_priv | 允许从服务器上读全局可读文件和写文件。该权限比较危险,不应该随便授予。|
+———–+————-+———————————————————————–+
mysql> desc db;
+—————–+—————–+——+—–+———+——-+
| field | type | null | key | default | extra |
+—————–+—————–+——+—–+———+——-+
| host | char(60) binary | | pri | | |
| db | char(64) binary | | pri | | |
| user | char(16) binary | | pri | | |
| select_priv | enum(n,y) | | | n | |
| insert_priv | enum(n,y) | | | n | |
| update_priv | enum(n,y) | | | n | |
| delete_priv | enum(n,y) | | | n | |
| create_priv | enum(n,y) | | | n | |
| drop_priv | enum(n,y) | | | n | |
| grant_priv | enum(n,y) | | | n | |
| references_priv | enum(n,y) | | | n | |
| index_priv | enum(n,y) | | | n | |
| alter_priv | enum(n,y) | | | n | |
+—————–+—————–+——+—–+———+——-+
13 rows in set (0.01 sec)
db表列出数据库,而用户有权限访问它们。在这里指定的权限适用于一个数据库中的所有表。
mysql> desc host;
+—————–+—————–+——+—–+———+——-+
| field | type | null | key | default | extra |
+—————–+—————–+——+—–+———+——-+
| host | char(60) binary | | pri | | |
| db | char(64) binary | | pri | | |
| select_priv | enum(n,y) | | | n | |
| insert_priv | enum(n,y) | | | n | |
| update_priv | enum(n,y) | | | n | |
| delete_priv | enum(n,y) | | | n | |
| create_priv | enum(n,y) | | | n | |
| drop_priv | enum(n,y) | | | n | |
| grant_priv | enum(n,y) | | | n | |
| references_priv | enum(n,y) | | | n | |
| index_priv | enum(n,y) | | | n | |
| alter_priv | enum(n,y) | | | n | |
+—————–+—————–+——+—–+———+——-+
12 rows in set (0.01 sec)
host表与db表结合使用在一个较好层次上控制特定主机对数据库的访问权限,这可能比单独使用db好些。这个表不受grant和revoke语句的影响
,所以,你可能发觉你根本不是用它。
mysql> desc tables_priv;
+————-+—————————–+——+—–+———+——-+
| field | type | null | key | default | extra |
+————-+—————————–+——+—–+———+——-+
| host | char(60) binary | | pri | | |
| db | char(64) binary | | pri | | |
| user | char(16) binary | | pri | | |
| table_name | char(60) binary | | pri | | |
| grantor | char(77) | | mul | | |
| timestamp | timestamp(14) | yes | | null | |
| table_priv | set(select,insert, | | | | |
| | update,delete,create, | | | | |
| | drop,grant,references,| | | | |
| | index,alter) | | | | |
| column_priv | set(select,insert, | | | | |
| | update,references) | | | | |
+————-+—————————–+——+—–+———+——-+
8 rows in set (0.01 sec)
tables_priv表指定表级权限。在这里指定的一个权限适用于一个表的所有列。
mysql> desc columns_priv;
+————-+————————+——+—–+———+—-+
| field | type | null | key | default | extra |
+————-+————————+——+—–+———+—-+
| host | char(60) binary | | pri | | |
| db | char(64) binary | | pri | | |
| user | char(16) binary | | pri | | |
| table_name | char(64) binary | | pri | | |
| column_name | char(64) binary | | pri | | |
| timestamp | timestamp(14) | yes | | null | |
| column_priv | set(select,insert, | | | | |
| | update,references) | | | | |
+————-+————————+——+—–+———+——-+
7 rows in set (0.00 sec)
columns_priv表指定列级权限。在这里指定的权限适用于一个表的特定列。
2)mysql授权表运行机制
mysql的访问控制分两个步骤:
a)服务器检查是否允许该用户连接。
b)如果该用户有权连接,那么服务器还会检查它的每一个请求是否有足够的权限。比如:用户检索数据库中的一个表需要有这个数据库的select权限,用户删除数据库中的一个表需要有这个数据库的drop权限。
授权表的user, db, host表使用这两个步骤,tables_priv和columns_priv表只使用第二步(检查请求)。每个授权表包含决定一个权限何时运用的范围列和决定授予哪种权限的权限列。
范围列指定表中的权限何时运用。每个授权表条目包含user和host列来指定权限何时运用于一个给定用户从给定主机的连接。其他表包含附加的范围列,如db表包含一个db列指出权限运用于哪个数据库。类似地,tables_priv和columns_priv表包含范围字段,缩小范围到一个数据库中的特定表或一个表的特定列。
下面是user表的host字段和user字段组合的一些例子:
+—————————–+——–+————————————————+
| host值 | user值 | 匹配的连接 |
+—————————–+——–+———————————————–+
| x.y.z | test | test用户只能从x.y.z连接数据库 |
+—————————–+——–+———————————————–+
| x.y.z | | 任何用户可以从x.y.z连接数据库 |
+—————————–+——–+———————————————–+
| % | test | test用户可以从任意主机连接数据库 |
+—————————–+——–+———————————————-+
| | | 任何用户可以从任意主机连接数据库 |
+—————————–+——–+———————————————–+
| %.y.z | test | test用户可以从y.z域的任意主机连接数据库 |
+—————————–+——–+———————————————-+
| x.y.% | test | test用户可以从x.y.net, x.y.com, x.y.edu等主机连接数据库|
+—————————–+——–+———————————————-+
| 192.168.1.1 | test | test用户可以从ip地址为192.168.1.1的主机连接数据库 |
+—————————–+——–+———————————————+
| 192.168.1.% | test | test用户可以从c类子网192.168.1中的任意主机连接数据库 |
+—————————–+——–+———————————————+
| 192.168.1.0/255.255.255.0 | test | 同上 |
+—————————–+——–+——————————————–+
sql的字符串通配符%表示匹配任意字符,可以是0个字符,通配符_表示匹配一个字符。
权限列指出在范围列中指定的用户拥有何种权限。该表使用grant语句的权限名称。对于绝大多数在user、db和host表中的权限列的名称与grant语句中有明显的联系。如select_priv对应于select权限。
3)授权表使用举例
grant用于给增加用户和创建权限,revoke用于删除用户权限。
下面是一些用grant增加用户和创建权限的例子:
mysql> grant all privileges on *.* to test@localhost identified by test with grant option;
这句增加一个本地具有所有权限的test用户(超级用户),密码是test。on子句中的*.*意味着”所有数据库、所有表”。with grant option表示它具有grant权限。
mysql> grant select,insert,update,delete,create,drop privileges on test.* to test1@192.168.1.0/255.255.255.0 identified by test;
这句是增加了一个test1用户,口令是test,但是它只能从c类子网192.168.1连接,对test库有select,insert,update,delete,create,drop操作权限。
用grant语句创建权限是不需要再手工刷新授权表的,因为它已经自动刷新了。
给用户创建权限还可以通过直接修改授权表:
mysql> insert into user values(“localhost”,”test”,password(“test”),”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”);
mysql> flush privileges;
这两句和上面第一句grant的效果是一样的,也是增加了一个本地的test超级用户。我们看到用grant方便多了,而且还不需flush privileges
。
mysql> insert into user (host,user,password) values(“192.168.1.0/255.255.255.0″,”test1”,password(“test”));
mysql> insert into db values(“192.168.1.0/255.255.255.0″,”test”,”test1″,”y”,”y”,”y”,”y”,”y”,”y”,”n”,”n”,”n”,”n”)
mysql> flush privileges;
这三句和上面第二句grant的效果也是一样的,也是增加了一个只能从c类子网192.168.1连接,对test库有select,insert,update,delete,create,drop操作权限的test1用户,口令是test。要取消一个用户的权限,使用revoke语句。revoke的语法非常类似于grant语句,除了to用from取代并且没有identified by和with grant
option子句,下面是用revoke删除用户权限的例子:
mysql> revoke all on test.* from test1@192.168.1.0/255.255.255.0;
这句revoke就撤消了上面第二句grant创建的权限,但是test1用户并没有被删除,必须手工从user表删除:
mysql> delete from user where user=test1;
mysql> flush privileges;
这样,test1用户就彻底删除了。
这些只是mysql授权表的简单使用,更多详细的资料请见mysql提供的手册。
3、编程需要注意的一些问题
不管是用哪种程序语言写连接mysql数据库的程序,有一条准则是永远不要相信用户提交的数据!
对于数字字段,我们要使用查询语句:select * from table where id=234,不要使用select * from table where id=234这样的查询语句。mysql会自动把字串转换为数字字符并且去除非数字字符。如果用户提交的数据经过了mysql_escape_string处理,这样我们就可以完全杜绝了sql inject攻击,关于sql inject攻击请参考下面链接的文章:
http://www.spidynamics.com/papers/sqlinjectionwhitepaper.pdf
http://www.ngssoftware.com/papers/advanced_sql_injection.pdf
各种编程语言该注意的问题:
1)所有web程序:
a)尝试在web表单输入单引号和双引号来测试可能出现的错误,并找出原因所在。
b)修改url参数带的%22 (“), %23 (#), 和 %27 ()。
c)对于数字字段的变量,我们的应用程序必须进行严格的检查,否则是非常危险的。
d)检查用户提交的数据是否超过字段的长度。
e)不要给自己程序连接数据库的用户过多的访问权限。
2)php:
a)检查用户提交的数据在查询之前是否经过addslashes处理,在php 4.0.3以后提供了基于mysql c api的函数mysql_escape_string()。
3)mysql c api:
a)检查查询字串是否用了mysql_escape_string() api调用。
4)mysql++:
a)检查查询字串是否用了escape和quote处理。
5)perl dbi:
a)检查查询字串是否用了quote()方法。
6)java jdbc:
a)检查查询字串是否用了preparedstatement对象。
4、一些小窍门
1)如果不慎忘记了mysql的root密码,我们可以在启动mysql服务器时加上参数–skip-grant-tables来跳过授权表的验证 (./safe_mysqld –skip-grant-tables &),这样我们就可以直接登陆mysql服务器,然后再修改root用户的口令,重启mysql就可以用新口令登陆了。
2)启动mysql服务器时加上–skip-show-database使一般数据库用户不能浏览其它数据库。
3)启动mysql服务器时加上–chroot=path参数,让mysqld守护进程运行在chroot环境中。这样sql语句load data infile和select … into outfile就限定在chroot_path下读写文件了。这里有一点要注意,mysql启动后会建立一个mysql.sock文件,默认是在/tmp目录下。使用了chroot后,mysql会在chroot_path/tmp去建立mysql.sock文件,如果没有chroot_path/tmp目录或启动mysql的用户没有这个目录写权限就不能建立mysql.sock文件,mysql会启动失败。比如我们加了–chroot=/usr/local/mysql/启动参数,那么最好建立一个启动mysql的用户能写的
/usr/local/mysql/tmp目录,当然我们也可以用–socket=path来指定mysql.sock文件的路径,但这个path一定要在chroot_path里面。
4)启动mysql服务器时加上–log-slow-queries[=file]参数,这样mysqld会把sql命令执行时间超过long_query_time的写入file文件。如果没有指定=file,mysqld默认会写到数据目录下的hostname-slow.log。如果只指定了filename,没有指定路径,那么mysqld也会把filename写到数据目录下。我们通过这个日志文件可以找出执行时间超长的查询语句,然后尽可能的优化它减轻mysql服务器的负担。
5)如果我们只需本机使用mysql服务,那么我们还可以加上–skip-networking启动参数使mysql不监听任何tcp/ip连接,增加安全性。(非常推荐)
6)mysql的更多mysqld启动选项请见mysql手册4.16.4 mysqld command-line options
5、references
mysql manual(http://www.mysql.com/documentation/index.html)
晏子的mysql管理员指南(http://clyan.hongnet.com/index.html)
access granted(http://www.devshed.com/server_side/mysql/access)