MySQL安全性指南(3)(转)

正在看的oracle教程是:mysql安全性指南(3)(转)。 mysql安全性指南(3)    

作 者: 晏子

2.4 不用grant设置用户

如果你有一个早于3.22.11的mysql版本,你不能使用grant(或revoke)语句设置用户及其访问权限,但你可以直接修改授权表的内容。如果你理解grant语句如何修改授权表,这很容易。那么你通过手工发出insert语句就能自己做同样的事情。

当你发出一条grant语句时,你指定一个用户名和主机名,可能还有口令。对该用户生成一个user表记录,并且这些值记录在user、host和password列中。如果你在grant语句中指定全局权限,这些权限记录在记录的权限列中。其中要留神的是grant语句为你加密口令,而insert不是,你需要在insert中使用password()函数加密口令。

如果你指定数据库级权限,用户名和主机名被记录在db表的user和host列。你为其授权的数据库记录在db列中,你授予的权限记录在权限列中。

对于表级和列级权限,效果是类似的。在tables_priv和columns_priv表中创建记录以记录用户名、主机名和数据库,还有相关的表和列。授予的权限记录在权限列中。

如果你还记得前面的介绍,你应该能即使不用grant语句也能做grant做的事情。记住在你直接修改授权表时,你将通知服务器重载授权表,否则他不知道你的改变。你可以执行一个mysqladmin flush-privileges或mysqladmin reload命令强迫一个重载。如果你忘记做这个,你会疑惑为什么服务器不做你想做的事情。

下列grant语句创建一个拥有所有权的超级用户。包括授权给别人的能力:

grant all on *.* to anyname@localhost identified by “passwd”

    with grant option

该语句将在user表中为anyname@localhost创建一个记录,打开所有权限,因为这里是超级用户(全局)权限存储的地方,要用insert语句做同样的事情,语句是:

insert into user  values(“localhost”,”anyname”,password(“passwd”),

    “y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”,”y”)

你可能发现它不工作,这要看你的mysql版本。授权表的结构已经改变而且你在你的user表可能没有14个权限列。用show columns找出你的授权表包含的每个权限列,相应地调整你的insert语句。 下列grant语句也创建一个拥有超级用户身份的用户,但是只有一个单个的权限:

grant reload on *.* to flush@localhost identified by “flushpass”

本例的insert语句比前一个简单,它很容易列出列名并只指定一个权限列。所有其它列将设置为缺省的”n”:

insert into user (host,password,reload) values(“localhost”,”flush”,password(“flushpass”),”y”)

数据库级权限用一个on db_name.*子句而不是on *.*进行授权:

grant all on sample.* to boris@localhost identified by “ruby”

这些权限不是全局的,所以它们不存储在user表中,我们仍然需要在user表中创建一条记录(使得用户能连接),但我们也需要创建一个db表记录记录数据库集权限:

insert into user (host,user,password) values(“localhost”,”boris”,password(“ruby”))

insert into db values(“localhost”,”sample_db”,”boris”,”y”,”y”,”y”,”y”,”y”,”y”,”n”,”y”,”y”,”y”)

“n”列是为grant权限;对末尾的一个数据库级具有with grant option的grant语句,你要设置该列为”y”。

要设置表级或列级权限,你对tables_priv或columns_priv使用insert语句。当然,如果你没有grant语句,你将没有这些表,因为它们在mysql中同时出现。如果你确实有这些表并且为了某些原因想要手工操作它们,要知道你不能用单独的列启用权限。

你设置tables_priv.table_priv或columns_priv.column_priv列来设置包含你想启用的权限值。例如,要对一个表启用select和insert权限,你要在相关的tables_priv的记录中设置table_priv为”select,insert”。

如果你想对一个拥有mysql账号的用户修改权限,使用update而不是insert,不管你增加或撤销权限都是这样。要完全删除一个用户,从用户使用的每个表中删除记录。

如果你愿意避免发一个查询来直接修改全权表,你可以看一下mysql自带的mysqlaccess和mysql_setpermissions脚本。

附录1 小测验

在你刚刚新安装了一个mysql服务器,在你增加了一个允许连接mysql的用户,用下列语句:

grant all on samp_db.* to fred@*.snake.net identified “cocoa”

而fred碰巧在服务器主机上有个账号,所以他试图连接服务器:

%mysql -u fred -pcocoa samp_db

error 1045: access denied for user: fred@localhost (using password: yes)

为什么?

原因是:

先考虑一下mysql_install_db如何建立初始权限表和服务器如何使用user表记录匹配客户连接。在你用mysql_install_db初始化你的数据库时,它创建类似这样的user表:

host user

localhost

pit.snake.net

localhost

pit.snake.net root

root

头两个记录允许root指定localhost或主机名连接本地服务器,后两个允许匿名用户从本地连接。当增加fred用户后,

host user

localhost

pit.snake.net

localhost

pit.snake.net

%.snake.net root

root

fred

在服务器启动时,它读取记录并排序它们(首先按主机,然后按主机上的用户),越具体越排在前面:

host user

localhost

localhost

pit.snake.net

pit.snake.net

%.snake.net root

root

fred

有localhost的两个记录排在一起,而对root的记录排在第一,因为它比空值更具体。pit.snake.net的记录也类似。所有这些均是没有任何通配符的字面上的host值,所以它们排在对fred记录的前面,特别是匿名用户排在fred之前。

结果是在fred试图从localhost连接时,host列中的一个空用户名的记录在包含%.snake.net的记录前匹配。该记录的口令是空的,因为缺省的匿名用户没有口令。因为在fred连接时指定了一个口令,由一个错配且连接失败。

这里要记住的是,虽然用通配符指定用户可以从其连接的主机是很方便。但你从本地主机连接时会有问题,只要你在table表中保留匿名用户记录。

一般地,建议你删除匿名用户记录:

mysql> delete from user where user=””;

更进一步,同时删除其他授权表中的任何匿名用户,有user列的表有db、tables_priv和columns_priv。

附录2 使一个新的mysql安装更安全

在你自己安装了一个新的mysql服务器后,你需要为mysql的root用户指定一个目录(缺省无口令),否则如果你忘记这点,你将你

[1] [2] 下一页

正在看的oracle教程是:mysql安全性指南(3)(转)。的mysql处于极不安全的状态(至少在一段时间内)。

在unix(linux)上,在按照手册的指令安装好mysql后,你必须运行mysql_install_db脚本建立包含授权表的mysql数据库和初始权限。在windows上,运行分发中的setup程序初始化数据目录和mysql数据库。假定服务器也在运行。

当你第一次在机器上安装mysql时,mysql数据库中的授权表是这样初始化的:

你可以从本地主机(localhost)上以root连接而不指定口令。root用户拥有所有权限(包括管理权限)并可做任何事情。(顺便说明,mysql超级用户与unix超级用户有相同的名字,他们彼此毫无关系。)

匿名访问被授予用户可从本地连接名为test和任何名字以test_开始的数据库。匿名用户可对数据库做任何事情,但无管理权限。

从本地主机多服务器的连接是允许的,不管连接的用户使用一个localhost主机名或真实主机名。如:

% mysql -h localhost test

% mysql -h pit.snake.net test

你以root连接mysql甚至不指定口令的事实只是意味着初始安装不安全,所以作为管理员的你首先要做的应该是设置root口令,然后根据你设置口令使用的方法,你也可以告诉服务器重载授权表是它知道这个改变。(在服务器启动时,它重载表到内存中而可能不知道你已经修改了它们。)

对mysql 3.22和以上版本,你可以用mysqladmin设置口令:

% mysqladmin -u root password yourpassword

对于mysql的任何版本,你可以用mysql程序并直接修改mysql数据库中的user授权表:

% mysql -u root mysql

mysql>update user set password=password(“yourpassword”) where user=”root”;

如果你有mysql的老版本,使用mysql和update。

在你设置完口令后,通过运行下列命令检查你是否需要告诉服务器重载授权表:

% mysqladmin -u root status

如果服务器仍然让你以root而不指定口令而连接服务器,重载授权表:

% mysqladmin -u root reload

在你设置了root的口令后(并且如果需要重载了授权表),你将需要在任何时候以root连接服务器时指定口令。

上一页  [1] [2] 

(0)
上一篇 2022年3月22日
下一篇 2022年3月22日

相关推荐