Migrating stored procedures and functions from SQL Server to PostgreSQL is a crucial step when transitioning from Microsoft’s proprietary database system to the open-source PostgreSQL platform. While both databases support procedural code, there are fundamental differences in syntax, features, and execution methodologies.
This article explores how to convert SQL Server stored procedures and functions into PostgreSQL equivalents, using coding examples to demonstrate key differences.
Understanding Key Differences
Before diving into the conversion process, it is essential to understand the fundamental differences between SQL Server’s Transact-SQL (T-SQL) and PostgreSQL’s PL/pgSQL:
- Stored Procedures vs. Functions:
- SQL Server distinguishes between stored procedures (
CREATE PROCEDURE
) and functions (CREATE FUNCTION
), where procedures can execute without returning a value, while functions must return a value. - PostgreSQL allows stored procedures (
CREATE PROCEDURE
) but relies more heavily on functions (CREATE FUNCTION
) withRETURNS void
when returning no value.
- SQL Server distinguishes between stored procedures (
- Variable Declaration:
- SQL Server uses
DECLARE @variableName DataType
. - PostgreSQL uses
variableName DataType := value;
inside functions andDECLARE variableName DataType;
inside procedural blocks.
- SQL Server uses
- Control Flow Statements:
- SQL Server uses
BEGIN...END
,IF...ELSE
, andWHILE
loops. - PostgreSQL uses
PL/pgSQL
blocks withBEGIN...END
,IF...THEN...ELSE
, andLOOP
constructs.
- SQL Server uses
- Exception Handling:
- SQL Server uses
TRY...CATCH
. - PostgreSQL uses
EXCEPTION WHEN ... THEN
.
- SQL Server uses
- Returning Values:
- SQL Server procedures use
RETURN
orOUTPUT
parameters. - PostgreSQL functions return values using
RETURNS
.
- SQL Server procedures use
Converting SQL Server Stored Procedures to PostgreSQL
Example 1: Simple Stored Procedure with Output Parameter
SQL Server Version
CREATE PROCEDURE GetEmployeeName
@EmployeeID INT,
@EmployeeName NVARCHAR(100) OUTPUT
AS
BEGIN
SELECT @EmployeeName = Name FROM Employees WHERE EmployeeID = @EmployeeID;
END;
PostgreSQL Equivalent
CREATE OR REPLACE FUNCTION GetEmployeeName(EmployeeID INT) RETURNS TEXT AS $$
DECLARE
EmployeeName TEXT;
BEGIN
SELECT name INTO EmployeeName FROM Employees WHERE EmployeeID = GetEmployeeName.EmployeeID;
RETURN EmployeeName;
END;
$$ LANGUAGE plpgsql;
Example 2: Procedure with Transaction Handling
SQL Server Version
CREATE PROCEDURE UpdateEmployeeSalary
@EmployeeID INT,
@NewSalary DECIMAL(10,2)
AS
BEGIN
BEGIN TRANSACTION;
UPDATE Employees SET Salary = @NewSalary WHERE EmployeeID = @EmployeeID;
COMMIT TRANSACTION;
END;
PostgreSQL Equivalent
CREATE OR REPLACE PROCEDURE UpdateEmployeeSalary(EmployeeID INT, NewSalary DECIMAL(10,2)) AS $$
BEGIN
UPDATE Employees SET Salary = NewSalary WHERE EmployeeID = UpdateEmployeeSalary.EmployeeID;
COMMIT;
END;
$$ LANGUAGE plpgsql;
Converting SQL Server Functions to PostgreSQL
Example 3: Scalar Function
SQL Server Version
CREATE FUNCTION GetTotalOrders(@CustomerID INT) RETURNS INT
AS
BEGIN
DECLARE @TotalOrders INT;
SELECT @TotalOrders = COUNT(*) FROM Orders WHERE CustomerID = @CustomerID;
RETURN @TotalOrders;
END;
PostgreSQL Equivalent
CREATE OR REPLACE FUNCTION GetTotalOrders(CustomerID INT) RETURNS INT AS $$
DECLARE
TotalOrders INT;
BEGIN
SELECT COUNT(*) INTO TotalOrders FROM Orders WHERE CustomerID = GetTotalOrders.CustomerID;
RETURN TotalOrders;
END;
$$ LANGUAGE plpgsql;
Example 4: Table-Valued Function
SQL Server Version
CREATE FUNCTION GetOrdersByCustomer(@CustomerID INT)
RETURNS TABLE
AS
RETURN (
SELECT OrderID, OrderDate, TotalAmount FROM Orders WHERE CustomerID = @CustomerID
);
PostgreSQL Equivalent
CREATE OR REPLACE FUNCTION GetOrdersByCustomer(CustomerID INT)
RETURNS TABLE(OrderID INT, OrderDate DATE, TotalAmount DECIMAL) AS $$
BEGIN
RETURN QUERY SELECT OrderID, OrderDate, TotalAmount FROM Orders WHERE CustomerID = GetOrdersByCustomer.CustomerID;
END;
$$ LANGUAGE plpgsql;
Handling Exception and Error Logging
SQL Server Version
BEGIN TRY
INSERT INTO Orders (CustomerID, OrderDate) VALUES (1, GETDATE());
END TRY
BEGIN CATCH
PRINT 'An error occurred';
END CATCH;
PostgreSQL Equivalent
BEGIN
INSERT INTO Orders (CustomerID, OrderDate) VALUES (1, CURRENT_DATE);
EXCEPTION WHEN OTHERS THEN
RAISE NOTICE 'An error occurred';
END;
Conclusion
Converting SQL Server stored procedures and functions to PostgreSQL requires careful attention to syntax differences, procedural structures, and error handling mechanisms. While SQL Server uses T-SQL with a procedural approach, PostgreSQL relies heavily on PL/pgSQL and supports both procedures and functions with flexible return types.
Key takeaways from this conversion process include:
- PostgreSQL does not support output parameters in stored procedures the same way SQL Server does; instead, functions returning values are preferred.
DECLARE
andINTO
are crucial for handling variables and query results in PostgreSQL.- PostgreSQL exception handling differs from SQL Server’s
TRY...CATCH
mechanism but provides similar robustness withEXCEPTION
blocks. - Transactions in PostgreSQL require explicit
COMMIT
andROLLBACK
when used within procedures.
By following these guidelines and understanding the differences, organizations can efficiently migrate SQL Server procedural code to PostgreSQL, ensuring seamless functionality and improved performance in an open-source environment.