-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCode_Appendix.txt
More file actions
1595 lines (1402 loc) · 67.6 KB
/
Code_Appendix.txt
File metadata and controls
1595 lines (1402 loc) · 67.6 KB
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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
-- ============================================================
-- CORE BANKING SYSTEM (CBS) --- DATABASE SCHEMA & DCL
-- DEVELOPED BY: HUZAIFA IMRAN & MUHAMMAD ARSLAN
-- Phase I & II: Table Definitions, Constraints, and Permissions
-- ============================================================
-- ------ DROP in reverse dependency order ------
DROP TABLE IF EXISTS interest_accruals CASCADE;
DROP TABLE IF EXISTS collateral CASCADE;
DROP TABLE IF EXISTS audit_log CASCADE;
DROP TABLE IF EXISTS loan_payments CASCADE;
DROP TABLE IF EXISTS loans CASCADE;
DROP TABLE IF EXISTS transactions CASCADE;
DROP TABLE IF EXISTS accounts CASCADE;
DROP TABLE IF EXISTS cards CASCADE;
DROP TABLE IF EXISTS employees CASCADE;
DROP TABLE IF EXISTS branches CASCADE;
DROP TABLE IF EXISTS customers CASCADE;
DROP TABLE IF EXISTS account_types CASCADE;
DROP TABLE IF EXISTS loan_types CASCADE;
-- ============================================================
-- 1. LOOKUP / REFERENCE TABLES
-- ============================================================
CREATE TABLE account_types (
account_type_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
type_name VARCHAR(30) NOT NULL UNIQUE,
interest_rate NUMERIC(5,2) NOT NULL CHECK (interest_rate >= 0),
min_balance NUMERIC(12,2) DEFAULT 0 NOT NULL CHECK (min_balance >= 0),
description VARCHAR(200)
);
CREATE TABLE loan_types (
loan_type_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
type_name VARCHAR(40) NOT NULL UNIQUE,
base_rate NUMERIC(5,2) NOT NULL CHECK (base_rate > 0),
max_amount NUMERIC(14,2) NOT NULL,
max_tenure_months INTEGER NOT NULL CHECK (max_tenure_months > 0),
description VARCHAR(200)
);
-- ============================================================
-- 2. CORE ENTITY TABLES
-- ============================================================
CREATE TABLE branches (
branch_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
branch_code VARCHAR(10) NOT NULL UNIQUE,
branch_name VARCHAR(100) NOT NULL,
city VARCHAR(60) NOT NULL,
address VARCHAR(200) NOT NULL,
phone VARCHAR(20),
ifsc_code VARCHAR(15) NOT NULL UNIQUE,
established_date DATE DEFAULT CURRENT_DATE NOT NULL,
is_active CHAR(1) DEFAULT 'Y' NOT NULL CHECK (is_active IN ('Y','N'))
);
CREATE TABLE customers (
customer_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
cnic VARCHAR(15) NOT NULL UNIQUE,
first_name VARCHAR(60) NOT NULL,
last_name VARCHAR(60) NOT NULL,
date_of_birth DATE NOT NULL,
gender CHAR(1) NOT NULL CHECK (gender IN ('M','F','O')),
email VARCHAR(120) UNIQUE,
phone VARCHAR(20) NOT NULL,
address VARCHAR(300) NOT NULL,
city VARCHAR(60) NOT NULL,
credit_score INTEGER DEFAULT 650 CHECK (credit_score BETWEEN 300 AND 850),
kyc_status VARCHAR(20) DEFAULT 'PENDING' CHECK (kyc_status IN ('PENDING','VERIFIED','REJECTED')),
registration_date DATE DEFAULT CURRENT_DATE NOT NULL,
is_active CHAR(1) DEFAULT 'Y' NOT NULL CHECK (is_active IN ('Y','N'))
);
CREATE TABLE employees (
employee_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
branch_id INTEGER NOT NULL REFERENCES branches(branch_id),
first_name VARCHAR(60) NOT NULL,
last_name VARCHAR(60) NOT NULL,
cnic VARCHAR(15) NOT NULL UNIQUE,
designation VARCHAR(80) NOT NULL,
department VARCHAR(60) NOT NULL,
salary NUMERIC(12,2) NOT NULL CHECK (salary > 0),
email VARCHAR(120) NOT NULL UNIQUE,
hire_date DATE DEFAULT CURRENT_DATE NOT NULL,
manager_id INTEGER REFERENCES employees(employee_id),
is_active CHAR(1) DEFAULT 'Y' NOT NULL CHECK (is_active IN ('Y','N'))
);
-- ============================================================
-- 3. ACCOUNTS
-- ============================================================
CREATE TABLE accounts (
account_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
account_number VARCHAR(20) NOT NULL UNIQUE,
customer_id INTEGER NOT NULL REFERENCES customers(customer_id),
branch_id INTEGER NOT NULL REFERENCES branches(branch_id),
account_type_id INTEGER NOT NULL REFERENCES account_types(account_type_id),
balance NUMERIC(14,2) DEFAULT 0 NOT NULL,
opened_date DATE DEFAULT CURRENT_DATE NOT NULL,
status VARCHAR(15) DEFAULT 'ACTIVE' CHECK (status IN ('ACTIVE','DORMANT','CLOSED','FROZEN')),
currency CHAR(3) DEFAULT 'PKR' NOT NULL,
last_txn_date DATE
);
CREATE TABLE cards (
card_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
account_id INTEGER NOT NULL REFERENCES accounts(account_id),
card_number VARCHAR(19) NOT NULL UNIQUE,
card_type VARCHAR(15) NOT NULL CHECK (card_type IN ('DEBIT','CREDIT','PREPAID')),
expiry_date DATE NOT NULL,
cvv_hash VARCHAR(64) NOT NULL,
daily_limit NUMERIC(10,2) DEFAULT 50000 NOT NULL,
is_active CHAR(1) DEFAULT 'Y' NOT NULL CHECK (is_active IN ('Y','N')),
issued_date DATE DEFAULT CURRENT_DATE NOT NULL
);
-- ============================================================
-- 4. TRANSACTIONS
-- ============================================================
CREATE TABLE transactions (
transaction_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
account_id INTEGER NOT NULL REFERENCES accounts(account_id),
txn_type VARCHAR(20) NOT NULL CHECK (txn_type IN ('DEPOSIT','WITHDRAWAL','TRANSFER_IN','TRANSFER_OUT','FEE','INTEREST')),
amount NUMERIC(14,2) NOT NULL CHECK (amount > 0),
balance_after NUMERIC(14,2) NOT NULL,
reference_no VARCHAR(30) NOT NULL UNIQUE,
description VARCHAR(300),
channel VARCHAR(20) DEFAULT 'BRANCH' CHECK (channel IN ('BRANCH','ATM','MOBILE','INTERNET','POS')),
txn_date DATE DEFAULT CURRENT_DATE NOT NULL,
performed_by INTEGER REFERENCES employees(employee_id),
status VARCHAR(15) DEFAULT 'SUCCESS' CHECK (status IN ('SUCCESS','FAILED','REVERSED','PENDING'))
);
-- ============================================================
-- 5. LOANS
-- ============================================================
CREATE TABLE loans (
loan_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
customer_id INTEGER NOT NULL REFERENCES customers(customer_id),
loan_type_id INTEGER NOT NULL REFERENCES loan_types(loan_type_id),
branch_id INTEGER NOT NULL REFERENCES branches(branch_id),
approved_by INTEGER REFERENCES employees(employee_id),
principal_amount NUMERIC(14,2) NOT NULL CHECK (principal_amount > 0),
interest_rate NUMERIC(5,2) NOT NULL CHECK (interest_rate > 0),
tenure_months INTEGER NOT NULL CHECK (tenure_months > 0),
emi_amount NUMERIC(12,2) NOT NULL,
disbursement_date DATE,
maturity_date DATE,
outstanding NUMERIC(14,2),
status VARCHAR(15) DEFAULT 'PENDING' CHECK (status IN ('PENDING','APPROVED','ACTIVE','CLOSED','DEFAULTED','REJECTED')),
application_date DATE DEFAULT CURRENT_DATE NOT NULL
);
CREATE TABLE loan_payments (
payment_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
loan_id INTEGER NOT NULL REFERENCES loans(loan_id),
payment_date DATE DEFAULT CURRENT_DATE NOT NULL,
amount_paid NUMERIC(12,2) NOT NULL CHECK (amount_paid > 0),
principal_component NUMERIC(12,2) NOT NULL,
interest_component NUMERIC(12,2) NOT NULL,
balance_remaining NUMERIC(14,2) NOT NULL,
payment_mode VARCHAR(20) DEFAULT 'ACCOUNT_DEBIT' CHECK (payment_mode IN ('ACCOUNT_DEBIT','CASH','CHEQUE','ONLINE')),
status VARCHAR(15) DEFAULT 'SUCCESS' CHECK (status IN ('SUCCESS','FAILED','BOUNCED'))
);
CREATE TABLE collateral (
collateral_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
loan_id INTEGER NOT NULL REFERENCES loans(loan_id),
asset_type VARCHAR(30) NOT NULL, -- e.g. PROPERTY, VEHICLE, GOLD
estimated_value NUMERIC(14,2) NOT NULL,
valuation_date DATE DEFAULT CURRENT_DATE,
description VARCHAR(500),
status VARCHAR(20) DEFAULT 'HELD' CHECK (status IN ('HELD','RELEASED','LIQUIDATED'))
);
CREATE TABLE interest_accruals (
accrual_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
account_id INTEGER NOT NULL REFERENCES accounts(account_id),
accrual_date DATE DEFAULT CURRENT_DATE NOT NULL,
interest_amount NUMERIC(12,2) NOT NULL,
rate_applied NUMERIC(5,2) NOT NULL,
remarks VARCHAR(100)
);
-- ============================================================
-- 6. AUDIT LOG
-- ============================================================
CREATE TABLE audit_log (
log_id INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
table_name VARCHAR(50) NOT NULL,
operation VARCHAR(10) NOT NULL CHECK (operation IN ('INSERT','UPDATE','DELETE')),
record_id INTEGER NOT NULL,
changed_by VARCHAR(80) DEFAULT CURRENT_USER NOT NULL,
change_timestamp TIMESTAMPTZ DEFAULT NOW() NOT NULL,
old_value TEXT,
new_value TEXT,
remarks VARCHAR(500)
);
-- ============================================================
-- 7. SEQUENCES
-- ============================================================
-- Required by the account-number auto-generation trigger in 04_plsql_module.sql
CREATE SEQUENCE IF NOT EXISTS acc_seq START 1;
-- ============================================================
-- 8. INDEXES
-- ============================================================
CREATE INDEX idx_accounts_customer ON accounts(customer_id);
CREATE INDEX idx_accounts_branch ON accounts(branch_id);
CREATE INDEX idx_txn_account_date ON transactions(account_id, txn_date DESC);
CREATE INDEX idx_txn_date ON transactions(txn_date DESC);
CREATE INDEX idx_loans_customer ON loans(customer_id);
CREATE INDEX idx_loans_status ON loans(status);
CREATE INDEX idx_customers_cnic ON customers(cnic);
CREATE INDEX idx_customers_city ON customers(city);
CREATE INDEX idx_lp_loan ON loan_payments(loan_id);
CREATE INDEX idx_audit_table_op ON audit_log(table_name, operation);
-- ============================================================
-- 9. VIEWS
-- ============================================================
-- Customer portfolio
CREATE OR REPLACE VIEW vw_customer_portfolio AS
SELECT c.customer_id,
c.first_name || ' ' || c.last_name AS customer_name,
c.cnic, c.credit_score, c.kyc_status,
a.account_number, at.type_name AS account_type,
a.balance, a.status AS account_status, a.currency,
b.branch_name, b.city AS branch_city
FROM customers c
JOIN accounts a ON a.customer_id = c.customer_id
JOIN account_types at ON at.account_type_id = a.account_type_id
JOIN branches b ON b.branch_id = a.branch_id;
-- Active loan summary
CREATE OR REPLACE VIEW vw_active_loans AS
SELECT l.loan_id,
c.first_name || ' ' || c.last_name AS customer_name,
c.cnic, lt.type_name AS loan_type,
l.principal_amount, l.outstanding, l.interest_rate,
l.tenure_months, l.emi_amount, l.maturity_date, l.status,
b.branch_name,
e.first_name || ' ' || e.last_name AS approved_by
FROM loans l
JOIN customers c ON c.customer_id = l.customer_id
JOIN loan_types lt ON lt.loan_type_id = l.loan_type_id
JOIN branches b ON b.branch_id = l.branch_id
LEFT JOIN employees e ON e.employee_id = l.approved_by
WHERE l.status IN ('ACTIVE','APPROVED');
-- Monthly transaction summary
CREATE OR REPLACE VIEW vw_monthly_txn_summary AS
SELECT a.account_number,
c.first_name || ' ' || c.last_name AS customer_name,
EXTRACT(YEAR FROM t.txn_date)::INT AS txn_year,
EXTRACT(MONTH FROM t.txn_date)::INT AS txn_month,
t.txn_type,
COUNT(*) AS txn_count,
SUM(t.amount) AS total_amount,
AVG(t.amount) AS avg_amount
FROM transactions t
JOIN accounts a ON a.account_id = t.account_id
JOIN customers c ON c.customer_id = a.customer_id
GROUP BY a.account_number, c.first_name, c.last_name,
EXTRACT(YEAR FROM t.txn_date), EXTRACT(MONTH FROM t.txn_date), t.txn_type;
-- Branch performance
CREATE OR REPLACE VIEW vw_branch_performance AS
SELECT b.branch_id, b.branch_name, b.city,
COUNT(DISTINCT a.account_id) AS total_accounts,
SUM(a.balance) AS total_deposits,
COUNT(DISTINCT l.loan_id) AS total_loans,
SUM(l.principal_amount) AS total_loan_amount,
COUNT(DISTINCT e.employee_id) AS staff_count
FROM branches b
LEFT JOIN accounts a ON a.branch_id = b.branch_id AND a.status = 'ACTIVE'
LEFT JOIN loans l ON l.branch_id = b.branch_id AND l.status IN ('ACTIVE','APPROVED')
LEFT JOIN employees e ON e.branch_id = b.branch_id AND e.is_active = 'Y'
GROUP BY b.branch_id, b.branch_name, b.city;
-- ============================================================
-- 10. DCL --- ROLES & PERMISSIONS (PostgreSQL style)
-- Run these as superuser / owner of the database.
-- ============================================================
-- Create roles (idempotent pattern)
DO $$
BEGIN
IF NOT EXISTS (SELECT FROM pg_roles WHERE rolname = 'cbs_app') THEN CREATE ROLE cbs_app LOGIN PASSWORD 'App@2024#'; END IF;
IF NOT EXISTS (SELECT FROM pg_roles WHERE rolname = 'cbs_report') THEN CREATE ROLE cbs_report LOGIN PASSWORD 'Rep@2024#'; END IF;
IF NOT EXISTS (SELECT FROM pg_roles WHERE rolname = 'cbs_teller') THEN CREATE ROLE cbs_teller LOGIN PASSWORD 'Tel@2024#'; END IF;
IF NOT EXISTS (SELECT FROM pg_roles WHERE rolname = 'cbs_audit') THEN CREATE ROLE cbs_audit LOGIN PASSWORD 'Aud@2024#'; END IF;
END $$;
-- Application user
GRANT SELECT, INSERT, UPDATE ON customers TO cbs_app;
GRANT SELECT, INSERT, UPDATE ON accounts TO cbs_app;
GRANT SELECT, INSERT ON transactions TO cbs_app;
GRANT SELECT, INSERT, UPDATE ON loans TO cbs_app;
GRANT SELECT, INSERT ON loan_payments TO cbs_app;
GRANT SELECT, INSERT ON audit_log TO cbs_app;
GRANT SELECT ON account_types TO cbs_app;
GRANT SELECT ON loan_types TO cbs_app;
GRANT SELECT ON branches TO cbs_app;
-- Grant usage on all identity sequences so cbs_app can INSERT
GRANT USAGE ON ALL SEQUENCES IN SCHEMA public TO cbs_app;
-- Teller
GRANT SELECT, INSERT ON transactions TO cbs_teller;
GRANT SELECT, UPDATE ON accounts TO cbs_teller;
GRANT SELECT ON customers TO cbs_teller;
-- Report user
GRANT SELECT ON vw_customer_portfolio TO cbs_report;
GRANT SELECT ON vw_active_loans TO cbs_report;
GRANT SELECT ON vw_monthly_txn_summary TO cbs_report;
GRANT SELECT ON vw_branch_performance TO cbs_report;
-- Audit user
GRANT SELECT ON audit_log TO cbs_audit;
GRANT SELECT ON transactions TO cbs_audit;
-- Revoke from PUBLIC (PostgreSQL grants CONNECT to PUBLIC by default; tighten here)
REVOKE ALL ON customers FROM PUBLIC;
REVOKE ALL ON accounts FROM PUBLIC;
REVOKE ALL ON transactions FROM PUBLIC;
REVOKE ALL ON loans FROM PUBLIC;
-- ============================================================
-- CORE BANKING SYSTEM (CBS) --- DATA POPULATION
-- DEVELOPED BY: HUZAIFA IMRAN & MUHAMMAD ARSLAN
-- Phase II: Mock Data for Testing and Verification
-- ============================================================
-- CLEANUP OLD DATA (Ensures fresh IDs every time)
TRUNCATE TABLE audit_log, transactions, loan_payments, loans, cards, accounts, employees, customers, loan_types, account_types, branches, collateral, interest_accruals RESTART IDENTITY CASCADE;
-- ------ 1. ACCOUNT TYPES ---------------------------------------------------------------------------------------------------------------------------
INSERT INTO account_types (type_name, interest_rate, min_balance, description) VALUES
('Basic Savings', 4.00, 1000, 'Standard savings account'),
('Premium Savings', 8.00, 5000, 'High-yield savings for premium clients'),
('Current Account', 0.00, 0, 'Standard checking account'),
('Fixed Deposit 1Y', 11.50, 10000, '1-year fixed deposit account'),
('Salary Account', 5.00, 0, 'Zero-balance salary account'),
('Student Account', 4.00, 0, 'Zero-balance student account'),
('Senior Citizen Savings', 9.00, 2000, 'Enhanced profit for 60+ customers'),
('Foreign Currency USD', 2.50, 500, 'USD-denominated savings account'),
('Digital Account', 7.00, 0, 'App-only paperless account');
-- ------ 2. LOAN TYPES ---------------------------------------------------------------------------------------------------------------------------
INSERT INTO loan_types (type_name, base_rate, max_amount, max_tenure_months, description) VALUES
('Personal Loan', 18.00, 2000000, 60, 'Unsecured personal finance'),
('Home Loan', 14.00, 50000000, 240, 'Secured housing finance'),
('Auto Loan', 16.00, 5000000, 72, 'Vehicle purchase financing'),
('Business Loan', 20.00, 20000000, 120, 'SME / corporate lending'),
('Education Loan', 10.00, 3000000, 84, 'Higher education financing'),
('Agricultural Loan', 9.00, 5000000, 36, 'Crop / farming input loans'),
('Gold Loan', 15.00, 1000000, 24, 'Loan against gold collateral');
-- ------ 3. BRANCHES ---------------------------------------------------------------------------------------------------------------------------------
INSERT INTO branches (branch_code, branch_name, city, address, phone, ifsc_code, established_date) VALUES
('KHI001','Karachi Main Branch', 'Karachi', 'I.I. Chundrigar Road', '+92-21-35680001','CBS0KHI001','2005-03-15'),
('KHI002','DHA Branch', 'Karachi', 'Phase VI, Bukhari', '+92-21-35362002','CBS0KHI002','2008-06-01'),
('LHR001','Lahore Main Branch', 'Lahore', 'Mall Road', '+92-42-36010004','CBS0LHR001','2005-07-10'),
('ISB001','Islamabad Blue Area', 'Islamabad', 'Blue Area', '+92-51-28310006','CBS0ISB001','2006-11-30'),
('MUL001','Multan Hussain Agahi', 'Multan', 'Hussain Agahi Road', '+92-61-45220010','CBS0MUL001','2013-05-22');
-- ------ 4. CUSTOMERS ------------------------------------------------------------------------------------------------------------------------------
-- IDs 1 to 10 will have Accounts.
-- IDs 11 to 15 will be EMPTY for your Demo.
INSERT INTO customers (cnic,first_name,last_name,date_of_birth,gender,email,phone,address,city,credit_score,kyc_status) VALUES
('3520212345671','Ahmed', 'Khan', '1985-04-12','M','ahmed@email.com', '+92-300-1234567','Gulshan','Karachi',720,'VERIFIED'),
('3520212345672','Fatima', 'Ali', '1990-08-25','F','fatima@email.com', '+92-321-2345678','Defence','Karachi',685,'VERIFIED'),
('3520212345673','Bilal', 'Sheikh', '1978-11-03','M','bilal@email.com', '+92-333-3456789','Nazimabad','Karachi',760,'VERIFIED'),
('3520212345674','Ayesha', 'Siddiqui', '1995-02-17','F','ayesha@email.com', '+92-345-4567890','Johar Town','Lahore',640,'VERIFIED'),
('3520212345675','Usman', 'Malik', '1982-06-30','M','usman@email.com', '+92-311-5678901','F-7/2','Islamabad',800,'VERIFIED'),
('3520212345676','Sana', 'Qureshi', '1993-09-14','F','sana@email.com', '+92-300-6789012','Model Town','Lahore',700,'VERIFIED'),
('3520212345677','Tariq', 'Hussain', '1970-01-22','M','tariq@email.com', '+92-321-7890123','Cantt Area','Peshawar',715,'VERIFIED'),
('3520212345678','Zara', 'Baig', '1998-12-05','F','zara@email.com', '+92-333-8901234','UET','Lahore',580,'VERIFIED'),
('3520212345679','Kamran', 'Iqbal', '1975-07-19','M','kamran@email.com', '+92-345-9012345','Satellite','Quetta',690,'VERIFIED'),
('3520212345680','Nadia', 'Farooq', '1988-03-28','F','nadia@email.com', '+92-311-0123456','Bahria','Islamabad',730,'VERIFIED'),
-- TEST USERS (NO ACCOUNTS - DELETABLE)
('3520212345684','Faisal', 'Nawaz', '1990-11-22','M','faisal@email.com', '+92-300-1122334','DHA 6','Karachi',700,'VERIFIED'),
('3520212345699','Sung', 'Hua', '1995-05-15','M','sung@demo.com', '+92-333-9998887','Model Town','Lahore',750,'VERIFIED'),
('3520212345600','Pending','User', '1998-01-01','O','pending@test.com', '+92-300-0000000','F-7','Islamabad',500,'PENDING');
-- ------ 5. EMPLOYEES ------------------------------------------------------------------------------------------------------------------------------
INSERT INTO employees (branch_id,first_name,last_name,cnic,designation,department,salary,email,hire_date) VALUES
(1,'Khalid','Rahman','3520299999901','Manager','Management',250000,'k.rahman@cbs.pk','2010-01-15'),
(1,'Amina','Syed','3520299999902','Teller','Operations',80000,'a.syed@cbs.pk','2015-03-10'),
(3,'Imran','Shah','3520299999905','Manager','Management',240000,'i.shah@cbs.pk','2009-04-01');
-- ------ 6. ACCOUNTS (Only for IDs 1-10) ---------------------------------------------------------------------
INSERT INTO accounts (account_number,customer_id,branch_id,account_type_id,balance,opened_date,status) VALUES
('PK36CBS0KHI001000001',1,1,1,125000,'2020-01-10','ACTIVE'),
('PK36CBS0KHI001000002',2,1,2,500000,'2020-01-10','ACTIVE'),
('PK36CBS0KHI002000003',3,2,3,85000,'2021-03-15','ACTIVE'),
('PK36CBS0LHR001000004',4,3,1,42000,'2022-05-01','ACTIVE'),
('PK36CBS0ISB001000005',5,4,2,2500000,'2018-11-12','ACTIVE'),
('PK36CBS0LHR001000006',6,3,1,120000,'2023-02-28','ACTIVE'),
('PK36CBS0KHI001000007',7,1,1,750000,'2017-08-19','ACTIVE'),
('PK36CBS0LHR001000008',8,3,2,5000,'2023-09-01','ACTIVE'),
('PK36CBS0KHI001000009',9,1,1,98000,'2020-12-10','ACTIVE'),
('PK36CBS0ISB001000010',10,4,2,1800000,'2016-03-22','ACTIVE');
-- ------ 7. LOANS ------------------------------------------------------------------------------------------------------------------------------------------
INSERT INTO loans (customer_id,loan_type_id,branch_id,principal_amount,interest_rate,tenure_months,emi_amount,outstanding,status) VALUES
(1,1,1,500000,18.00,36,18076,450000,'ACTIVE'),
(2,2,1,12000000,14.00,120,186395,11500000,'ACTIVE');
-- ------ 8. AUDIT LOG (Initial Entry) ---------------------------------------------------------------------------
INSERT INTO audit_log (table_name, operation, record_id, changed_by, remarks) VALUES
('SYSTEM', 'INSERT', 0, 'SYSTEM', 'Database population completed for Viva demo.');
-- ============================================================
-- CORE BANKING SYSTEM (CBS) --- ADVANCED QUERIES & ANALYTICS
-- DEVELOPED BY: HUZAIFA IMRAN & MUHAMMAD ARSLAN
-- Includes Joins, Set Operations, Subqueries, and Aggregates
-- ============================================================
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION A: JOINS
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- A1. INNER JOIN --- Customer + Account + Branch details
SELECT c.first_name || ' ' || c.last_name AS customer_name,
c.cnic,
a.account_number,
at.type_name AS account_type,
a.balance,
a.status,
b.branch_name,
b.city
FROM customers c
JOIN accounts a ON a.customer_id = c.customer_id
JOIN account_types at ON at.account_type_id = a.account_type_id
JOIN branches b ON b.branch_id = a.branch_id
WHERE a.status = 'ACTIVE'
ORDER BY a.balance DESC;
-- A2. LEFT OUTER JOIN --- All customers, show loan info if exists
SELECT c.first_name || ' ' || c.last_name AS customer_name,
c.credit_score,
l.loan_id,
lt.type_name AS loan_type,
l.principal_amount,
l.status AS loan_status
FROM customers c
LEFT JOIN loans l ON l.customer_id = c.customer_id
LEFT JOIN loan_types lt ON lt.loan_type_id = l.loan_type_id
ORDER BY c.customer_id;
-- A3. RIGHT OUTER JOIN --- All branches even if no employees yet
SELECT b.branch_name, b.city,
e.first_name || ' ' || e.last_name AS employee_name,
e.designation,
e.department
FROM employees e
RIGHT JOIN branches b ON b.branch_id = e.branch_id
ORDER BY b.branch_id;
-- A4. FULL OUTER JOIN --- Customers vs Loans (see who never borrowed)
-- A4. FULL OUTER JOIN --- Customers vs Loans (see who never borrowed)
SELECT c.first_name || ' ' || c.last_name AS customer_name,
COALESCE(l.loan_id::TEXT, 'NO LOAN') AS loan_ref,
COALESCE(l.principal_amount::TEXT, '---') AS principal,
COALESCE(l.status, '---') AS loan_status
FROM customers c
FULL OUTER JOIN loans l ON l.customer_id = c.customer_id
ORDER BY c.customer_id NULLS LAST;
-- A5. SELF JOIN --- Employees and their managers
SELECT e.first_name || ' ' || e.last_name AS employee,
e.designation,
m.first_name || ' ' || m.last_name AS reports_to,
m.designation AS manager_designation
FROM employees e
LEFT JOIN employees m ON m.employee_id = e.manager_id
ORDER BY m.employee_id NULLS LAST, e.employee_id;
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION B: SET OPERATIONS
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- B1. UNION --- All cities that have either branches or active customers
SELECT city, 'Branch City' AS source FROM branches
UNION
SELECT city, 'Customer City' AS source FROM customers
ORDER BY city;
-- B2. UNION ALL --- Combined deposit + interest transactions
SELECT 'DEPOSIT' AS txn_category, account_id, amount, txn_date
FROM transactions WHERE txn_type = 'DEPOSIT'
UNION ALL
SELECT 'INTEREST', account_id, amount, txn_date
FROM transactions WHERE txn_type = 'INTEREST'
ORDER BY txn_date DESC;
-- B3. INTERSECT --- Customers who have BOTH an active account AND an active loan
SELECT customer_id FROM accounts WHERE status = 'ACTIVE'
INTERSECT
SELECT customer_id FROM loans WHERE status = 'ACTIVE';
-- B4. EXCEPT --- Customers with an account but NO loan of any kind
-- B4. EXCEPT --- Customers with an account but NO loan of any kind
SELECT customer_id FROM accounts
EXCEPT
SELECT customer_id FROM loans;
-- B5. EXCEPT --- Branch cities with branches but NO customers living there
SELECT city FROM branches
EXCEPT
SELECT city FROM customers;
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION C: SUBQUERIES
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- C1. Non-correlated Subquery --- Customers with above-average credit score
SELECT first_name || ' ' || last_name AS customer_name, credit_score, city
FROM customers
WHERE credit_score > (SELECT AVG(credit_score) FROM customers)
ORDER BY credit_score DESC;
-- C2. Non-correlated Subquery --- Accounts with the highest balance per type
SELECT a.account_number, at.type_name, a.balance,
c.first_name || ' ' || c.last_name AS owner
FROM accounts a
JOIN account_types at ON at.account_type_id = a.account_type_id
JOIN customers c ON c.customer_id = a.customer_id
WHERE (a.account_type_id, a.balance) IN (
SELECT account_type_id, MAX(balance)
FROM accounts
GROUP BY account_type_id
)
ORDER BY a.balance DESC;
-- C3. Correlated Subquery --- Customers whose total loan outstanding > account balance
-- C3. Correlated Subquery --- Customers whose total loan outstanding > account balance
SELECT c.first_name || ' ' || c.last_name AS customer_name,
c.credit_score,
(SELECT SUM(l.outstanding) FROM loans l
WHERE l.customer_id = c.customer_id AND l.status = 'ACTIVE') AS total_outstanding,
(SELECT SUM(a.balance) FROM accounts a
WHERE a.customer_id = c.customer_id) AS total_balance
FROM customers c
WHERE COALESCE((SELECT SUM(l.outstanding) FROM loans l
WHERE l.customer_id = c.customer_id AND l.status = 'ACTIVE'), 0)
> COALESCE((SELECT SUM(a.balance) FROM accounts a
WHERE a.customer_id = c.customer_id), 0);
-- C4. Correlated Subquery --- Accounts with NO transactions in last 90 days (dormancy check)
-- C4. Correlated Subquery --- Accounts with NO transactions in last 90 days (dormancy check)
SELECT a.account_number, c.first_name || ' ' || c.last_name AS owner,
a.balance, a.last_txn_date
FROM accounts a
JOIN customers c ON c.customer_id = a.customer_id
WHERE NOT EXISTS (
SELECT 1 FROM transactions t
WHERE t.account_id = a.account_id
AND t.txn_date >= CURRENT_DATE - INTERVAL '90 days'
)
AND a.status = 'ACTIVE';
-- C5. Subquery in FROM --- Branch loan-to-deposit ratio
-- C5. Subquery in FROM --- Branch loan-to-deposit ratio
SELECT bp.branch_name, bp.city,
bp.total_deposits,
bp.total_loan_amount,
ROUND((bp.total_loan_amount / NULLIF(bp.total_deposits,0) * 100)::NUMERIC, 2) AS ldr_pct
FROM vw_branch_performance bp
ORDER BY ldr_pct DESC NULLS LAST;
-- C6. Non-correlated --- Top 3 transaction amounts ever recorded
-- C6. Non-correlated --- Top 3 transaction amounts ever recorded
SELECT * FROM (
SELECT t.reference_no, t.txn_type, t.amount,
a.account_number, t.txn_date,
RANK() OVER (ORDER BY t.amount DESC) AS amount_rank
FROM transactions t
JOIN accounts a ON a.account_id = t.account_id
) ranked
WHERE amount_rank <= 3;
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION D: AGGREGATE / ANALYTICAL QUERIES
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- D1. Branch-wise total deposits, loans, and customer count
-- C3. Correlated Subquery --- Customers whose total loan outstanding > account balance
SELECT b.branch_name, b.city,
COUNT(DISTINCT a.customer_id) AS customers,
COUNT(DISTINCT a.account_id) AS accounts,
COALESCE(SUM(a.balance), 0) AS total_deposits,
COUNT(DISTINCT l.loan_id) AS loan_count,
COALESCE(SUM(l.outstanding), 0) AS loan_book
FROM branches b
LEFT JOIN accounts a ON a.branch_id = b.branch_id AND a.status = 'ACTIVE'
LEFT JOIN loans l ON l.branch_id = b.branch_id AND l.status = 'ACTIVE'
GROUP BY b.branch_name, b.city
ORDER BY total_deposits DESC;
-- D2. Monthly transaction volume and value for current year
-- D2. Monthly transaction volume and value for current year
SELECT EXTRACT(MONTH FROM txn_date)::INT AS month,
txn_type,
COUNT(*) AS txn_count,
SUM(amount) AS total_value,
AVG(amount) AS avg_value,
MAX(amount) AS max_single_txn
FROM transactions
WHERE EXTRACT(YEAR FROM txn_date) = EXTRACT(YEAR FROM CURRENT_DATE)
GROUP BY EXTRACT(MONTH FROM txn_date), txn_type
ORDER BY month, txn_type;
-- D3. Loan repayment health --- paid vs outstanding per loan
SELECT l.loan_id,
c.first_name || ' ' || c.last_name AS borrower,
lt.type_name AS loan_type,
l.principal_amount,
l.outstanding,
l.principal_amount - l.outstanding AS amount_repaid,
ROUND(((l.principal_amount - l.outstanding) / l.principal_amount * 100)::NUMERIC, 1)
AS repaid_pct
FROM loans l
JOIN customers c ON c.customer_id = l.customer_id
JOIN loan_types lt ON lt.loan_type_id = l.loan_type_id
WHERE l.status IN ('ACTIVE','CLOSED')
ORDER BY repaid_pct DESC;
-- ============================================================
-- CORE BANKING SYSTEM (CBS) --- PL/SQL & AUTOMATION
-- DEVELOPED BY: HUZAIFA IMRAN & MUHAMMAD ARSLAN
-- PHASE IV: PL/pgSQL MODULE
-- Includes Composite Types, Functions, Procedures, and Triggers
-- ============================================================
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION 1: COMPOSITE TYPES
-- Oracle Object Types become PostgreSQL composite types.
-- Member functions become standalone functions instead.
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- 1A. Transaction Receipt composite type
DROP TYPE IF EXISTS txn_receipt_t CASCADE;
CREATE TYPE txn_receipt_t AS (
reference_no VARCHAR(30),
account_number VARCHAR(20),
txn_type VARCHAR(20),
amount NUMERIC(14,2),
balance_after NUMERIC(14,2),
txn_timestamp TIMESTAMP,
status VARCHAR(15)
);
-- Equivalent of the Oracle MEMBER FUNCTION to_string()
CREATE OR REPLACE FUNCTION txn_receipt_to_string(r txn_receipt_t)
RETURNS TEXT
LANGUAGE plpgsql AS $$
BEGIN
RETURN 'Receipt[' || r.reference_no || '] ' ||
r.txn_type || ' PKR ' || TO_CHAR(r.amount, 'FM9,999,999.00') ||
' | Bal: PKR ' || TO_CHAR(r.balance_after, 'FM9,999,999.00') ||
' | ' || TO_CHAR(r.txn_timestamp, 'DD-Mon-YYYY HH24:MI:SS');
END;
$$;
-- 1B. EMI Schedule row composite type
DROP TYPE IF EXISTS emi_row_t CASCADE;
CREATE TYPE emi_row_t AS (
installment_no INT,
due_date DATE,
emi_amount NUMERIC(12,2),
principal_component NUMERIC(12,2),
interest_component NUMERIC(12,2),
balance_remaining NUMERIC(14,2)
);
-- emi_schedule_t (a TABLE OF emi_row_t) is expressed via RETURNS TABLE or SETOF emi_row_t
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SECTION 2: FUNCTIONS & PROCEDURES
-- Oracle Packages have no direct equivalent in PostgreSQL.
-- We use a dedicated schema per package to group related routines.
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- Create schemas to mirror Oracle package namespaces
CREATE SCHEMA IF NOT EXISTS pkg_account_ops;
CREATE SCHEMA IF NOT EXISTS pkg_loan_mgmt;
CREATE SCHEMA IF NOT EXISTS pkg_reports;
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SCHEMA pkg_account_ops --- Core account operations
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- ------ next_reference ------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE FUNCTION pkg_account_ops.next_reference()
RETURNS TEXT
LANGUAGE plpgsql AS $$
BEGIN
-- Oracle: 'REF' || TO_CHAR(SYSTIMESTAMP,'YYYYMMDDHH24MISSFF3')
RETURN 'REF' || TO_CHAR(CURRENT_TIMESTAMP, 'YYYYMMDDHH24MISSMS');
END;
$$;
-- ------ get_balance ---------------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE FUNCTION pkg_account_ops.get_balance(p_account_id INT)
RETURNS NUMERIC
LANGUAGE plpgsql AS $$
DECLARE
v_bal NUMERIC;
BEGIN
SELECT balance INTO v_bal FROM accounts WHERE account_id = p_account_id;
RETURN v_bal;
END;
$$;
-- ------ Private helper: validate_account ---------------------------------------------------------------------------
-- Returns a record of accounts for the given id (row-locked).
-- Raises an exception if the account is FROZEN/CLOSED or not found.
CREATE OR REPLACE FUNCTION pkg_account_ops.validate_account(p_account_id INT)
RETURNS accounts -- returns the full accounts row
LANGUAGE plpgsql AS $$
DECLARE
v_acc accounts%ROWTYPE;
BEGIN
SELECT * INTO v_acc FROM accounts WHERE account_id = p_account_id FOR UPDATE;
IF NOT FOUND THEN
RAISE EXCEPTION 'Account ID % not found.', p_account_id
USING ERRCODE = 'P0010';
END IF;
IF v_acc.status IN ('FROZEN', 'CLOSED') THEN
RAISE EXCEPTION 'Account % is %.', v_acc.account_number, v_acc.status
USING ERRCODE = 'P0002';
END IF;
RETURN v_acc;
END;
$$;
-- ------ deposit ---------------------------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE FUNCTION pkg_account_ops.deposit(
p_account_id INT,
p_amount NUMERIC,
p_description TEXT DEFAULT NULL,
p_channel TEXT DEFAULT 'BRANCH',
p_emp_id INT DEFAULT NULL
)
RETURNS txn_receipt_t
LANGUAGE plpgsql AS $$
DECLARE
v_acc accounts%ROWTYPE;
v_ref TEXT;
v_new_bal NUMERIC(14,2);
v_receipt txn_receipt_t;
BEGIN
IF p_amount <= 0 THEN
RAISE EXCEPTION 'Deposit amount must be positive.'
USING ERRCODE = 'P0003';
END IF;
v_acc := pkg_account_ops.validate_account(p_account_id);
v_ref := pkg_account_ops.next_reference();
v_new_bal := v_acc.balance + p_amount;
UPDATE accounts
SET balance = v_new_bal, last_txn_date = CURRENT_DATE
WHERE account_id = p_account_id;
INSERT INTO transactions
(account_id, txn_type, amount, balance_after, reference_no,
description, channel, txn_date, performed_by, status)
VALUES
(p_account_id, 'DEPOSIT', p_amount, v_new_bal, v_ref,
COALESCE(p_description, 'Cash Deposit'), p_channel,
CURRENT_DATE, p_emp_id, 'SUCCESS');
v_receipt := ROW(v_ref, v_acc.account_number, 'DEPOSIT',
p_amount, v_new_bal, CURRENT_TIMESTAMP, 'SUCCESS')::txn_receipt_t;
RETURN v_receipt;
END;
$$;
-- ------ withdraw ------------------------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE FUNCTION pkg_account_ops.withdraw(
p_account_id INT,
p_amount NUMERIC,
p_description TEXT DEFAULT NULL,
p_channel TEXT DEFAULT 'ATM'
)
RETURNS txn_receipt_t
LANGUAGE plpgsql AS $$
DECLARE
v_acc accounts%ROWTYPE;
v_ref TEXT;
v_new_bal NUMERIC(14,2);
v_receipt txn_receipt_t;
BEGIN
IF p_amount <= 0 THEN
RAISE EXCEPTION 'Withdrawal amount must be positive.'
USING ERRCODE = 'P0003';
END IF;
v_acc := pkg_account_ops.validate_account(p_account_id);
IF v_acc.balance < p_amount THEN
RAISE EXCEPTION 'Insufficient funds. Balance: %, Requested: %',
v_acc.balance, p_amount
USING ERRCODE = 'P0001';
END IF;
v_ref := pkg_account_ops.next_reference();
v_new_bal := v_acc.balance - p_amount;
UPDATE accounts
SET balance = v_new_bal, last_txn_date = CURRENT_DATE
WHERE account_id = p_account_id;
INSERT INTO transactions
(account_id, txn_type, amount, balance_after, reference_no,
description, channel, txn_date, status)
VALUES
(p_account_id, 'WITHDRAWAL', p_amount, v_new_bal, v_ref,
COALESCE(p_description, 'Cash Withdrawal'), p_channel,
CURRENT_DATE, 'SUCCESS');
v_receipt := ROW(v_ref, v_acc.account_number, 'WITHDRAWAL',
p_amount, v_new_bal, CURRENT_TIMESTAMP, 'SUCCESS')::txn_receipt_t;
RETURN v_receipt;
END;
$$;
-- ------ transfer ------------------------------------------------------------------------------------------------------------------------------------------------------
-- Oracle PROCEDURE with OUT --- PostgreSQL PROCEDURE (or FUNCTION returning void/composite)
CREATE OR REPLACE PROCEDURE pkg_account_ops.transfer(
p_from_account_id INT,
p_to_account_id INT,
p_amount NUMERIC,
p_description TEXT DEFAULT NULL
)
LANGUAGE plpgsql AS $$
DECLARE
v_from accounts%ROWTYPE;
v_to accounts%ROWTYPE;
v_ref TEXT;
BEGIN
IF p_amount <= 0 THEN
RAISE EXCEPTION 'Transfer amount must be positive.'
USING ERRCODE = 'P0003';
END IF;
v_from := pkg_account_ops.validate_account(p_from_account_id);
v_to := pkg_account_ops.validate_account(p_to_account_id);
IF v_from.balance < p_amount THEN
RAISE EXCEPTION 'Insufficient funds for transfer.'
USING ERRCODE = 'P0001';
END IF;
v_ref := pkg_account_ops.next_reference();
UPDATE accounts SET balance = balance - p_amount, last_txn_date = CURRENT_DATE
WHERE account_id = p_from_account_id;
UPDATE accounts SET balance = balance + p_amount, last_txn_date = CURRENT_DATE
WHERE account_id = p_to_account_id;
INSERT INTO transactions
(account_id, txn_type, amount, balance_after, reference_no, description, channel, status)
VALUES
(p_from_account_id, 'TRANSFER_OUT', p_amount,
v_from.balance - p_amount, v_ref,
COALESCE(p_description, 'Transfer to ' || v_to.account_number),
'INTERNET', 'SUCCESS');
INSERT INTO transactions
(account_id, txn_type, amount, balance_after, reference_no, description, channel, status)
VALUES
(p_to_account_id, 'TRANSFER_IN', p_amount,
v_to.balance + p_amount, v_ref,
COALESCE(p_description, 'Transfer from ' || v_from.account_number),
'INTERNET', 'SUCCESS');
END;
$$;
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- SCHEMA pkg_loan_mgmt --- Loan lifecycle & EMI calculations
-- ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
-- ------ calc_emi ------------------------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE FUNCTION pkg_loan_mgmt.calc_emi(
p_principal NUMERIC,
p_annual_rate NUMERIC,
p_months INT
)
RETURNS NUMERIC
LANGUAGE plpgsql AS $$
DECLARE
v_r NUMERIC;
BEGIN
v_r := p_annual_rate / 12.0 / 100.0;
RETURN ROUND(
(p_principal * v_r * POWER(1 + v_r, p_months)
/ (POWER(1 + v_r, p_months) - 1))::NUMERIC,
2
);
END;
$$;
-- ------ generate_schedule ---------------------------------------------------------------------------------------------------------------------------
-- Oracle: PIPELINED FUNCTION returning TABLE type
-- PostgreSQL: RETURNS TABLE with RETURN NEXT
CREATE OR REPLACE FUNCTION pkg_loan_mgmt.generate_schedule(
p_principal NUMERIC,
p_annual_rate NUMERIC,
p_months INT,
p_start_date DATE DEFAULT CURRENT_DATE
)
RETURNS TABLE (
installment_no INT,
due_date DATE,
emi_amount NUMERIC(12,2),
principal_component NUMERIC(12,2),
interest_component NUMERIC(12,2),
balance_remaining NUMERIC(14,2)
)
LANGUAGE plpgsql AS $$
DECLARE
v_emi NUMERIC := pkg_loan_mgmt.calc_emi(p_principal, p_annual_rate, p_months);
v_balance NUMERIC := p_principal;
v_r NUMERIC := p_annual_rate / 12.0 / 100.0;
v_interest NUMERIC;
v_principal NUMERIC;
v_due DATE := p_start_date + INTERVAL '1 month';
i INT;
BEGIN
FOR i IN 1 .. p_months LOOP
v_interest := ROUND((v_balance * v_r)::NUMERIC, 2);
v_principal := v_emi - v_interest;
IF i = p_months THEN -- last-instalment rounding fix
v_principal := v_balance;
v_emi := v_balance + v_interest;
END IF;
v_balance := v_balance - v_principal;
installment_no := i;
due_date := v_due;
emi_amount := ROUND(v_emi::NUMERIC, 2);
principal_component := ROUND(v_principal::NUMERIC, 2);
interest_component := ROUND(v_interest::NUMERIC, 2);
balance_remaining := ROUND(GREATEST(v_balance, 0)::NUMERIC, 2);
RETURN NEXT;
-- Oracle ADD_MONTHS(d,1) --- d + INTERVAL '1 month'
v_due := v_due + INTERVAL '1 month';
END LOOP;
END;
$$;
-- ------ approve_loan ------------------------------------------------------------------------------------------------------------------------------------------
CREATE OR REPLACE PROCEDURE pkg_loan_mgmt.approve_loan(
p_loan_id INT,
p_officer_id INT,
p_rate NUMERIC DEFAULT NULL
)
LANGUAGE plpgsql AS $$
DECLARE
v_loan loans%ROWTYPE;
v_ltype loan_types%ROWTYPE;
v_rate NUMERIC;
BEGIN
SELECT * INTO v_loan FROM loans WHERE loan_id = p_loan_id FOR UPDATE;
SELECT * INTO v_ltype FROM loan_types WHERE loan_type_id = v_loan.loan_type_id;
IF v_loan.status <> 'PENDING' THEN
RAISE EXCEPTION 'Loan % is not in PENDING status.', p_loan_id
USING ERRCODE = 'P0020';
END IF;
v_rate := COALESCE(p_rate, v_ltype.base_rate);
UPDATE loans SET
status = 'APPROVED',
approved_by = p_officer_id,
interest_rate = v_rate,
emi_amount = pkg_loan_mgmt.calc_emi(v_loan.principal_amount, v_rate, v_loan.tenure_months),
disbursement_date = CURRENT_DATE,
-- Oracle ADD_MONTHS --- interval arithmetic
maturity_date = CURRENT_DATE + (v_loan.tenure_months || ' months')::INTERVAL,
outstanding = v_loan.principal_amount