In REFRESH MATERIALIZED VIEW, set user ID before running user code.
authorNoah Misch <[email protected]>
Mon, 9 May 2022 15:35:08 +0000 (08:35 -0700)
committerNoah Misch <[email protected]>
Mon, 9 May 2022 15:35:13 +0000 (08:35 -0700)
It intended to, but did not, achieve this.  Adopt the new standard of
setting user ID just after locking the relation.  Back-patch to v10 (all
supported versions).

Reviewed by Simon Riggs.  Reported by Alvaro Herrera.

Security: CVE-2022-1552

src/backend/commands/matview.c
src/test/regress/expected/privileges.out
src/test/regress/sql/privileges.sql

index 2763c49e6aed1e384c82c9ef8690c982aa39dd83..d10eac8a1c1847511c5f1f002d56667544accead 100644 (file)
@@ -164,6 +164,17 @@ ExecRefreshMatView(RefreshMatViewStmt *stmt, const char *queryString,
                                          lockmode, false, false,
                                          RangeVarCallbackOwnsTable, NULL);
    matviewRel = heap_open(matviewOid, NoLock);
+   relowner = matviewRel->rd_rel->relowner;
+
+   /*
+    * Switch to the owner's userid, so that any functions are run as that
+    * user.  Also lock down security-restricted operations and arrange to
+    * make GUC variable changes local to this command.
+    */
+   GetUserIdAndSecContext(&save_userid, &save_sec_context);
+   SetUserIdAndSecContext(relowner,
+                          save_sec_context | SECURITY_RESTRICTED_OPERATION);
+   save_nestlevel = NewGUCNestLevel();
 
    /* Make sure it is a materialized view. */
    if (matviewRel->rd_rel->relkind != RELKIND_MATVIEW)
@@ -268,19 +279,6 @@ ExecRefreshMatView(RefreshMatViewStmt *stmt, const char *queryString,
     */
    SetMatViewPopulatedState(matviewRel, !stmt->skipData);
 
-   relowner = matviewRel->rd_rel->relowner;
-
-   /*
-    * Switch to the owner's userid, so that any functions are run as that
-    * user.  Also arrange to make GUC variable changes local to this command.
-    * Don't lock it down too tight to create a temporary table just yet.  We
-    * will switch modes when we are about to execute user code.
-    */
-   GetUserIdAndSecContext(&save_userid, &save_sec_context);
-   SetUserIdAndSecContext(relowner,
-                          save_sec_context | SECURITY_LOCAL_USERID_CHANGE);
-   save_nestlevel = NewGUCNestLevel();
-
    /* Concurrent refresh builds new data in temp tablespace, and does diff. */
    if (concurrent)
    {
@@ -303,12 +301,6 @@ ExecRefreshMatView(RefreshMatViewStmt *stmt, const char *queryString,
    LockRelationOid(OIDNewHeap, AccessExclusiveLock);
    dest = CreateTransientRelDestReceiver(OIDNewHeap);
 
-   /*
-    * Now lock down security-restricted operations.
-    */
-   SetUserIdAndSecContext(relowner,
-                          save_sec_context | SECURITY_RESTRICTED_OPERATION);
-
    /* Generate the data, if wanted. */
    if (!stmt->skipData)
        processed = refresh_matview_datafill(dest, dataQuery, queryString);
index 6c812f70a64edb4819dcd5d5cc2542680340fa86..890a486645693048cdc9b282d7fe56ca58089520 100644 (file)
@@ -1363,6 +1363,22 @@ CONTEXT:  SQL function "unwanted_grant" statement 1
 SQL statement "SELECT unwanted_grant()"
 PL/pgSQL function sro_trojan() line 1 at PERFORM
 SQL function "mv_action" statement 1
+-- REFRESH MATERIALIZED VIEW CONCURRENTLY use of eval_const_expressions()
+SET SESSION AUTHORIZATION regress_sro_user;
+CREATE FUNCTION unwanted_grant_nofail(int) RETURNS int
+   IMMUTABLE LANGUAGE plpgsql AS $$
+BEGIN
+   PERFORM unwanted_grant();
+   RAISE WARNING 'owned';
+   RETURN 1;
+EXCEPTION WHEN OTHERS THEN
+   RETURN 2;
+END$$;
+CREATE MATERIALIZED VIEW sro_index_mv AS SELECT 1 AS c;
+CREATE UNIQUE INDEX ON sro_index_mv (c) WHERE unwanted_grant_nofail(1) > 0;
+\c -
+REFRESH MATERIALIZED VIEW CONCURRENTLY sro_index_mv;
+REFRESH MATERIALIZED VIEW sro_index_mv;
 DROP OWNED BY regress_sro_user;
 DROP ROLE regress_sro_user;
 -- Admin options
index 81a86ebbf46fb2bbecf36adef1115db37bc46fe7..b136758f40259630b97915fe784850a4370e9c0d 100644 (file)
@@ -844,6 +844,23 @@ REFRESH MATERIALIZED VIEW sro_mv;
 REFRESH MATERIALIZED VIEW sro_mv;
 BEGIN; SET CONSTRAINTS ALL IMMEDIATE; REFRESH MATERIALIZED VIEW sro_mv; COMMIT;
 
+-- REFRESH MATERIALIZED VIEW CONCURRENTLY use of eval_const_expressions()
+SET SESSION AUTHORIZATION regress_sro_user;
+CREATE FUNCTION unwanted_grant_nofail(int) RETURNS int
+   IMMUTABLE LANGUAGE plpgsql AS $$
+BEGIN
+   PERFORM unwanted_grant();
+   RAISE WARNING 'owned';
+   RETURN 1;
+EXCEPTION WHEN OTHERS THEN
+   RETURN 2;
+END$$;
+CREATE MATERIALIZED VIEW sro_index_mv AS SELECT 1 AS c;
+CREATE UNIQUE INDEX ON sro_index_mv (c) WHERE unwanted_grant_nofail(1) > 0;
+\c -
+REFRESH MATERIALIZED VIEW CONCURRENTLY sro_index_mv;
+REFRESH MATERIALIZED VIEW sro_index_mv;
+
 DROP OWNED BY regress_sro_user;
 DROP ROLE regress_sro_user;