IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> Postgresql行安全策略(Row Security Policies) -> 正文阅读

[大数据]Postgresql行安全策略(Row Security Policies)

PG行安全策略其实就是对不同用户,不同行数据的可见性,和可修改性。在我看来其实就是权限的维度不同,PG在9.5之前提供表级别,列级别的权限控制,9.5之后添加行级策略,这样用户就可以更加灵活的控制数据,对不同用户之间的数据隐私起到了保护作用。

先举一个列权限的例子,也可以通过视图的方式进行权限限制:

#一张表sex
\c postgres postgres 
postgres=> \d  sex
                    Table "public.sex"
  Column   |     Type     | Collation | Nullable | Default 
-----------+--------------+-----------+----------+---------
 sex       | character(1) |           |          | 
 otherinfo | text         |           |          | 
Indexes:
    "idx_sex_1" btree (sex)
#赋予otherinfo列给hank查看
grant select (otherinfo) on sex to hank;

#hank登陆后,查询所有列,报错
\c postgres hank
select * from sex;
ERROR:  permission denied for table sex

#查询otherinfo列,可正常查询
select otherinfo from sex;
      otherinfo      
---------------------
 4157362this is test

表默认是没有任何行安全策略的, 行安全策略可以针对命令和角色指定。如命令SELECT、INSERT、UPDATE或DELETE。

 CREATE POLICY    #创建
  ALTER POLICY    #修改
   DROP POLICY    #删除

命令帮助如下:

Command:     CREATE POLICY
Description: define a new row level security policy for a table
Syntax:
CREATE POLICY name ON table_name
    [ AS { PERMISSIVE | RESTRICTIVE } ]
    [ FOR { ALL | SELECT | INSERT | UPDATE | DELETE } ]
    [ TO { role_name | PUBLIC | CURRENT_USER | SESSION_USER } [, ...] ]
    [ USING ( using_expression ) ]
    [ WITH CHECK ( check_expression ) ]

这里需要注意USING和WITH CHECK选项:

  1. USING语句用于检查现有表中行的策略表达式
  2. WITH CHECK语句用于检查新行,也就是insert和update
#有accounts这样一张表
postgres=> \d accounts
                          Table "public.accounts"
 Column |  Type   | Collation | Nullable |             Default              
--------+---------+-----------+----------+----------------------------------
 id     | integer |           | not null | generated by default as identity
 number | text    |           |          | 
 client | text    |           |          | 
 amount | numeric |           |          | 
Indexes:
    "accounts_pkey" PRIMARY KEY, btree (id)
    "accounts_number_key" UNIQUE CONSTRAINT, btree (number)
#数据如下
postgres=# select * from accounts;
 id | number | client |  amount  
----+--------+--------+----------
  1 | 1001   | alice  |  1000.00
  2 | 2001   | bob    | 910.0000
  3 | 2002   | bob    |     0.00
  4 | a      | a      |        3
(4 rows)
#创建一个policy,这里是给hank用户查询
CREATE POLICY account_ids ON accounts TO hank  USING (id = 1 or id =2);

#启用策略
ALTER TABLE accounts ENABLE ROW LEVEL SECURITY

#使用hank可以查看到id为1和2的行
postgres=# \c  postgres hank
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "hank".
postgres=> select * from accounts;
 id | number | client |  amount  
----+--------+--------+----------
  1 | 1001   | alice  |  1000.00
  2 | 2001   | bob    | 910.0000

如果启用行策略,但是实际没有行策略,那么查询不到数据

postgres=> \c postgres postgres 
postgres=# \d  accounts
                          Table "public.accounts"
 Column |  Type   | Collation | Nullable |             Default              
--------+---------+-----------+----------+----------------------------------
 id     | integer |           | not null | generated by default as identity
 number | text    |           |          | 
 client | text    |           |          | 
 amount | numeric |           |          | 
Indexes:
    "accounts_pkey" PRIMARY KEY, btree (id)
    "accounts_number_key" UNIQUE CONSTRAINT, btree (number)
Policies (forced row security enabled): (none)

postgres=# \c postgres hank
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "hank".
postgres=> select * from accounts;
(0 rows)

如果用户想要访问所有数据,不受行策略影响,也可以修改role属性为bypassrls,绕过行安全策略,示例如下:

postgres=> \c postgres postgres 
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "postgres".
postgres=# alter role hank bypassrls ;
ALTER ROLE
postgres=# \c postgres hank
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "hank".
postgres=> select * from accounts;
 id | number | client |  amount  
----+--------+--------+----------
  1 | 1001   | alice  |  1000.00
  2 | 2001   | bob    | 910.0000
  3 | 2002   | bob    |     0.00
  4 | a      | a      |        3

可查询策略相关视图:

postgres=> select * from pg_policies ;
 schemaname | tablename | policyname  | permissive | roles  | cmd |          qual          | with_check 
------------+-----------+-------------+------------+--------+-----+------------------------+------------
 public     | accounts  | account_ids | PERMISSIVE | {hank} | ALL | ((id = 1) OR (id = 2)) | 
(1 row)

ostgres=> select * from pg_policy;
-[ RECORD 1 ]-+---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
oid           | 106391
polname       | account_ids
polrelid      | 48472
polcmd        | *
polpermissive | t
polroles      | {16449}
polqual       | {BOOLEXPR :boolop or :args ({OPEXPR :opno 96 :opfuncid 65 :opresulttype 16 :opretset false :opcollid 0 :inputcollid 0 :args ({VAR :varno 1 :varattno 1 :vartype 23 :vartypmod -1 :varcollid 0 :varlevelsup 0 :varnoold 1 :varoattno 1 :location 54} {CONST :consttype 23 :consttypmod -1 :constcollid 0 :constlen 4 :constbyval true :constisnull false :location 59 :constvalue 4 [ 1 0 0 0 0 0 0 0 ]}) :location 57} {OPEXPR :opno 96 :opfuncid 65 :opresulttype 16 :opretset false :opcollid 0 :inputcollid 0 :args ({VAR :varno 1 :varattno 1 :vartype 23 :vartypmod -1 :varcollid 0 :varlevelsup 0 :varnoold 1 :varoattno 1 :location 64} {CONST :consttype 23 :consttypmod -1 :constcollid 0 :constlen 4 :constbyval true :constisnull false :location 69 :constvalue 4 [ 2 0 0 0 0 0 0 0 ]}) :location 67}) :location 61}
polwithcheck  | 

另外一下两个解释一下:
PERMISSIVE:默认选项,适用于一个给定查询的所有宽容性策略将被使用布尔“OR”操作符组合在一起。通过创该策略,可以在能被访问的记录集合中进行增加。如:

CREATE POLICY account_ids on accounts  TO hank  USING (id = 1 or  id =2);
#增加id=3,hank用户查询的时候,可以显示id=3的记录
CREATE POLICY account_ids on accounts  TO hank  USING (id = 1 or  id =2  or id =3);

RESTRICTIVE:限制性策略。适用于一个给定查询的所有限制性策略将被使用布
尔“AND”操作符组合在一起。通过创建限制性策略,管理员可以减少能被访问的记录集合,因为每一条记录都必须通过所有的限制性策略。在限制性策略作用之前,需要至少一条PERMISSIVE策略授予行的访问权限。如果只有限制性策略存在,则所有记录都不能被访问。当PERMISSIVE和RESTRICTIVE混合存在时,只有当一个记录能通过至少一条PERMISSIVE策略以及所有的限制性策略时,该记录才是可访问的。

以上说明是官方文档解释,一下难以理解,下面举个例子说明:
只有RESTRICTIVE的policy,记录都不显示

postgres=# CREATE POLICY account_ids_restric on accounts as RESTRICTIVE  TO hank  USING (id = 1 and client='alice');
CREATE POLICY
postgres=# \c  postgres hank
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "hank".
postgres=> select * from accounts;
 id | number | client | amount 
----+--------+--------+--------
(0 rows)

添加PERMISSIVE属性的policy

postgres=> \c postgres postgres 
psql (13.4, server 12.6)
You are now connected to database "postgres" as user "hank".
postgres=# CREATE POLICY account_ids on accounts  TO hank  USING (id = 1 or  id =2);
CREATE POLICY

#可见id = 1 and client='alice'的行了
postgres=# \c  postgres hank
psql (13.4, server 12.6)
postgres=> select * from accounts;
 id | number | client | amount  
----+--------+--------+---------
  1 | 1001   | alice  | 1000.00
  
postgres=> \d accounts
                          Table "public.accounts"
 Column |  Type   | Collation | Nullable |             Default              
--------+---------+-----------+----------+----------------------------------
 id     | integer |           | not null | generated by default as identity
 number | text    |           |          | 
 client | text    |           |          | 
 amount | numeric |           |          | 
Indexes:
    "accounts_pkey" PRIMARY KEY, btree (id)
    "accounts_number_key" UNIQUE CONSTRAINT, btree (number)
Policies (forced row security enabled):
    POLICY "account_ids"
      TO hank
      USING (((id = 1) OR (id = 2)))
    POLICY "account_ids_restric" AS RESTRICTIVE
      TO hank
      USING (((id = 1) AND (client = 'alice'::text)))
      
postgres=> select * from pg_policies ;
 schemaname | tablename |     policyname      | permissive  |    roles    |  cmd   |                           qual                            | with_check 
------------+-----------+---------------------+-------------+-------------+--------+-----------------------------------------------------------+------------
 public     | accounts  | account_ids_restric | RESTRICTIVE | {hank}      | ALL    | ((id = 1) AND (client = 'alice'::text))                   | 
 public     | accounts  | account_ids         | PERMISSIVE  | {hank}      | ALL    | ((id = 1) OR (id = 2))                                    | 

参考:
https://www.postgresql.org/docs/13/sql-createpolicy.html
https://www.postgresql.org/docs/13/ddl-rowsecurity.html

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2021-10-30 12:37:23  更:2021-10-30 12:39:04 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/18 1:57:03-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码