Stored Procedures & Exception Handling when migrating from Oracle to PostgreSQL or YugabyteDB

Franck Pachot - Jul 3 - - Dev Community

Lot of companies hope to find a solution to easily move their PL/SQL code from Oracle Database to PostgreSQL or PostgreSQL-compatible managed services such as Amazon Aurora or YugabyteDB. The AWS Schema Conversion Tool and YugabyteDB Voyager are helpful tools for this, but migrating the syntax is just one aspect of the process. Each database engine behaves differently, and failing to understand these differences can result in unexpected outcomes.

An example

I have two tables: gamers, each with a specified amount of cash, and bids. The business logic is deployed as a stored procedure. A business rule specifies that no bid can exceed $10, which is enforced by a check constraint. If this rule is violated, no error is raised, but the bid is considered a no-op, resulting in a $0 bid with a rejection comment. The constraint violation is catched by the stored procedure code.

The original Oracle PL/SQL code

Here is my code in Oracle:

create table gamers (
    name varchar2(50) primary key,
    cash number(10, 2)
);

create table bids (
    name varchar2(50) references gamers,
    bid_amount number(5, 2),
    bid_time timestamp default current_timestamp,
    operation_text varchar2(255),
    constraint bid_amount_check check (bid_amount <= 10)
);

create or replace procedure insert_bid 
(p_name in varchar2, p_bid_amount in number) is
    bid_too_high exception;
    pragma exception_init(bid_too_high, -2290); -- check constraint violated
    l_cash number(10, 2);
begin
    -- Deduct the cash amount
    update gamers
     set cash = cash - p_bid_amount
     where name = p_name;
    -- Insert the bid
    insert into bids (name, bid_amount, operation_text)
     values (p_name, p_bid_amount, 'new bid');
exception
    when bid_too_high then
        -- Reverse the cash deduction
        update gamers
        set cash = cash + p_bid_amount
        where name = p_name;
        -- Insert the no-op bid
        insert into bids (name, bid_amount, operation_text) values 
         (p_name, 0, 'attempted bid of ' || p_bid_amount || ' rejected');
  end;
/

Enter fullscreen mode Exit fullscreen mode

Let's run it. I have a gamer with $42 and making 3 bids:

  • Bid $20, which exceeds the maximum possible bid in the table.
  • Bid $2, which should be successful.

The final cash amount should be $40.

SQL> column name format a8
SQL> column bid_time format a30
SQL> column operation_text format a30
SQL> set linesize 100
SQL>
SQL> insert into gamers values ('G4mR', 42);

1 row created.

SQL> commit;

Commit complete.

SQL>
SQL> execute insert_bid('G4mR', 20);

PL/SQL procedure successfully completed.

SQL> commit;

Commit complete.

SQL> execute insert_bid('G4mR', 2);

PL/SQL procedure successfully completed.

SQL> commit;

Commit complete.

SQL> select * from gamers;

NAME           CASH
-------- ----------
G4mR             40

SQL> select * from bids;

NAME     BID_AMOUNT BID_TIME                       OPERATION_TEXT
-------- ---------- ------------------------------ ------------------------------
G4mR              0 03-JUL-24 08.09.29.742321 AM   attempted bid of 20 rejected
G4mR              2 03-JUL-24 08.09.31.626613 AM   new bid

Enter fullscreen mode Exit fullscreen mode

That's the correct result. Let's try to convert that to PostgreSQL.

Using AWS SCT to transform it to PostgreSQL

I've used AWS Schema Conversion Tool to get the equivalent PostgreSQL syntax:

Image description

SCT used the proprietary aws_oracle_ext.ora_exception which I simplified to be PostgreSQL compatible by simply checking check_violation:

CREATE OR REPLACE PROCEDURE insert_bid
 (IN p_name TEXT, IN p_bid_amount DOUBLE PRECISION)
AS 
$BODY$
DECLARE
    l_cash NUMERIC(10, 2);
BEGIN
    /* Deduct the cash amount */
    UPDATE gamers
    SET cash = cash - p_bid_amount
        WHERE name = p_name;
    /* Insert the bid */
    INSERT INTO bids (name, bid_amount, operation_text)
    VALUES (p_name, p_bid_amount, 'new bid');
    EXCEPTION
        WHEN check_violation THEN
                    UPDATE gamers
                    SET cash = cash + p_bid_amount
                        WHERE name = p_name;
                    /* Insert the no-op bid */
                    INSERT INTO bids (name, bid_amount, operation_text)
                    VALUES (p_name, 0, CONCAT_WS('', 'attempted bid of ', p_bid_amount, ' rejected'));
END;
$BODY$
LANGUAGE plpgsql;

Enter fullscreen mode Exit fullscreen mode

If you know how PostgreSQL works you already spot the problem.

Wrong result with the same logic in PostgreSQL

Let's test it:

yugabyte=# insert into gamers values ('G4mR', 42);
INSERT 0 1

yugabyte=# begin; call insert_bid('G4mR', 20); commit;
BEGIN
CALL
COMMIT

yugabyte=# begin; call insert_bid('G4mR', 2); commit;
BEGIN
CALL
COMMIT

yugabyte=# select * from gamers;
 name | cash
------+-------
 G4mR | 60.00
(1 row)

yugabyte=# select * from bids;
 name | bid_amount |          bid_time          |        operation_text
------+------------+----------------------------+------------------------------
 G4mR |       2.00 | 2024-07-03 12:43:08.643212 | new bid
 G4mR |       0.00 | 2024-07-03 12:43:07.522352 | attempted bid of 20 rejected
(2 rows)
Enter fullscreen mode Exit fullscreen mode

The logic that was written for Oracle doesn't work correctly in PostgreSQL. Instead of decreasing to $40, the cash increases from $42 to $60.

This is because, in PostgreSQL and YugabyteDB, the PL/pgSQL block is atomic and doesn't need compensation in the exception block like in Oracle.

To prevent partial changes when entering the exception block, the main block rolls back before entering the exception block. Unlike in Oracle, there's no need to clean up or compensate for partial changes in the exception block.

The right logic for PostgreSQL and YugabyteDB

Here is the correct code, much simpler. It implicitly rolls back the update of the cash in the gamers' table when entering the exception so that the exception only has to insert the no-op bid.

CREATE OR REPLACE PROCEDURE insert_bid
 (IN p_name TEXT, IN p_bid_amount DOUBLE PRECISION)
AS 
$BODY$
DECLARE
    l_cash NUMERIC(10, 2);
BEGIN
    /* Deduct the cash amount */
    UPDATE gamers
    SET cash = cash - p_bid_amount
        WHERE name = p_name;
    /* Insert the bid */
    INSERT INTO bids (name, bid_amount, operation_text)
    VALUES (p_name, p_bid_amount, 'new bid');
    EXCEPTION
        WHEN check_violation THEN
                    /* Insert the no-op bid */
                    INSERT INTO bids (name, bid_amount, operation_text)
                    VALUES (p_name, 0, CONCAT_WS('', 'attempted bid of ', p_bid_amount, ' rejected'));
END;
$BODY$
LANGUAGE plpgsql;
Enter fullscreen mode Exit fullscreen mode

Without any attempt to compensate for changes, because that is done by the database, it provides the right result:

yugabyte=# insert into gamers values ('G4mR', 42);
INSERT 0 1
yugabyte=# begin; call insert_bid('G4mR', 20); commit;
BEGIN
CALL
COMMIT
yugabyte=# begin; call insert_bid('G4mR', 2); commit;
BEGIN
CALL
COMMIT
yugabyte=# select * from gamers;
 name | cash
------+-------
 G4mR | 40.00
(1 row)

yugabyte=# select * from bids;
 name | bid_amount |          bid_time          |        operation_text
------+------------+----------------------------+------------------------------
 G4mR |       2.00 | 2024-07-03 12:53:50.889344 | new bid
 G4mR |       0.00 | 2024-07-03 12:53:49.787083 | attempted bid of 20 rejected
(2 rows)

Enter fullscreen mode Exit fullscreen mode

PostgreSQL implicitly creates a savepoint at the beginning of execution. It's important to note that savepoints can be expensive in PostgreSQL, even though it has been optimized for PG17. On the other hand, YugabyteDB behaves similarly to PostgreSQL but with a scalable implementation that avoids the issues related to savepoints.

This example demonstrates that the most crucial aspect of migration is not just switching between languages but also verifying the behavior and outcome. To avoid being locked in with a particular vendor, it is important to have a good understanding of how both databases work and good regression tests that encompass all your business logic.

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .