This repository was archived by the owner on Oct 11, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 37
/
Copy path05_create_table.sql
77 lines (59 loc) · 3.41 KB
/
05_create_table.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
\c pg_features_demo
-- assume "application" role
-- it's a good practice to own all objects by one "application" role, so that changes could be done using the same role,
-- not requiring the samewhat dangerous "superuser".
SET ROLE TO demorole;
-- CREATE TABLEs for our super-simplified banking schema. For those more familiar with Postgres you may notice the schema is
-- very similar to the one used by default Postgres benchmarking tool "pgbench"
CREATE TABLE banking.branch(
branch_id int NOT NULL PRIMARY KEY, -- using just "id" for name here is not recommended, the more explicit the better for important stuff
balance numeric NOT NULL DEFAULT 0
);
CREATE TABLE banking.teller(
teller_id serial NOT NULL PRIMARY KEY, -- SERIAL / BIGSERIAL can be used to define auto-incrementing columns
-- that don't specifying on INSERT. NB! Gaps in numbering can occur!
branch_id int NOT NULL,
balance numeric NOT NULL DEFAULT 0
);
CREATE TABLE banking.account(
account_id int GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY, -- as of v10 it's better to use IDENTITY columns
-- instead of SERIAL
branch_id int NOT NULL,
teller_id int NOT NULL,
balance numeric NOT NULL DEFAULT 0
);
CREATE TABLE banking.transaction_history(
id SERIAL NOT NULL,
teller_id int NOT NULL,
branch_id int NOT NULL,
account_id int NOT NULL,
delta numeric NOT NULL,
created_on timestamp with time zone NOT NULL DEFAULT now()
);
-- Generally it's also a good practice to at least minimally comment the tables and columns for complex applications
COMMENT ON TABLE banking.transaction_history IS 'A simple banking table';
COMMENT ON COLUMN banking.transaction_history.delta IS 'Change in account balance for one transaction';
-- generate 1 branch, 10 tellers for branch, 10K accounts for each teller with random balances
INSERT INTO banking.branch (branch_id)
VALUES (1);
INSERT INTO banking.teller (teller_id, branch_id)
SELECT generate_series(1, 10), 1;
INSERT INTO banking.account (account_id, teller_id, branch_id)
SELECT i, i % 10 + 1, 1 FROM generate_series(1, 1e5) i;
-- Adding foreign keys and indexes
-- (more correct would be to add them before inserting data but also inserts would be slower then)
CREATE INDEX ON banking.account (teller_id);
CREATE INDEX ON banking.account (branch_id);
CREATE INDEX ON banking.transaction_history (account_id);
CREATE INDEX ON banking.transaction_history (teller_id);
CREATE INDEX ON banking.transaction_history (created_on);
ALTER TABLE banking.teller ADD FOREIGN KEY (branch_id) REFERENCES banking.branch;
ALTER TABLE banking.account ADD FOREIGN KEY (branch_id) REFERENCES banking.branch;
ALTER TABLE banking.account ADD FOREIGN KEY (teller_id) REFERENCES banking.teller;
ALTER TABLE banking.transaction_history ADD FOREIGN KEY (branch_id) REFERENCES banking.branch;
ALTER TABLE banking.transaction_history ADD FOREIGN KEY (teller_id) REFERENCES banking.teller;
ALTER TABLE banking.transaction_history ADD FOREIGN KEY (account_id) REFERENCES banking.account;
-- Also when adding/changing a lot of row that will be used immediately it is benefical to explicitly force gathering
-- of column statistics with ANALYZE
ANALYZE banking.teller;
ANALYZE banking.account;