Timeout on advisory locks in postgresql

前端 未结 1 1642
甜味超标
甜味超标 2021-02-13 12:54

I\'m migrating from ORACLE. Currently I\'m trying to port this call:

lkstat := DBMS_LOCK.REQUEST(lkhndl, DBMS_LOCK.X_MODE, lktimeout, true);

Th

1条回答
  •  暗喜
    暗喜 (楼主)
    2021-02-13 13:08

    This is a prototype of a wrapper that poorly emulates DBMS_LOCK.REQUEST - constrained to only one type of lock (transaction-scope advisory lock).

    To make function fully compatible with Oracle's, it would need several hundreds lines. But that's a start.

    CREATE OR REPLACE FUNCTION
    advisory_xact_lock_request(p_key bigint, p_timeout numeric)
    RETURNS integer
    LANGUAGE plpgsql AS $$
    /*  Imitate DBMS_LOCK.REQUEST for PostgreSQL advisory lock. 
    Return 0 on Success, 1 on Timeout, 3 on Parameter Error. */
    DECLARE
        t0 timestamptz := clock_timestamp();
    BEGIN
        IF p_timeout NOT BETWEEN 0 AND 86400 THEN
            RAISE WARNING 'Invalid timeout parameter';
            RETURN 3;
        END IF;
        LOOP
            IF pg_try_advisory_xact_lock(key) THEN
                RETURN 0;
            ELSIF clock_timestamp() > t0 + (p_timeout||' seconds')::interval THEN
                RAISE WARNING 'Could not acquire lock in % seconds', p_timeout;
                RETURN 1;
            ELSE
                PERFORM pg_sleep(0.01); /* 10 ms */
            END IF;
        END LOOP;
    END;
    $$;
    

    Test it using this code:

    SELECT CASE 
        WHEN advisory_xact_lock_request(1, 2.5) = 0
        THEN pg_sleep(120)
    END; -- and repeat this in parallel session 
    
    /* Usage in Pl/PgSQL */
    
    lkstat := advisory_xact_lock_request(lkhndl, lktimeout);
    

    0 讨论(0)
提交回复
热议问题