用好PostgreSQL role membership来管理继承组权限

作者: digoal


背景

在数据库中,如果你想把A用户创建的对象权限赋予给B用户,或者其他用户。

通常我们会对需要赋权的对象使用grant的语法来赋权。

但是这种方法比较繁琐,因为需要对每个对象,每一组需要的权限进行赋权。

如果你需要整个A用户所有对象的权限,以及它将来创建的对象的所有权限,有没有好的方法呢?

没错,你一定会想到role来管理。

role membership & inherit

例子

数据库有一个a 用户,创建了一些对象,需求是把a 创建的对象,自动赋予给b 用户。

--创建a用户
postgres=# create role a login;
CREATE ROLE

--创建b用户
postgres=# create role b login;
CREATE ROLE

--把a赋予给b
postgres=# grant a to b;
GRANT ROLE

--查看membership,可以看到b是a的member
postgres=# \du+ a
                  List of roles
 Role name | Attributes | Member of | Description 
-----------+------------+-----------+-------------
 a         |            | {}        | 

postgres=# \du+ b
                  List of roles
 Role name | Attributes | Member of | Description 
-----------+------------+-----------+-------------
 b         |            | {a}       | 

--使用a连接数据库,创建一张表
postgres=# \c postgres a
You are now connected to database "postgres" as user "a".
postgres=> create table r1(id int);
CREATE TABLE

--使用b连接数据库,可以直接使用这张表
postgres=> \c postgres b
You are now connected to database "postgres" as user "b".
postgres=> insert into r1 values (1);
INSERT 0 1
postgres=> \d+ r1
                          Table "public.r1"
 Column |  Type   | Modifiers | Storage | Stats target | Description 
--------+---------+-----------+---------+--------------+-------------
 id     | integer |           | plain   |              | 

postgres=> \dt+ r1
                    List of relations
 Schema | Name | Type  | Owner |    Size    | Description 
--------+------+-------+-------+------------+-------------
 public | r1   | table | a     | 8192 bytes | 
(1 row)

如果b用户为noinherit的,那么需要set role才能切换到对应的role.

--将b设置为noinherit
postgres=> \c postgres postgres
You are now connected to database "postgres" as user "postgres".
postgres=# alter role b noinherit;
ALTER ROLE

--那么b不会自动继承a这个角色,需要显示的set role a;
postgres=# \c postgres b
You are now connected to database "postgres" as user "b".
postgres=> insert into r1 values (1);
ERROR:  permission denied for relation r1

postgres=> set role a;
SET
postgres=> insert into r1 values (1);
INSERT 0 1

另一种加入member的方法是在创建角色时加入,可以一次指定多个.

postgres=# create role c  in role a login;
CREATE ROLE
postgres=# \du+ c
                  List of roles
 Role name | Attributes | Member of | Description 
-----------+------------+-----------+-------------
 c         |            | {a}       | 


--d用户登陆后,会自动继承a,b的权限
postgres=# create role d  in role a,b login;
CREATE ROLE
postgres=# \du+ d
                  List of roles
 Role name | Attributes | Member of | Description 
-----------+------------+-----------+-------------
 d         |            | {a,b}     | 

WITH ADMIN OPTION

与SQL标准一样,加了WITH ADMIN OPTION 则允许被授予的用户继续将权限授予给其他人。

postgres=# grant a to b with admin option;
GRANT ROLE

postgres=# \c postgres b
You are now connected to database "postgres" as user "b".

postgres=> grant a to digoal;
GRANT ROLE

postgres=> \c postgres c
You are now connected to database "postgres" as user "c".

postgres=> grant a to digoal;
ERROR:  must have admin option on role "a"

default privilege

使用角色继承来管理有些时候还是不能满足业务需求,因为业务可能只是想把少量的权限给其他用户,而不是所有权限。

例如,使用角色继承的方法是比较危险的,被授予权限的用户,可以删除对象。

postgres=# alter role b inherit;
ALTER ROLE
postgres=# \c postgres b
You are now connected to database "postgres" as user "b".
postgres=> \dt+ r1
                    List of relations
 Schema | Name | Type  | Owner |    Size    | Description 
--------+------+-------+-------+------------+-------------
 public | r1   | table | a     | 8192 bytes | 
(1 row)

postgres=> drop table r1;
DROP TABLE

如果只想要a用户将来创建的所有表的查询权限,怎么做呢?

Command:     ALTER DEFAULT PRIVILEGES
Description: define default access privileges
Syntax:
ALTER DEFAULT PRIVILEGES
    [ FOR { ROLE | USER } target_role [, ...] ]
    [ IN SCHEMA schema_name [, ...] ]
    abbreviated_grant_or_revoke

where abbreviated_grant_or_revoke is one of:

GRANT { { SELECT | INSERT | UPDATE | DELETE | TRUNCATE | REFERENCES | TRIGGER }
    [, ...] | ALL [ PRIVILEGES ] }
    ON TABLES
    TO { [ GROUP ] role_name | PUBLIC } [, ...] [ WITH GRANT OPTION ]

GRANT { { USAGE | SELECT | UPDATE }
    [, ...] | ALL [ PRIVILEGES ] }
    ON SEQUENCES
    TO { [ GROUP ] role_name | PUBLIC } [, ...] [ WITH GRANT OPTION ]

GRANT { EXECUTE | ALL [ PRIVILEGES ] }
    ON FUNCTIONS
    TO { [ GROUP ] role_name | PUBLIC } [, ...] [ WITH GRANT OPTION ]

GRANT { USAGE | ALL [ PRIVILEGES ] }
    ON TYPES
    TO { [ GROUP ] role_name | PUBLIC } [, ...] [ WITH GRANT OPTION ]

REVOKE [ GRANT OPTION FOR ]
    { { SELECT | INSERT | UPDATE | DELETE | TRUNCATE | REFERENCES | TRIGGER }
    [, ...] | ALL [ PRIVILEGES ] }
    ON TABLES
    FROM { [ GROUP ] role_name | PUBLIC } [, ...]
    [ CASCADE | RESTRICT ]

REVOKE [ GRANT OPTION FOR ]
    { { USAGE | SELECT | UPDATE }
    [, ...] | ALL [ PRIVILEGES ] }
    ON SEQUENCES
    FROM { [ GROUP ] role_name | PUBLIC } [, ...]
    [ CASCADE | RESTRICT ]

REVOKE [ GRANT OPTION FOR ]
    { EXECUTE | ALL [ PRIVILEGES ] }
    ON FUNCTIONS
    FROM { [ GROUP ] role_name | PUBLIC } [, ...]
    [ CASCADE | RESTRICT ]

REVOKE [ GRANT OPTION FOR ]
    { USAGE | ALL [ PRIVILEGES ] }
    ON TYPES
    FROM { [ GROUP ] role_name | PUBLIC } [, ...]
    [ CASCADE | RESTRICT ]

例子

postgres=# alter default privileges for role a grant select on tables to b;
ALTER DEFAULT PRIVILEGES

postgres=# \c postgres a
You are now connected to database "postgres" as user "a".
postgres=> create table r1(id int);
CREATE TABLE
postgres=> insert into r1 values (1);
INSERT 0 1
postgres=> \c postgres b
You are now connected to database "postgres" as user "b".
postgres=> select * from r1;
 id 
----
  1
(1 row)

/images/news/2016/pg_bot_banner.jpg
请在登录后发表评论,否则无法保存。
1楼 xcvxcvsdf
2025-01-01 15:19:02+08

https://zulin.tiancebbs.cn/sh/3541.html https://zulin.tiancebbs.cn/sh/1889.html https://sh.tiancebbs.cn/hjzl/468456.html https://zulin.tiancebbs.cn/sh/3219.html https://taicang.tiancebbs.cn/hjzl/458315.html https://www.tiancebbs.cn/ershoufang/471966.html https://xjxinjinqu.tiancebbs.cn/qths/462703.html https://su.tiancebbs.cn/hjzl/456547.html https://www.tiancebbs.cn/ershoufang/471910.html https://zulin.tiancebbs.cn/sh/3228.html https://zulin.tiancebbs.cn/sh/3866.html https://www.tiancebbs.cn/ershouwang/467741.html https://taicang.tiancebbs.cn/hjzl/459824.html https://taicang.tiancebbs.cn/hjzl/463710.html https://changshushi.tiancebbs.cn/hjzl/460112.html https://aihuishou.tiancebbs.cn/sh/2155.html https://www.tiancebbs.cn/qtfwxx/473305.html

2楼 xiaowu
2024-04-22 09:29:55+08

很甜很撩的网名女生:https://www.nanss.com/mingcheng/1803.html 培训简讯:https://www.nanss.com/gongzuo/2091.html 我的梦想演讲稿600字:https://www.nanss.com/xuexi/2299.html 深夜致自己的短句:https://www.nanss.com/wenan/1902.html 高考必胜:https://www.nanss.com/xuexi/2117.html 细节决定成败读后感:https://www.nanss.com/xuexi/2226.html 配风景的文案:https://www.nanss.com/wenan/2285.html 以变化为话题的作文:https://www.nanss.com/xuexi/2108.html 兄弟情深的经典句子:https://www.nanss.com/xuexi/1993.html 高级有质感的句子:https://www.nanss.com/wenan/2501.html 谢谢你的礼物作文600字:https://www.nanss.com/xuexi/2422.html 感谢家长的配合支持句子:https://www.nanss.com/gongzuo/2221.html 绿色环保作文500字:https://www.nanss.com/xuexi/2063.html 桂花的作文:https://www.nanss.com/xuexi/2164.html 曾经的长发飘飘变短发的说说:https://www.nanss.com/wenan/2404.html 最酷群名:https://www.nanss.com/mingcheng/2042.html 一听就不好惹的名字:https://www.nanss.com/mingcheng/2259.html 四十不惑的朋友圈精辟:https://www.nanss.com/wenan/2279.html 爸爸生日幽默说说:https://www.nanss.com/wenan/1857.html 根鸟读后感:https://www.nanss.com/xuexi/2136.html 我的动物朋友:https://www.nanss.com/xuexi/2138.html 夸奖群主的精美句子:https://www.nanss.com/yulu/2293.html 捐款感谢信:https://www.nanss.com/shenghuo/2435.html 给老师的一封信作文:https://www.nanss.com/xuexi/2186.html 毕业作文:https://www.nanss.com/xuexi/2355.html 爱情的句子唯美短句:https://www.nanss.com/yulu/2506.html 在尝试中成长作文600字:https://www.nanss.com/xuexi/2027.html 感恩作文:https://www.nanss.com/xuexi/2178.html 仿写句子:https://www.nanss.com/xuexi/2040.html 经典短句早安:https://www.nanss.com/wenan/1909.html

© 2010 PostgreSQL中文社区