mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-12 18:34:36 +08:00
Use appropriate command type when retrieving relation's policies.
When retrieving policies, if not working on the root target relation, we actually want the relation's SELECT policies, regardless of the top level query command type. For example in UPDATE t1...FROM t2 we need to apply t1's UPDATE policies and t2's SELECT policies. Previously top level query command type was applied to all relations, which was wrong. Add some regression coverage to ensure we don't violate this principle in the future. Report and patch by Dean Rasheed. Cherry picked from larger refactoring patch and tweaked by me. Back-patched to 9.5 where RLS was introduced.
This commit is contained in:
parent
8693ebe37d
commit
1e15b21229
@ -147,8 +147,18 @@ get_row_security_policies(Query *root, CmdType commandType, RangeTblEntry *rte,
|
||||
return;
|
||||
}
|
||||
|
||||
/* Grab the built-in policies which should be applied to this relation. */
|
||||
/*
|
||||
* RLS is enabled for this relation.
|
||||
*
|
||||
* Get the security policies that should be applied, based on the command
|
||||
* type. Note that if this isn't the target relation, we actually want
|
||||
* the relation's SELECT policies, regardless of the query command type,
|
||||
* for example in UPDATE t1 ... FROM t2 we need to apply t1's UPDATE
|
||||
* policies and t2's SELECT policies.
|
||||
*/
|
||||
rel = heap_open(rte->relid, NoLock);
|
||||
if (rt_index != root->resultRelation)
|
||||
commandType = CMD_SELECT;
|
||||
|
||||
rowsec_policies = pull_row_security_policies(commandType, rel,
|
||||
user_id);
|
||||
|
@ -3033,6 +3033,89 @@ CREATE POLICY p ON t USING (max(c)); -- fails: aggregate functions are not allow
|
||||
ERROR: aggregate functions are not allowed in policy expressions
|
||||
ROLLBACK;
|
||||
--
|
||||
-- Non-target relations are only subject to SELECT policies
|
||||
--
|
||||
SET SESSION AUTHORIZATION rls_regress_user0;
|
||||
CREATE TABLE r1 (a int);
|
||||
CREATE TABLE r2 (a int);
|
||||
INSERT INTO r1 VALUES (10), (20);
|
||||
INSERT INTO r2 VALUES (10), (20);
|
||||
GRANT ALL ON r1, r2 TO rls_regress_user1;
|
||||
CREATE POLICY p1 ON r1 USING (true);
|
||||
ALTER TABLE r1 ENABLE ROW LEVEL SECURITY;
|
||||
CREATE POLICY p1 ON r2 FOR SELECT USING (true);
|
||||
CREATE POLICY p2 ON r2 FOR INSERT WITH CHECK (false);
|
||||
CREATE POLICY p3 ON r2 FOR UPDATE USING (false);
|
||||
CREATE POLICY p4 ON r2 FOR DELETE USING (false);
|
||||
ALTER TABLE r2 ENABLE ROW LEVEL SECURITY;
|
||||
SET SESSION AUTHORIZATION rls_regress_user1;
|
||||
SELECT * FROM r1;
|
||||
a
|
||||
----
|
||||
10
|
||||
20
|
||||
(2 rows)
|
||||
|
||||
SELECT * FROM r2;
|
||||
a
|
||||
----
|
||||
10
|
||||
20
|
||||
(2 rows)
|
||||
|
||||
-- r2 is read-only
|
||||
INSERT INTO r2 VALUES (2); -- Not allowed
|
||||
ERROR: new row violates row level security policy for "r2"
|
||||
UPDATE r2 SET a = 2 RETURNING *; -- Updates nothing
|
||||
a
|
||||
---
|
||||
(0 rows)
|
||||
|
||||
DELETE FROM r2 RETURNING *; -- Deletes nothing
|
||||
a
|
||||
---
|
||||
(0 rows)
|
||||
|
||||
-- r2 can be used as a non-target relation in DML
|
||||
INSERT INTO r1 SELECT a + 1 FROM r2 RETURNING *; -- OK
|
||||
a
|
||||
----
|
||||
11
|
||||
21
|
||||
(2 rows)
|
||||
|
||||
UPDATE r1 SET a = r2.a + 2 FROM r2 WHERE r1.a = r2.a RETURNING *; -- OK
|
||||
a | a
|
||||
----+----
|
||||
12 | 10
|
||||
22 | 20
|
||||
(2 rows)
|
||||
|
||||
DELETE FROM r1 USING r2 WHERE r1.a = r2.a + 2 RETURNING *; -- OK
|
||||
a | a
|
||||
----+----
|
||||
12 | 10
|
||||
22 | 20
|
||||
(2 rows)
|
||||
|
||||
SELECT * FROM r1;
|
||||
a
|
||||
----
|
||||
11
|
||||
21
|
||||
(2 rows)
|
||||
|
||||
SELECT * FROM r2;
|
||||
a
|
||||
----
|
||||
10
|
||||
20
|
||||
(2 rows)
|
||||
|
||||
SET SESSION AUTHORIZATION rls_regress_user0;
|
||||
DROP TABLE r1;
|
||||
DROP TABLE r2;
|
||||
--
|
||||
-- Clean up objects
|
||||
--
|
||||
RESET SESSION AUTHORIZATION;
|
||||
|
@ -1298,6 +1298,46 @@ CREATE TABLE t (c) AS VALUES ('bar'::text);
|
||||
CREATE POLICY p ON t USING (max(c)); -- fails: aggregate functions are not allowed in policy expressions
|
||||
ROLLBACK;
|
||||
|
||||
--
|
||||
-- Non-target relations are only subject to SELECT policies
|
||||
--
|
||||
SET SESSION AUTHORIZATION rls_regress_user0;
|
||||
CREATE TABLE r1 (a int);
|
||||
CREATE TABLE r2 (a int);
|
||||
INSERT INTO r1 VALUES (10), (20);
|
||||
INSERT INTO r2 VALUES (10), (20);
|
||||
|
||||
GRANT ALL ON r1, r2 TO rls_regress_user1;
|
||||
|
||||
CREATE POLICY p1 ON r1 USING (true);
|
||||
ALTER TABLE r1 ENABLE ROW LEVEL SECURITY;
|
||||
|
||||
CREATE POLICY p1 ON r2 FOR SELECT USING (true);
|
||||
CREATE POLICY p2 ON r2 FOR INSERT WITH CHECK (false);
|
||||
CREATE POLICY p3 ON r2 FOR UPDATE USING (false);
|
||||
CREATE POLICY p4 ON r2 FOR DELETE USING (false);
|
||||
ALTER TABLE r2 ENABLE ROW LEVEL SECURITY;
|
||||
|
||||
SET SESSION AUTHORIZATION rls_regress_user1;
|
||||
SELECT * FROM r1;
|
||||
SELECT * FROM r2;
|
||||
|
||||
-- r2 is read-only
|
||||
INSERT INTO r2 VALUES (2); -- Not allowed
|
||||
UPDATE r2 SET a = 2 RETURNING *; -- Updates nothing
|
||||
DELETE FROM r2 RETURNING *; -- Deletes nothing
|
||||
|
||||
-- r2 can be used as a non-target relation in DML
|
||||
INSERT INTO r1 SELECT a + 1 FROM r2 RETURNING *; -- OK
|
||||
UPDATE r1 SET a = r2.a + 2 FROM r2 WHERE r1.a = r2.a RETURNING *; -- OK
|
||||
DELETE FROM r1 USING r2 WHERE r1.a = r2.a + 2 RETURNING *; -- OK
|
||||
SELECT * FROM r1;
|
||||
SELECT * FROM r2;
|
||||
|
||||
SET SESSION AUTHORIZATION rls_regress_user0;
|
||||
DROP TABLE r1;
|
||||
DROP TABLE r2;
|
||||
|
||||
--
|
||||
-- Clean up objects
|
||||
--
|
||||
|
Loading…
Reference in New Issue
Block a user