From: | Adrian Klaver <adrian(dot)klaver(at)aklaver(dot)com> |
---|---|
To: | Dominique Devienne <ddevienne(at)gmail(dot)com> |
Cc: | pgsql-general(at)lists(dot)postgresql(dot)org |
Subject: | Re: SET LOCAL ROLE inside SECURITY INVOKER (LANGUAGE plpgsql) function |
Date: | 2025-07-30 19:42:39 |
Message-ID: | 662792ed-810d-46f1-a0c3-d4b55e5469fc@aklaver.com |
Views: | Whole Thread | Raw Message | Download mbox | Resend email |
Thread: | |
Lists: | pgsql-general |
On 7/30/25 09:21, Adrian Klaver wrote:
> On 7/30/25 08:47, Dominique Devienne wrote:
>> On Wed, Jul 30, 2025 at 5:23 PM Adrian Klaver
>> <adrian(dot)klaver(at)aklaver(dot)com> wrote:
>>> On 7/30/25 04:37, Dominique Devienne wrote:
>>>> Are there special consideration I'm unaware of, regarding SET ROLE
>>>> inside routines?
>>
>>> What is the ROLE that defined the function?
>>
>> A 3rd role. But does it matter? Given that this is in SECURITY INVOKER
>> function?
>
> My mistake, a BC(Before Coffee) issue.
>
>
>> The function and the table belong to yet another role.
>> And when we enter the function, we're yet another one (obviously with
>> USAGE+EXECUTE, since could call it).
>> But once we SET LOCAL ROLE, the effective permissions used should be
>> for :OWNER1 and the inherited :SOWNER.
>
> Could this be a search_path and/or naming issue, where the table
> SchemaMapping appears in more then one schema or different name case?
>
If the above is not the issue, then a simple test case:
grant db_user to app_user with set true, inherit true;
-- As db_user
create table fnc_set_role_test(id integer, fld1 varchar);
insert into fnc_set_role_test values (1, 'test');
CREATE OR REPLACE FUNCTION public.role_set()
RETURNS void
LANGUAGE plpgsql
AS $function$
BEGIN
EXECUTE format('SET LOCAL ROLE %I', 'db_user');
raise notice 'CURRENT_USER = %, can DELETE = %', CURRENT_USER,
has_table_privilege('fnc_set_role_test', 'DELETE');
DELETE FROM fnc_set_role_test;
END;
$function$
;
-- As app_user
\c - app_user
select * from fnc_set_role_test ;
select role_set();
NOTICE: CURRENT_USER = db_user, can DELETE = t
role_set
----------
(1 row)
select * from fnc_set_role_test ;
id | fld1
----+------
(0 rows)
My suspicion is that there is a missing piece in your chain of roles.
--
Adrian Klaver
adrian(dot)klaver(at)aklaver(dot)com
From | Date | Subject | |
---|---|---|---|
Next Message | Dominique Devienne | 2025-07-31 08:41:18 | Re: SET LOCAL ROLE inside SECURITY INVOKER (LANGUAGE plpgsql) function |
Previous Message | Adrian Klaver | 2025-07-30 16:21:08 | Re: SET LOCAL ROLE inside SECURITY INVOKER (LANGUAGE plpgsql) function |