-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpgledger.sql
188 lines (165 loc) · 7.82 KB
/
pgledger.sql
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
CREATE TABLE pgledger_accounts (
id BIGSERIAL PRIMARY KEY,
name TEXT NOT NULL,
currency TEXT NOT NULL,
balance NUMERIC NOT NULL DEFAULT 0,
version BIGINT NOT NULL DEFAULT 0,
allow_negative_balance BOOLEAN NOT NULL,
allow_positive_balance BOOLEAN NOT NULL,
created_at TIMESTAMPTZ NOT NULL,
updated_at TIMESTAMPTZ NOT NULL
);
CREATE TABLE pgledger_transfers (
id BIGSERIAL PRIMARY KEY,
from_account_id BIGINT NOT NULL REFERENCES pgledger_accounts(id),
to_account_id BIGINT NOT NULL REFERENCES pgledger_accounts(id),
amount NUMERIC NOT NULL,
created_at TIMESTAMPTZ NOT NULL,
CHECK (amount > 0 AND from_account_id != to_account_id)
);
CREATE INDEX ON pgledger_transfers(from_account_id);
CREATE INDEX ON pgledger_transfers(to_account_id);
CREATE TABLE pgledger_entries (
id BIGSERIAL PRIMARY KEY,
account_id BIGINT NOT NULL REFERENCES pgledger_accounts(id),
transfer_id BIGINT NOT NULL REFERENCES pgledger_transfers(id),
amount NUMERIC NOT NULL,
account_previous_balance NUMERIC NOT NULL,
account_current_balance NUMERIC NOT NULL,
account_version BIGINT NOT NULL,
created_at TIMESTAMPTZ NOT NULL
);
CREATE INDEX ON pgledger_entries(account_id);
CREATE INDEX ON pgledger_entries(transfer_id);
CREATE OR REPLACE FUNCTION pgledger_add_account(name_param TEXT, currency_param TEXT) RETURNS TABLE(id BIGINT, name TEXT, currency TEXT, balance NUMERIC) AS $$
BEGIN
RETURN QUERY
INSERT INTO pgledger_accounts (name, currency, allow_negative_balance, allow_positive_balance, created_at, updated_at)
VALUES (name_param, currency_param, TRUE, TRUE, now(), now())
RETURNING pgledger_accounts.id, pgledger_accounts.name, pgledger_accounts.currency, pgledger_accounts.balance;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION pgledger_create_account(
name_param TEXT,
currency_param TEXT,
allow_negative_balance_param BOOLEAN DEFAULT TRUE,
allow_positive_balance_param BOOLEAN DEFAULT TRUE
)
RETURNS TABLE(id BIGINT, name TEXT, currency TEXT, balance NUMERIC, version BIGINT, allow_negative_balance BOOLEAN, allow_positive_balance BOOLEAN, created_at TIMESTAMPTZ, updated_at TIMESTAMPTZ) AS $$
BEGIN
RETURN QUERY
INSERT INTO pgledger_accounts (name, currency, allow_negative_balance, allow_positive_balance, created_at, updated_at)
VALUES (name_param, currency_param, allow_negative_balance_param, allow_positive_balance_param, now(), now())
RETURNING pgledger_accounts.id, pgledger_accounts.name, pgledger_accounts.currency, pgledger_accounts.balance, pgledger_accounts.version,
pgledger_accounts.allow_negative_balance, pgledger_accounts.allow_positive_balance,
pgledger_accounts.created_at, pgledger_accounts.updated_at;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION pgledger_get_account(id_param BIGINT)
RETURNS TABLE(id BIGINT, name TEXT, currency TEXT, balance NUMERIC, version BIGINT, allow_negative_balance BOOLEAN, allow_positive_balance BOOLEAN, created_at TIMESTAMPTZ, updated_at TIMESTAMPTZ) AS $$
BEGIN
RETURN QUERY
SELECT pgledger_accounts.id, pgledger_accounts.name, pgledger_accounts.currency, pgledger_accounts.balance, pgledger_accounts.version,
pgledger_accounts.allow_negative_balance, pgledger_accounts.allow_positive_balance,
pgledger_accounts.created_at, pgledger_accounts.updated_at
FROM pgledger_accounts
WHERE pgledger_accounts.id = id_param;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION pgledger_get_transfer(id_param BIGINT) RETURNS TABLE(id BIGINT, from_account_id BIGINT, to_account_id BIGINT, amount NUMERIC, created_at TIMESTAMPTZ) AS $$
BEGIN
RETURN QUERY
SELECT
t.id,
t.from_account_id,
t.to_account_id,
t.amount,
t.created_at
FROM pgledger_transfers t
WHERE t.id = id_param;
END;
$$ LANGUAGE plpgsql;
-- Helper function to check account balance constraints
CREATE OR REPLACE FUNCTION pgledger_check_account_balance_constraints(account RECORD) RETURNS VOID AS $$
BEGIN
-- If account doesn't allow negative balance and balance is negative, raise an error
IF NOT account.allow_negative_balance AND (account.balance < 0) THEN
RAISE EXCEPTION 'Account (id=%, name=%) does not allow negative balance', account.id, account.name;
END IF;
-- If account doesn't allow positive balance and balance is positive, raise an error
IF NOT account.allow_positive_balance AND (account.balance > 0) THEN
RAISE EXCEPTION 'Account (id=%, name=%) does not allow positive balance', account.id, account.name;
END IF;
END;
$$ LANGUAGE plpgsql;
CREATE OR REPLACE FUNCTION pgledger_create_transfer(from_account_id_param BIGINT, to_account_id_param BIGINT, amount_param NUMERIC) RETURNS TABLE(id BIGINT, from_account_id BIGINT, to_account_id BIGINT, amount NUMERIC, created_at TIMESTAMPTZ) AS $$
DECLARE
transfer_id BIGINT;
account_ids INT[] := ARRAY[
LEAST(from_account_id_param, to_account_id_param),
GREATEST(from_account_id_param, to_account_id_param)
];
from_account RECORD;
to_account RECORD;
BEGIN
-- Preliminary checks
IF amount_param <= 0 THEN
RAISE EXCEPTION 'Amount (%) must be positive', amount_param;
END IF;
IF from_account_id_param = to_account_id_param THEN
RAISE EXCEPTION 'Cannot transfer to the same account (id=%)', from_account_id_param;
END IF;
-- Lock accounts in a consistent order to prevent deadlocks
-- Lock first account
PERFORM pgledger_accounts.id
FROM pgledger_accounts
WHERE pgledger_accounts.id = account_ids[1]
FOR UPDATE;
-- Lock second account
PERFORM pgledger_accounts.id
FROM pgledger_accounts
WHERE pgledger_accounts.id = account_ids[2]
FOR UPDATE;
-- Update account balances
UPDATE pgledger_accounts
SET balance = balance - amount_param,
version = version + 1,
updated_at = now()
WHERE pgledger_accounts.id = from_account_id_param
RETURNING * INTO from_account;
-- Check balance constraints for the source account
PERFORM pgledger_check_account_balance_constraints(from_account);
UPDATE pgledger_accounts
SET balance = balance + amount_param,
version = version + 1,
updated_at = now()
WHERE pgledger_accounts.id = to_account_id_param
RETURNING * INTO to_account;
-- Check balance constraints for the destination account
PERFORM pgledger_check_account_balance_constraints(to_account);
-- Check that currencies match
IF from_account.currency != to_account.currency THEN
RAISE EXCEPTION 'Cannot transfer between different currencies (% and %)', from_account.currency, to_account.currency;
END IF;
-- Create transfer record
INSERT INTO pgledger_transfers (from_account_id, to_account_id, amount, created_at)
VALUES (from_account_id_param, to_account_id_param, amount_param, now())
RETURNING pgledger_transfers.id INTO transfer_id;
-- Create entry for the source account (negative amount)
INSERT INTO pgledger_entries (account_id, transfer_id, amount, account_previous_balance, account_current_balance, account_version, created_at)
VALUES (from_account_id_param, transfer_id, -amount_param, from_account.balance + amount_param, from_account.balance, from_account.version, now());
-- Create entry for the destination account (positive amount)
INSERT INTO pgledger_entries (account_id, transfer_id, amount, account_previous_balance, account_current_balance, account_version, created_at)
VALUES (to_account_id_param, transfer_id, amount_param, to_account.balance - amount_param, to_account.balance, to_account.version, now());
-- Return the transfer details
RETURN QUERY
SELECT
t.id,
t.from_account_id,
t.to_account_id,
t.amount,
t.created_at
FROM pgledger_transfers t
WHERE t.id = transfer_id;
END;
$$ LANGUAGE plpgsql;