@platform-modules/civil-aviation-authority 2.3.264 → 2.3.266

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (80) hide show
  1. package/.env +0 -15
  2. package/dist/models/AnnualIncrementRequestEmployeeModel.d.ts +2 -1
  3. package/dist/models/AnnualIncrementRequestEmployeeModel.js +7 -2
  4. package/dist/models/AnnualIncrementRequestModel.d.ts +2 -1
  5. package/dist/models/AnnualIncrementRequestModel.js +7 -2
  6. package/dist/models/SlaApprovalsViewModel.js +47 -47
  7. package/dist/models/SlaMyRequestsViewModel.js +50 -50
  8. package/package.json +3 -2
  9. package/scripts/sync-sla-reports-sql.js +98 -0
  10. package/sql/README.md +21 -0
  11. package/sql/sla-reports-sync.manifest.json +7 -0
  12. package/sql/sla_reports_admin_procedures.sql +283 -0
  13. package/sql/sla_reports_approvals_workbook.sql +383 -0
  14. package/sql/sla_reports_procedures.sql +874 -0
  15. package/sql/vw_sla_approvals.sql +108 -108
  16. package/sql/vw_sla_my_requests.sql +53 -54
  17. package/src/data-source.ts +517 -517
  18. package/src/index.ts +495 -495
  19. package/src/models/AccessCardRequestModel.ts +135 -135
  20. package/src/models/AirportEntryPermitModel.ts +276 -276
  21. package/src/models/AnnualIncrementRequestEmployeeModel.ts +6 -1
  22. package/src/models/AnnualIncrementRequestModel.ts +6 -1
  23. package/src/models/AnnualTrainingPlanRequestModel.ts +153 -153
  24. package/src/models/DepartmentsModel.ts +25 -25
  25. package/src/models/DocumentDriveModel.ts +28 -28
  26. package/src/models/DocumentFolderModel.ts +45 -45
  27. package/src/models/HotelApprovalModel.ts +83 -83
  28. package/src/models/HousingContractCancelApprovalModel.ts +64 -64
  29. package/src/models/HousingContractCancelChatModel.ts +56 -56
  30. package/src/models/HousingContractRenewalApprovalModel.ts +64 -64
  31. package/src/models/HousingContractRenewalChatModel.ts +59 -59
  32. package/src/models/ITRequestAttachmentModel.ts +73 -73
  33. package/src/models/ITRequestChatModel.ts +74 -74
  34. package/src/models/ItApprovalsModel.ts +84 -84
  35. package/src/models/ItWorkflowModel.ts +55 -55
  36. package/src/models/LegalConsultationApprovalModel.ts +65 -65
  37. package/src/models/MissionTravelPassportExpiryNotificationConfigModel.ts +36 -36
  38. package/src/models/NotificationModel.ts +89 -89
  39. package/src/models/ResidentialUnitRentalApprovalModel.ts +143 -143
  40. package/src/models/ResidentialUnitRentalChatModel.ts +56 -56
  41. package/src/models/ResidentialUnitRentalRequestModel.ts +218 -218
  42. package/src/models/ServiceExtensionAfter60ApprovalModel.ts +87 -87
  43. package/src/models/ServiceSlaApprovalModel.ts +64 -64
  44. package/src/models/ServicesNotificationConfigModel.ts +55 -55
  45. package/src/models/SlaApprovalsViewModel.ts +135 -135
  46. package/src/models/SlaMyRequestsViewModel.ts +170 -170
  47. package/src/models/SlaRequestModel.ts +65 -65
  48. package/src/models/StudyLeaveRequestModel.ts +144 -144
  49. package/src/models/TrainingRequestModel.ts +164 -164
  50. package/src/models/TrainingRoomBookingRequestModel.ts +142 -142
  51. package/src/models/TrainingRoomNotificationConfigModel.ts +30 -30
  52. package/src/models/role.ts +34 -34
  53. package/src/models/user.ts +233 -233
  54. package/src/sla/sla-table-sync.service.ts +90 -90
  55. package/dist/models/DocumentMetadataModel.d.ts +0 -45
  56. package/dist/models/DocumentMetadataModel.js +0 -171
  57. package/dist/models/DocumentationDepartmentsModel.d.ts +0 -13
  58. package/dist/models/DocumentationDepartmentsModel.js +0 -53
  59. package/dist/models/FolderModel.d.ts +0 -16
  60. package/dist/models/FolderModel.js +0 -85
  61. package/dist/models/ImportExportMaterialModels.d.ts +0 -92
  62. package/dist/models/ImportExportMaterialModels.js +0 -307
  63. package/dist/models/PermissionModel.d.ts +0 -18
  64. package/dist/models/PermissionModel.js +0 -68
  65. package/dist/models/SecurityAccessApprovalModel.d.ts +0 -23
  66. package/dist/models/SecurityAccessApprovalModel.js +0 -82
  67. package/dist/models/SecurityAccessAttachmentModel.d.ts +0 -12
  68. package/dist/models/SecurityAccessAttachmentModel.js +0 -56
  69. package/dist/models/SecurityAccessChatModel.d.ts +0 -12
  70. package/dist/models/SecurityAccessChatModel.js +0 -56
  71. package/dist/models/SecurityAccessRequestModel.d.ts +0 -25
  72. package/dist/models/SecurityAccessRequestModel.js +0 -80
  73. package/dist/models/SecurityAccessWorkflowModel.d.ts +0 -24
  74. package/dist/models/SecurityAccessWorkflowModel.js +0 -84
  75. package/dist/models/ServiceExtensionAfterAge60Models.d.ts +0 -93
  76. package/dist/models/ServiceExtensionAfterAge60Models.js +0 -312
  77. package/dist/models/UUIDBaseModel.d.ts +0 -14
  78. package/dist/models/UUIDBaseModel.js +0 -66
  79. package/dist/models/WorkingHoursExtensionModels.d.ts +0 -88
  80. package/dist/models/WorkingHoursExtensionModels.js +0 -295
@@ -0,0 +1,283 @@
1
+ -- Admin SLA reports + approvals Excel export
2
+ -- Prerequisites: sla_reports_procedures.sql, vw_sla_my_requests.sql, vw_sla_approvals.sql
3
+ -- Applied via: npm run sync:sla-sql (after sla_reports_procedures.sql in manifest)
4
+
5
+ CREATE OR REPLACE FUNCTION sla_filtered_request_fields(p_request_obj JSONB)
6
+ RETURNS JSONB
7
+ LANGUAGE sql
8
+ IMMUTABLE
9
+ AS $$
10
+ SELECT
11
+ COALESCE(p_request_obj, '{}'::jsonb)
12
+ - 'id' - 'status' - 'role_id' - 'user_id' - 'createdBy' - 'created_at'
13
+ - 'updated_by' - 'updated_at' - 'req_user_department_id' - 'workflow_execution_id'
14
+ - 'department_id' - 'req_user_section_id' - 'service_type_id' - 'service_type'
15
+ - 'created_by' - 'service_id' - 'sub_service_id' - 'attachments'
16
+ - 'is_deleted' - 'sla_request' - 'sla_request_id';
17
+ $$;
18
+
19
+ -- Admin list views: served by Reports_Service TypeORM (sla-my-requests-view.query.ts, sla-approvals-view.query.ts)
20
+
21
+ -- ---------------------------------------------------------------------------
22
+ -- ADMIN — my requests Excel (optional p_target_user_id)
23
+ -- Per sub-service Requests + Approvals sheets (same as user my-requests download)
24
+ -- ---------------------------------------------------------------------------
25
+ CREATE OR REPLACE FUNCTION sp_sla_admin_my_requests_excel_export(
26
+ p_target_user_id INT DEFAULT NULL,
27
+ p_service_ids INT[] DEFAULT NULL,
28
+ p_sub_service_ids INT[] DEFAULT NULL,
29
+ p_statuses TEXT[] DEFAULT ARRAY['Pending','In Progress','Approved','Rejected'],
30
+ p_from_date DATE DEFAULT NULL,
31
+ p_to_date DATE DEFAULT NULL,
32
+ p_search_text TEXT DEFAULT NULL,
33
+ p_sort_by TEXT DEFAULT 'created_at',
34
+ p_sort_order TEXT DEFAULT 'DESC'
35
+ )
36
+ RETURNS TABLE (
37
+ sheet_order INT,
38
+ sub_service_id INT,
39
+ sub_service_name TEXT,
40
+ sheet_name TEXT,
41
+ headers JSONB,
42
+ rows JSONB
43
+ )
44
+ LANGUAGE plpgsql
45
+ STABLE
46
+ AS $$
47
+ DECLARE
48
+ v_sort_col TEXT;
49
+ v_sort_dir TEXT;
50
+ v_sub RECORD;
51
+ v_row RECORD;
52
+ v_dyn_keys TEXT[];
53
+ v_dyn_labels TEXT[];
54
+ v_common_labels TEXT[] := ARRAY[
55
+ 'SLA Request ID', 'Request ID', 'User', 'Status', 'Created At',
56
+ 'Created By', 'Department', 'Section'
57
+ ];
58
+ v_headers_labels TEXT[];
59
+ v_headers_json JSONB;
60
+ v_cells TEXT[];
61
+ v_all_rows JSONB;
62
+ v_k TEXT;
63
+ v_sheet_order INT := 0;
64
+ v_user_filter TEXT;
65
+ BEGIN
66
+ v_user_filter := CASE WHEN p_target_user_id IS NULL THEN 'TRUE' ELSE format('v.user_id = %s', p_target_user_id) END;
67
+
68
+ v_sort_col := CASE lower(trim(p_sort_by))
69
+ WHEN 'updated_at' THEN 'v.created_at'
70
+ WHEN 'id' THEN 'v.sla_request_id'
71
+ WHEN 'status' THEN 'v.status'
72
+ WHEN 'request_id' THEN 'v.request_id'
73
+ ELSE 'v.created_at'
74
+ END;
75
+ v_sort_dir := CASE WHEN upper(trim(p_sort_order)) = 'ASC' THEN 'ASC' ELSE 'DESC' END;
76
+
77
+ FOR v_sub IN
78
+ EXECUTE format($q$
79
+ SELECT DISTINCT
80
+ COALESCE(sr.sub_service_id, 0) AS sub_service_id_num,
81
+ COALESCE(NULLIF(trim(subsvc.sub_service_name), ''), NULLIF(trim(v.sub_service_name), ''), 'Uncategorized') AS sub_service_name
82
+ FROM vw_sla_my_requests v
83
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
84
+ LEFT JOIN caa_sub_services subsvc ON subsvc.id = sr.sub_service_id
85
+ WHERE %s
86
+ AND v.status = ANY($1)
87
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
88
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
89
+ AND ($4 IS NULL OR v.created_at::date >= $4)
90
+ AND ($5 IS NULL OR v.created_at::date <= $5)
91
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
92
+ ORDER BY sub_service_name ASC
93
+ $q$, v_user_filter)
94
+ USING p_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
95
+ LOOP
96
+ v_sheet_order := v_sheet_order + 1;
97
+
98
+ EXECUTE format($q$
99
+ SELECT COALESCE(array_agg(DISTINCT keys.k ORDER BY keys.k), ARRAY[]::TEXT[])
100
+ FROM vw_sla_my_requests v
101
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
102
+ CROSS JOIN LATERAL jsonb_object_keys(COALESCE(v.request_fields, '{}'::jsonb)) AS keys(k)
103
+ WHERE %s
104
+ AND v.status = ANY($1)
105
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
106
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
107
+ AND ($4 IS NULL OR v.created_at::date >= $4)
108
+ AND ($5 IS NULL OR v.created_at::date <= $5)
109
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
110
+ AND COALESCE(sr.sub_service_id, 0) = %s
111
+ $q$, v_user_filter, v_sub.sub_service_id_num)
112
+ INTO v_dyn_keys
113
+ USING p_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text;
114
+
115
+ v_dyn_labels := ARRAY[]::TEXT[];
116
+ IF v_dyn_keys IS NOT NULL THEN
117
+ FOREACH v_k IN ARRAY v_dyn_keys LOOP
118
+ v_dyn_labels := array_append(v_dyn_labels, sla_excel_header_label(v_k));
119
+ END LOOP;
120
+ END IF;
121
+
122
+ v_headers_labels := ARRAY(
123
+ SELECT sla_excel_header_label(lbl)
124
+ FROM unnest(v_common_labels || COALESCE(v_dyn_labels, ARRAY[]::TEXT[])) AS u(lbl)
125
+ );
126
+ v_headers_json := to_jsonb(v_headers_labels);
127
+ v_all_rows := '[]'::jsonb;
128
+
129
+ FOR v_row IN
130
+ EXECUTE format($q$
131
+ SELECT
132
+ v.sla_request_id, v.request_id, v.status, v.created_at, v.created_by,
133
+ v.req_user_department_id, v.req_user_section_id, v.request_fields,
134
+ TRIM(COALESCE(u.employee_name, '')) AS user_name
135
+ FROM vw_sla_my_requests v
136
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
137
+ LEFT JOIN users u ON u.id = v.user_id AND COALESCE(u.is_deleted, false) = false
138
+ WHERE %s
139
+ AND v.status = ANY($1)
140
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
141
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
142
+ AND ($4 IS NULL OR v.created_at::date >= $4)
143
+ AND ($5 IS NULL OR v.created_at::date <= $5)
144
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
145
+ AND COALESCE(sr.sub_service_id, 0) = %s
146
+ ORDER BY %s %s, v.sla_request_id ASC
147
+ $q$, v_user_filter, v_sub.sub_service_id_num, v_sort_col, v_sort_dir)
148
+ USING p_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
149
+ LOOP
150
+ v_cells := ARRAY[
151
+ COALESCE(v_row.sla_request_id::TEXT, ''),
152
+ COALESCE(v_row.request_id::TEXT, ''),
153
+ COALESCE(v_row.user_name, ''),
154
+ COALESCE(v_row.status, ''),
155
+ COALESCE(to_char(v_row.created_at AT TIME ZONE 'UTC', 'YYYY-MM-DD HH24:MI:SS'), ''),
156
+ COALESCE(v_row.created_by, ''),
157
+ COALESCE(v_row.req_user_department_id, ''),
158
+ COALESCE(v_row.req_user_section_id, '')
159
+ ];
160
+ IF v_dyn_keys IS NOT NULL THEN
161
+ FOREACH v_k IN ARRAY v_dyn_keys LOOP
162
+ v_cells := array_append(v_cells, COALESCE(sla_request_field_cell(v_row.request_fields, v_k), ''));
163
+ END LOOP;
164
+ END IF;
165
+ v_all_rows := v_all_rows || jsonb_build_array(to_jsonb(v_cells));
166
+ END LOOP;
167
+
168
+ sheet_order := v_sheet_order;
169
+ sub_service_id := NULLIF(v_sub.sub_service_id_num, 0);
170
+ sub_service_name := v_sub.sub_service_name;
171
+ sheet_name := sla_excel_sheet_name(v_sub.sub_service_name);
172
+ headers := v_headers_json;
173
+ rows := v_all_rows;
174
+ RETURN NEXT;
175
+ END LOOP;
176
+
177
+ RETURN QUERY
178
+ SELECT *
179
+ FROM sla_my_requests_approvals_excel_sheet(
180
+ v_user_filter,
181
+ p_statuses,
182
+ p_service_ids,
183
+ p_sub_service_ids,
184
+ p_from_date,
185
+ p_to_date,
186
+ p_search_text,
187
+ v_sheet_order + 1
188
+ );
189
+
190
+ RETURN;
191
+ END;
192
+ $$;
193
+
194
+ -- ---------------------------------------------------------------------------
195
+ -- APPROVALS — Excel export (approver-scoped; user download)
196
+ -- Workbook: per sub-service Requests + Approvals (see sla_reports_approvals_workbook.sql)
197
+ -- ---------------------------------------------------------------------------
198
+ DROP FUNCTION IF EXISTS sp_sla_approvals_excel_export_internal(
199
+ TEXT, BOOLEAN, INT, INT[], INT[], TEXT[], TEXT[], DATE, DATE, TEXT, TEXT, TEXT
200
+ );
201
+
202
+ CREATE OR REPLACE FUNCTION sp_sla_approvals_excel_export(
203
+ p_approver_user_id INT,
204
+ p_service_ids INT[] DEFAULT NULL,
205
+ p_sub_service_ids INT[] DEFAULT NULL,
206
+ p_request_statuses TEXT[] DEFAULT ARRAY['Pending','In Progress','Approved','Rejected'],
207
+ p_approval_statuses TEXT[] DEFAULT ARRAY['In Progress','Approved','Rejected'],
208
+ p_from_date DATE DEFAULT NULL,
209
+ p_to_date DATE DEFAULT NULL,
210
+ p_search_text TEXT DEFAULT NULL,
211
+ p_sort_by TEXT DEFAULT 'created_at',
212
+ p_sort_order TEXT DEFAULT 'DESC'
213
+ )
214
+ RETURNS TABLE (
215
+ sheet_order INT, sub_service_id INT, sub_service_name TEXT, sheet_name TEXT, headers JSONB, rows JSONB
216
+ )
217
+ LANGUAGE plpgsql
218
+ STABLE
219
+ AS $$
220
+ DECLARE
221
+ v_exists TEXT := sla_approval_queue_exists_sql(p_approver_user_id);
222
+ v_own TEXT := format('sr.user_id != %s', p_approver_user_id);
223
+ BEGIN
224
+ RETURN QUERY
225
+ SELECT * FROM sp_sla_approvals_workbook_excel_export(
226
+ v_exists,
227
+ v_own,
228
+ p_service_ids,
229
+ p_sub_service_ids,
230
+ p_request_statuses,
231
+ p_approval_statuses,
232
+ p_from_date,
233
+ p_to_date,
234
+ p_search_text,
235
+ p_sort_by,
236
+ p_sort_order
237
+ );
238
+ END;
239
+ $$;
240
+
241
+ -- ---------------------------------------------------------------------------
242
+ -- ADMIN — approvals Excel
243
+ -- Per sub-service Requests + Approvals; optional approver_user_id (NULL = all queues)
244
+ -- Does not exclude own requests (unlike user approvals download)
245
+ -- ---------------------------------------------------------------------------
246
+ CREATE OR REPLACE FUNCTION sp_sla_admin_approvals_excel_export(
247
+ p_target_approver_user_id INT DEFAULT NULL,
248
+ p_service_ids INT[] DEFAULT NULL,
249
+ p_sub_service_ids INT[] DEFAULT NULL,
250
+ p_request_statuses TEXT[] DEFAULT ARRAY['Pending','In Progress','Approved','Rejected'],
251
+ p_approval_statuses TEXT[] DEFAULT ARRAY['In Progress','Approved','Rejected'],
252
+ p_from_date DATE DEFAULT NULL,
253
+ p_to_date DATE DEFAULT NULL,
254
+ p_search_text TEXT DEFAULT NULL,
255
+ p_sort_by TEXT DEFAULT 'created_at',
256
+ p_sort_order TEXT DEFAULT 'DESC'
257
+ )
258
+ RETURNS TABLE (
259
+ sheet_order INT, sub_service_id INT, sub_service_name TEXT, sheet_name TEXT, headers JSONB, rows JSONB
260
+ )
261
+ LANGUAGE plpgsql
262
+ STABLE
263
+ AS $$
264
+ DECLARE
265
+ v_exists TEXT := sla_approval_queue_exists_sql_optional(p_target_approver_user_id);
266
+ v_own TEXT := 'TRUE';
267
+ BEGIN
268
+ RETURN QUERY
269
+ SELECT * FROM sp_sla_approvals_workbook_excel_export(
270
+ v_exists,
271
+ v_own,
272
+ p_service_ids,
273
+ p_sub_service_ids,
274
+ p_request_statuses,
275
+ p_approval_statuses,
276
+ p_from_date,
277
+ p_to_date,
278
+ p_search_text,
279
+ p_sort_by,
280
+ p_sort_order
281
+ );
282
+ END;
283
+ $$;
@@ -0,0 +1,383 @@
1
+ -- Approvals Excel download: per sub-service "{name} Requests" + "{name} Approvals"
2
+ -- Approver queue filter uses EXISTS on sla_approval (same pattern as HR annual-training-plan).
3
+ -- Applied via: npm run sync:sla-sql
4
+
5
+ CREATE OR REPLACE FUNCTION sla_approval_matches_user_view(p_va_alias TEXT, p_user_id INT)
6
+ RETURNS TEXT
7
+ LANGUAGE sql
8
+ IMMUTABLE
9
+ AS $$
10
+ SELECT format('(
11
+ %1$s.approver_user_id = %2$s
12
+ OR %1$s.delegate_user_id = %2$s
13
+ OR (
14
+ %1$s.approver_user_id IS NULL
15
+ AND %1$s.delegate_user_id IS NULL
16
+ AND %1$s.approval_role_id IS NOT NULL
17
+ AND %1$s.approval_department_id IS NOT NULL
18
+ AND EXISTS (
19
+ SELECT 1 FROM user_role ur
20
+ WHERE ur.user_id = %2$s
21
+ AND ur.role_id = %1$s.approval_role_id
22
+ AND ur.department_id = %1$s.approval_department_id
23
+ AND (
24
+ %1$s.approval_section_id IS NULL
25
+ OR (%1$s.approval_section_id IS NOT NULL AND (ur.section_id = %1$s.approval_section_id OR ur.section_id IS NULL))
26
+ )
27
+ AND ur.is_deleted = false AND ur.is_active = true
28
+ )
29
+ )
30
+ OR (
31
+ %1$s.approver_user_id IS NULL
32
+ AND %1$s.delegate_user_id IS NULL
33
+ AND %1$s.approval_role_id IS NULL
34
+ AND %1$s.approval_department_id IS NOT NULL
35
+ AND %1$s.approval_section_id IS NOT NULL
36
+ AND EXISTS (
37
+ SELECT 1 FROM users u
38
+ WHERE u.id = %2$s AND COALESCE(u.is_deleted, false) = false
39
+ AND u.department = %1$s.approval_department_id AND u.section = %1$s.approval_section_id
40
+ )
41
+ )
42
+ OR (
43
+ %1$s.approver_user_id IS NULL
44
+ AND %1$s.delegate_user_id IS NULL
45
+ AND %1$s.approval_role_id IS NULL
46
+ AND %1$s.approval_department_id IS NOT NULL
47
+ AND %1$s.approval_section_id IS NULL
48
+ AND EXISTS (
49
+ SELECT 1 FROM users u
50
+ WHERE u.id = %2$s AND COALESCE(u.is_deleted, false) = false
51
+ AND u.department = %1$s.approval_department_id
52
+ )
53
+ )
54
+ )', p_va_alias, p_user_id);
55
+ $$;
56
+
57
+ -- EXISTS (sla_approval appr ...) — mirrors annual_training_plan_approvals queue filter
58
+ CREATE OR REPLACE FUNCTION sla_approval_queue_exists_sql(p_user_id INT)
59
+ RETURNS TEXT
60
+ LANGUAGE sql
61
+ IMMUTABLE
62
+ AS $$
63
+ SELECT format(
64
+ 'EXISTS (
65
+ SELECT 1
66
+ FROM sla_approval appr
67
+ WHERE appr.request_id = sr.request_id
68
+ AND COALESCE(appr.is_deleted, false) = false
69
+ AND (appr.service_id = sr.service_id OR appr.service_id IS NULL)
70
+ AND (appr.sub_service_id = sr.sub_service_id OR appr.sub_service_id IS NULL)
71
+ AND (%s)
72
+ AND (appr.approval_status = ANY($1) OR appr.approval_status IS NULL)
73
+ )',
74
+ sla_approval_matches_user('appr', p_user_id)
75
+ );
76
+ $$;
77
+
78
+ -- Admin: all approvers (NULL user) — any request with a matching approval row
79
+ CREATE OR REPLACE FUNCTION sla_approval_queue_exists_sql_optional(p_user_id INT)
80
+ RETURNS TEXT
81
+ LANGUAGE sql
82
+ STABLE
83
+ AS $$
84
+ SELECT CASE
85
+ WHEN p_user_id IS NULL THEN
86
+ 'EXISTS (
87
+ SELECT 1
88
+ FROM sla_approval appr
89
+ WHERE appr.request_id = sr.request_id
90
+ AND COALESCE(appr.is_deleted, false) = false
91
+ AND (appr.service_id = sr.service_id OR appr.service_id IS NULL)
92
+ AND (appr.sub_service_id = sr.sub_service_id OR appr.sub_service_id IS NULL)
93
+ AND (appr.approval_status = ANY($1) OR appr.approval_status IS NULL)
94
+ )'
95
+ ELSE sla_approval_queue_exists_sql(p_user_id)
96
+ END;
97
+ $$;
98
+
99
+ DROP FUNCTION IF EXISTS sp_sla_approvals_workbook_excel_export(
100
+ TEXT, TEXT, INT[], INT[], TEXT[], TEXT[], DATE, DATE, TEXT, TEXT, TEXT
101
+ );
102
+
103
+ CREATE OR REPLACE FUNCTION sp_sla_approvals_workbook_excel_export(
104
+ p_queue_exists_sql TEXT,
105
+ p_own_requests_filter_sql TEXT,
106
+ p_service_ids INT[] DEFAULT NULL,
107
+ p_sub_service_ids INT[] DEFAULT NULL,
108
+ p_request_statuses TEXT[] DEFAULT ARRAY['Pending','In Progress','Approved','Rejected'],
109
+ p_approval_statuses TEXT[] DEFAULT ARRAY['In Progress','Approved','Rejected'],
110
+ p_from_date DATE DEFAULT NULL,
111
+ p_to_date DATE DEFAULT NULL,
112
+ p_search_text TEXT DEFAULT NULL,
113
+ p_sort_by TEXT DEFAULT 'created_at',
114
+ p_sort_order TEXT DEFAULT 'DESC'
115
+ )
116
+ RETURNS TABLE (
117
+ sheet_order INT,
118
+ sub_service_id INT,
119
+ sub_service_name TEXT,
120
+ sheet_name TEXT,
121
+ headers JSONB,
122
+ rows JSONB
123
+ )
124
+ LANGUAGE plpgsql
125
+ STABLE
126
+ AS $$
127
+ DECLARE
128
+ v_sort_col_req TEXT;
129
+ v_sort_dir TEXT;
130
+ v_sub RECORD;
131
+ v_row RECORD;
132
+ v_approval RECORD;
133
+ v_dyn_keys TEXT[];
134
+ v_dyn_labels TEXT[];
135
+ v_common_req_labels TEXT[] := ARRAY[
136
+ 'SLA Request ID', 'Request ID', 'User', 'Status', 'Created At',
137
+ 'Created By', 'Department', 'Section'
138
+ ];
139
+ v_approval_headers TEXT[] := ARRAY[
140
+ 'Request ID', 'Service ID', 'Service Name', 'Sub Service ID', 'Sub Service Name',
141
+ 'SLA Approval ID', 'Source Approval ID', 'Level', 'Approval Status', 'Request Status',
142
+ 'Approval Role ID', 'Approval Role Name', 'Approval Department ID', 'Approval Department Name',
143
+ 'Approval Section ID', 'Approval Section Name',
144
+ 'Approver User ID', 'Approver User Name', 'Delegate User ID', 'Delegate User Name',
145
+ 'Approved By ID', 'Approved By Name', 'Comment', 'Created At', 'Updated At'
146
+ ];
147
+ v_headers_labels TEXT[];
148
+ v_headers_json JSONB;
149
+ v_cells TEXT[];
150
+ v_all_rows JSONB;
151
+ v_k TEXT;
152
+ v_sheet_order INT := 0;
153
+ v_sheet_label TEXT;
154
+ BEGIN
155
+ v_sort_col_req := CASE lower(trim(p_sort_by))
156
+ WHEN 'updated_at' THEN 'v.created_at'
157
+ WHEN 'id' THEN 'v.sla_request_id'
158
+ WHEN 'status' THEN 'v.status'
159
+ WHEN 'request_id' THEN 'v.request_id'
160
+ ELSE 'v.created_at'
161
+ END;
162
+ v_sort_dir := CASE WHEN upper(trim(p_sort_order)) = 'ASC' THEN 'ASC' ELSE 'DESC' END;
163
+
164
+ -- ----- Per sub-service: Requests (queue requests with dynamic request_fields) -----
165
+ FOR v_sub IN
166
+ EXECUTE format($q$
167
+ SELECT DISTINCT
168
+ COALESCE(sr.sub_service_id, 0) AS sub_service_id_num,
169
+ COALESCE(NULLIF(trim(subsvc.sub_service_name), ''), NULLIF(trim(v.sub_service_name), ''), 'Uncategorized') AS sub_service_name
170
+ FROM vw_sla_my_requests v
171
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
172
+ LEFT JOIN caa_sub_services subsvc ON subsvc.id = sr.sub_service_id
173
+ WHERE (%s)
174
+ AND (%s)
175
+ AND sr.status::TEXT = ANY($1)
176
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
177
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
178
+ AND ($4 IS NULL OR v.created_at::date >= $4)
179
+ AND ($5 IS NULL OR v.created_at::date <= $5)
180
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
181
+ ORDER BY sub_service_name ASC
182
+ $q$, p_own_requests_filter_sql, p_queue_exists_sql)
183
+ USING p_request_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
184
+ LOOP
185
+ v_sheet_order := v_sheet_order + 1;
186
+
187
+ EXECUTE format($q$
188
+ SELECT COALESCE(array_agg(DISTINCT keys.k ORDER BY keys.k), ARRAY[]::TEXT[])
189
+ FROM vw_sla_my_requests v
190
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
191
+ CROSS JOIN LATERAL jsonb_object_keys(COALESCE(v.request_fields, '{}'::jsonb)) AS keys(k)
192
+ WHERE (%s)
193
+ AND (%s)
194
+ AND sr.status::TEXT = ANY($1)
195
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
196
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
197
+ AND ($4 IS NULL OR v.created_at::date >= $4)
198
+ AND ($5 IS NULL OR v.created_at::date <= $5)
199
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
200
+ AND COALESCE(sr.sub_service_id, 0) = %s
201
+ $q$, p_own_requests_filter_sql, p_queue_exists_sql, v_sub.sub_service_id_num)
202
+ INTO v_dyn_keys
203
+ USING p_request_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text;
204
+
205
+ v_dyn_labels := ARRAY[]::TEXT[];
206
+ IF v_dyn_keys IS NOT NULL THEN
207
+ FOREACH v_k IN ARRAY v_dyn_keys LOOP
208
+ v_dyn_labels := array_append(v_dyn_labels, sla_excel_header_label(v_k));
209
+ END LOOP;
210
+ END IF;
211
+
212
+ v_headers_labels := ARRAY(
213
+ SELECT sla_excel_header_label(lbl)
214
+ FROM unnest(v_common_req_labels || COALESCE(v_dyn_labels, ARRAY[]::TEXT[])) AS u(lbl)
215
+ );
216
+ v_headers_json := to_jsonb(v_headers_labels);
217
+ v_all_rows := '[]'::jsonb;
218
+
219
+ FOR v_row IN
220
+ EXECUTE format($q$
221
+ SELECT
222
+ v.sla_request_id,
223
+ v.request_id,
224
+ v.status,
225
+ v.created_at,
226
+ v.created_by,
227
+ v.req_user_department_id,
228
+ v.req_user_section_id,
229
+ v.request_fields,
230
+ TRIM(COALESCE(u.employee_name, '')) AS user_name
231
+ FROM vw_sla_my_requests v
232
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
233
+ LEFT JOIN users u ON u.id = v.user_id AND COALESCE(u.is_deleted, false) = false
234
+ WHERE (%s)
235
+ AND (%s)
236
+ AND sr.status::TEXT = ANY($1)
237
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
238
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
239
+ AND ($4 IS NULL OR v.created_at::date >= $4)
240
+ AND ($5 IS NULL OR v.created_at::date <= $5)
241
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
242
+ AND COALESCE(sr.sub_service_id, 0) = %s
243
+ ORDER BY %s %s, v.sla_request_id ASC
244
+ $q$, p_own_requests_filter_sql, p_queue_exists_sql, v_sub.sub_service_id_num, v_sort_col_req, v_sort_dir)
245
+ USING p_request_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
246
+ LOOP
247
+ v_cells := ARRAY[
248
+ COALESCE(v_row.sla_request_id::TEXT, ''),
249
+ COALESCE(v_row.request_id::TEXT, ''),
250
+ COALESCE(v_row.user_name, ''),
251
+ COALESCE(v_row.status, ''),
252
+ COALESCE(to_char(v_row.created_at AT TIME ZONE 'UTC', 'YYYY-MM-DD HH24:MI:SS'), ''),
253
+ COALESCE(v_row.created_by, ''),
254
+ COALESCE(v_row.req_user_department_id, ''),
255
+ COALESCE(v_row.req_user_section_id, '')
256
+ ];
257
+ IF v_dyn_keys IS NOT NULL THEN
258
+ FOREACH v_k IN ARRAY v_dyn_keys LOOP
259
+ v_cells := array_append(v_cells, COALESCE(sla_request_field_cell(v_row.request_fields, v_k), ''));
260
+ END LOOP;
261
+ END IF;
262
+ v_all_rows := v_all_rows || jsonb_build_array(to_jsonb(v_cells));
263
+ END LOOP;
264
+
265
+ sheet_order := v_sheet_order;
266
+ sub_service_id := NULLIF(v_sub.sub_service_id_num, 0);
267
+ sub_service_name := v_sub.sub_service_name;
268
+ sheet_name := sla_excel_sheet_name(v_sub.sub_service_name);
269
+ headers := v_headers_json;
270
+ rows := v_all_rows;
271
+ RETURN NEXT;
272
+ END LOOP;
273
+
274
+ -- ----- Per sub-service: Approvals (all steps for queue requests) -----
275
+ FOR v_sub IN
276
+ EXECUTE format($q$
277
+ SELECT DISTINCT
278
+ COALESCE(sr.sub_service_id, 0) AS sub_service_id_num,
279
+ COALESCE(NULLIF(trim(subsvc.sub_service_name), ''), NULLIF(trim(v.sub_service_name), ''), 'Uncategorized') AS sub_service_name
280
+ FROM vw_sla_my_requests v
281
+ INNER JOIN sla_requests sr ON sr.id = v.sla_request_id AND COALESCE(sr.is_deleted, false) = false
282
+ LEFT JOIN caa_sub_services subsvc ON subsvc.id = sr.sub_service_id
283
+ WHERE (%s)
284
+ AND (%s)
285
+ AND sr.status::TEXT = ANY($1)
286
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
287
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
288
+ AND ($4 IS NULL OR v.created_at::date >= $4)
289
+ AND ($5 IS NULL OR v.created_at::date <= $5)
290
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
291
+ ORDER BY sub_service_name ASC
292
+ $q$, p_own_requests_filter_sql, p_queue_exists_sql)
293
+ USING p_request_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
294
+ LOOP
295
+ v_sheet_order := v_sheet_order + 1;
296
+ v_sheet_label := sla_excel_approvals_sheet_name(v_sub.sub_service_name);
297
+ v_all_rows := '[]'::jsonb;
298
+
299
+ FOR v_approval IN
300
+ EXECUTE format($q$
301
+ SELECT
302
+ va.request_id,
303
+ sr.service_id,
304
+ va.service_name,
305
+ sr.sub_service_id,
306
+ va.sub_service_name,
307
+ va.sla_approval_id,
308
+ va.source_approval_id,
309
+ va.level,
310
+ va.approval_status,
311
+ va.request_status,
312
+ va.approval_role_id,
313
+ va.approval_role_name,
314
+ va.approval_department_id,
315
+ va.approval_department_name,
316
+ va.approval_section_id,
317
+ va.approval_section_name,
318
+ va.approver_user_id,
319
+ va.approver_user_name,
320
+ va.delegate_user_id,
321
+ va.delegate_user_name,
322
+ va.approved_by,
323
+ va.approved_by_name,
324
+ va.comment,
325
+ va.created_at,
326
+ va.updated_at
327
+ FROM vw_sla_approvals va
328
+ INNER JOIN sla_requests sr
329
+ ON sr.request_id = va.request_id AND COALESCE(sr.is_deleted, false) = false
330
+ INNER JOIN vw_sla_my_requests v ON v.sla_request_id = sr.id
331
+ WHERE (%s)
332
+ AND (%s)
333
+ AND sr.status::TEXT = ANY($1)
334
+ AND ($2 IS NULL OR cardinality($2) = 0 OR sr.service_id = ANY($2))
335
+ AND ($3 IS NULL OR cardinality($3) = 0 OR sr.sub_service_id = ANY($3))
336
+ AND ($4 IS NULL OR v.created_at::date >= $4)
337
+ AND ($5 IS NULL OR v.created_at::date <= $5)
338
+ AND ($6 IS NULL OR trim($6) = '' OR CAST(v.request_id AS TEXT) ILIKE '%%' || trim($6) || '%%')
339
+ AND COALESCE(sr.sub_service_id, 0) = %s
340
+ ORDER BY va.request_id ASC, va.level ASC NULLS LAST, va.sla_approval_id ASC
341
+ $q$, p_own_requests_filter_sql, p_queue_exists_sql, v_sub.sub_service_id_num)
342
+ USING p_request_statuses, p_service_ids, p_sub_service_ids, p_from_date, p_to_date, p_search_text
343
+ LOOP
344
+ v_cells := ARRAY[
345
+ COALESCE(v_approval.request_id::TEXT, ''),
346
+ COALESCE(v_approval.service_id::TEXT, ''),
347
+ COALESCE(v_approval.service_name, ''),
348
+ COALESCE(v_approval.sub_service_id::TEXT, ''),
349
+ COALESCE(v_approval.sub_service_name, ''),
350
+ COALESCE(v_approval.sla_approval_id::TEXT, ''),
351
+ COALESCE(v_approval.source_approval_id::TEXT, ''),
352
+ COALESCE(v_approval.level::TEXT, ''),
353
+ COALESCE(v_approval.approval_status, ''),
354
+ COALESCE(v_approval.request_status, ''),
355
+ COALESCE(v_approval.approval_role_id::TEXT, ''),
356
+ COALESCE(v_approval.approval_role_name, ''),
357
+ COALESCE(v_approval.approval_department_id::TEXT, ''),
358
+ COALESCE(v_approval.approval_department_name, ''),
359
+ COALESCE(v_approval.approval_section_id::TEXT, ''),
360
+ COALESCE(v_approval.approval_section_name, ''),
361
+ COALESCE(v_approval.approver_user_id::TEXT, ''),
362
+ COALESCE(v_approval.approver_user_name, ''),
363
+ COALESCE(v_approval.delegate_user_id::TEXT, ''),
364
+ COALESCE(v_approval.delegate_user_name, ''),
365
+ COALESCE(v_approval.approved_by::TEXT, ''),
366
+ COALESCE(v_approval.approved_by_name, ''),
367
+ COALESCE(v_approval.comment, ''),
368
+ COALESCE(to_char(v_approval.created_at AT TIME ZONE 'UTC', 'YYYY-MM-DD HH24:MI:SS'), ''),
369
+ COALESCE(to_char(v_approval.updated_at AT TIME ZONE 'UTC', 'YYYY-MM-DD HH24:MI:SS'), '')
370
+ ];
371
+ v_all_rows := v_all_rows || jsonb_build_array(to_jsonb(v_cells));
372
+ END LOOP;
373
+
374
+ sheet_order := v_sheet_order;
375
+ sub_service_id := NULLIF(v_sub.sub_service_id_num, 0);
376
+ sub_service_name := v_sub.sub_service_name;
377
+ sheet_name := v_sheet_label;
378
+ headers := to_jsonb(v_approval_headers);
379
+ rows := v_all_rows;
380
+ RETURN NEXT;
381
+ END LOOP;
382
+ END;
383
+ $$;