mysql存储过程的definer和invoker

【存储过程的权限】mysql

ALTER ROUTINE 编辑或删除存储过程sql

CREATE ROUTINE 建立存储过程数据库

EXECUTE运行存储过程安全


【存储过程的建立语法】函数

delimiter //    -- 声明分隔符(命令结束符)this

create ci

 definer = user@hostname | current_user it

 procedure 存储过程名 (参数)io

 comment '注释'ast

 sql security definer | invoker   -- sql 的安全设置

begin

  存储过程的body

end

//

delimiter ;    -- 声明分隔符(命令结束符)


【函数的建立语句】

delimiter //    -- 声明分隔符(命令结束符)

create

 definer = user@hostname | current_user

 function 函数名(参数)

 return 返回值类型

 comment '注释'

 sql security definer | invoker   -- sql 的安全设置

begin

  函数的body

end


//

delimiter ;    -- 声明分隔符(命令结束符)


【definer和invoker的解释】

    建立存储过程的时候能够指定 SQL SECURITY属性,设置为 DEFINER 或者INVOKER,用来奉告mysql在执行存储过程的时候,,是以DEFINER用户的权限来执行,仍是以调用者的权限来执行。

   默认状况下,使用DEFINER方式,此时调用存储过程的用户必须有存储过程的EXECUTE权限,而且DEFINER指定的用户必须是在mysql.user表中存在的用户。

   DEFINER模式下,默认DEFINER=CURRENT_USER,在存储过程执行时,mysql会检查DEFINER定义的用户'user_name'@'host_name'的权限;

   INVOKER模式下,在存储过程执行时,会检查存储过程调用者的权限。


  案例一:DEFINER

   CREATE DEFINER = 'admin'@'localhost' PROCEDURE account_count()

   BEGIN

       SELECT 'Number of accounts:', COUNT(*) FROM mysql.user;

   END;

   在这个案例中,不论哪一个用户A调用存储过程,存储过程都会以'admin'@'localhost'的权限去执行,即便这个用户A没有查询mysql.user表的权限。


 案例二:INVOKER

   CREATE DEFINER = 'admin'@'localhost' PROCEDURE account_count()

       SQL SECURITY INVOKER

   BEGIN

       SELECT 'Number of accounts:', COUNT(*) FROM mysql.user;

   END;

   在这个案例中,虽然存储过程语句中仍然带有DEFINER参数,可是因为SQL SECURITY指定了INVOKER,因此在存储过程执行的时候,会以调用者的额身份去执行。此时这个存储过程是否能成功执行,取决于调用者是否有mysql.user表的查询权限。


【案例】

案例一:调用存储过程

 存储过程的调用者是 : admin@192.168.1.1

 存储过程的DEFINER是   : admin@%

 MySQL中存在的用户是 : admin@192.168.%.%


   此时admin@192.168.1.1是能够访问数据库的,由于它符合admin@192.168.%.%的受权规则,可是当它调用DEFINER='admin@%'的存储过程的时候,mysql会检查mysql.user用户表中是否存在admin@%这个用户,mysql的检查结果是admin@%这个用户不存在,此时就会返回报错,提示“Ther user specified as a definer ('admin@%') does not exist.。

 

案例二:建立存储过程


 使用用户admin@192.168.1.1链接mysql,该用户有test库的all privileges,执行建立存储过程的操做:

         存储过程当中定义的DEFINER是   : admin@%

         MySQL中存在的用户是 : admin@192.168.%.%

 此时,会遇到报错,提示”ERROR 1227 (42000): Access denied; you need (at least one of) the SUPER privilege(s) for this operation“

 修复DEFINER='admin@192.168.%.%',或者去掉 DEFINER参数,均可以恢复正常。


说明:

   案例一中是存在问题的,存储过程的调用者和拥有者都是admin@192.168.1.1,可是DEFINER倒是admin@%,这是因为建立存储过程的命令是由root用户执行的,因此没有遇到案例二中的报错。


【存储过程经常使用命令】


查看存储过程的建立语句:

show create procedure 存储过程名;

查看存储过程的信息:

show procedure status like '存储过程名'G

查看存储过程的Definer信息:

select db,name,type,sql_security,definer from mysql.proc where  type='PROCEDURE' and db='数据库名' ;

修改存储过程的DEFINER:

update  mysql.proc  set `definer` ='admin@192.168.%.%' where db like 'db_%';

相关文章
相关标签/搜索