-
Notifications
You must be signed in to change notification settings - Fork 6
/
postgres.sql
136 lines (122 loc) · 4.55 KB
/
postgres.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
CREATE TABLE kv (
k text PRIMARY KEY,
v jsonb NOT NULL
);
CREATE TABLE accounts (
id text PRIMARY KEY,
lnurl_key text UNIQUE NOT NULL
);
CREATE TABLE contracts (
id text PRIMARY KEY, -- prefix to turn into the init invoice label
name text NOT NULL DEFAULT '',
readme text NOT NULL DEFAULT '',
code text NOT NULL,
state jsonb NOT NULL DEFAULT '{}',
created_at timestamp NOT NULL DEFAULT now(),
CONSTRAINT state_is_object CHECK (jsonb_typeof(state) = 'object'),
CONSTRAINT code_exists CHECK (code != '')
);
CREATE TABLE calls (
id text PRIMARY KEY,
time timestamp NOT NULL DEFAULT now(),
contract_id text NOT NULL REFERENCES contracts (id) ON DELETE CASCADE,
method text NOT NULL,
payload jsonb NOT NULL DEFAULT '{}',
msatoshi numeric(13) NOT NULL DEFAULT 0,
cost numeric(13) NOT NULL,
caller_account text REFERENCES accounts(id),
caller_contract text REFERENCES contracts(id),
diff text,
CONSTRAINT method_exists CHECK (method != ''),
CONSTRAINT cost_positive CHECK (method = '__init__' OR cost > 0),
CONSTRAINT caller_not_blank CHECK (caller != ''),
CONSTRAINT msatoshi_not_negative CHECK (msatoshi >= 0)
);
CREATE INDEX IF NOT EXISTS idx_calls_by_contract ON calls (contract_id, time);
CREATE TABLE internal_transfers (
call_id text NOT NULL REFERENCES calls (id),
time timestamp NOT NULL DEFAULT now(),
msatoshi numeric(13) NOT NULL,
from_contract text REFERENCES contracts(id),
from_account text REFERENCES accounts(id),
to_account text REFERENCES accounts(id),
to_contract text REFERENCES contracts(id),
to_burn bool,
CONSTRAINT one_receiver_or_burn CHECK (
(to_burn = false AND
to_contract IS NOT NULL AND to_contract != '' AND to_account IS NULL) OR
(to_burn = false AND
to_contract IS NULL AND to_account IS NOT NULL AND to_account != '') OR
(to_burn = true AND
to_contract IS NULL AND to_account IS NULL)
),
CONSTRAINT one_sender CHECK (
(from_contract IS NOT NULL AND from_contract != '' AND from_account IS NULL) OR
(from_contract IS NULL AND from_account IS NOT NULL AND from_account != '')
)
);
CREATE INDEX IF NOT EXISTS idx_internal_transfers_from_contract ON internal_transfers (from_contract);
CREATE INDEX IF NOT EXISTS idx_internal_transfers_to_contract ON internal_transfers (to_contract);
CREATE INDEX IF NOT EXISTS idx_internal_transfers_from_account ON internal_transfers (from_account);
CREATE INDEX IF NOT EXISTS idx_internal_transfers_to_account ON internal_transfers (to_account);
CREATE OR REPLACE FUNCTION funds(contract contracts) RETURNS numeric(13) AS $$
SELECT (
SELECT coalesce(sum(msatoshi), 0)
FROM calls WHERE calls.contract_id = contract.id
) - (
SELECT coalesce(sum(msatoshi), 0)
FROM internal_transfers WHERE from_contract = contract.id
) + (
SELECT coalesce(sum(msatoshi), 0)
FROM internal_transfers WHERE to_contract = contract.id
);
$$ LANGUAGE SQL;
CREATE TABLE withdrawals (
account_id text NOT NULL REFERENCES accounts(id),
time timestamp NOT NULL DEFAULT now(),
msatoshi numeric(13) NOT NULL,
fee_msat int NOT NULL DEFAULT 0,
fulfilled bool NOT NULL,
bolt11 text NOT NULL
);
CREATE OR REPLACE VIEW account_history (
time,
account_id,
msatoshi,
counterparty
) AS
SELECT * FROM (
SELECT time, from_account, -msatoshi, coalesce(to_contract, to_account, 'burned')
FROM internal_transfers
WHERE from_account IS NOT NULL
UNION ALL
SELECT time, to_account, msatoshi, coalesce(from_contract, from_account)
FROM internal_transfers
WHERE to_account IS NOT NULL
UNION ALL
SELECT time, account_id, -msatoshi-fee_msat, 'withdrawal'
FROM withdrawals
)u ORDER BY time DESC
;
CREATE OR REPLACE FUNCTION balance(id text) RETURNS numeric(13) AS $$
SELECT coalesce(sum(msatoshi), 0)::numeric(13)
FROM account_history
WHERE account_id = id;
$$ LANGUAGE SQL;
CREATE OR REPLACE FUNCTION transfers(call text, contract text) RETURNS jsonb AS $$
SELECT coalesce(jsonb_agg(transfers), '[]'::jsonb)
FROM
(
SELECT 'out' AS direction, msatoshi,
CASE WHEN to_account IS NOT NULL THEN to_account ELSE to_contract END AS counterparty
FROM internal_transfers
WHERE internal_transfers.call_id = call
AND internal_transfers.from_contract = contract
UNION ALL
SELECT 'in' AS direction, msatoshi,
CASE WHEN from_account IS NOT NULL THEN from_account ELSE from_contract END AS counterparty
FROM internal_transfers
WHERE internal_transfers.call_id = call
AND internal_transfers.to_contract = contract
) AS transfers
$$ LANGUAGE SQL;