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

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.
package/package.json CHANGED
@@ -1,11 +1,12 @@
1
1
  {
2
2
  "name": "@platform-modules/civil-aviation-authority",
3
- "version": "2.3.264",
3
+ "version": "2.3.265",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/index.d.ts",
6
6
  "scripts": {
7
7
  "build": "tsc",
8
- "dev": "ts-node src/scripts.ts"
8
+ "dev": "ts-node src/scripts.ts",
9
+ "sync:sla-sql": "node scripts/sync-sla-reports-sql.js"
9
10
  },
10
11
  "publishConfig": {
11
12
  "access": "public"
@@ -0,0 +1,94 @@
1
+ /**
2
+ * Applies SLA report views + stored procedures to PostgreSQL (idempotent CREATE OR REPLACE).
3
+ *
4
+ * Env (shared_models/.env or Reports_Service/.env):
5
+ * DB_HOST / TYPEORM_HOST
6
+ * DB_PORT / TYPEORM_PORT
7
+ * DB_USER / TYPEORM_USERNAME
8
+ * DB_PASS / TYPEORM_PASSWORD
9
+ * DB_NAME / TYPEORM_DATABASE
10
+ *
11
+ * Usage:
12
+ * node scripts/sync-sla-reports-sql.js
13
+ * npm run sync:sla-sql (from shared_models)
14
+ */
15
+
16
+ const fs = require('fs');
17
+ const path = require('path');
18
+ const { Client } = require('pg');
19
+
20
+ function loadEnvFiles() {
21
+ const dotenv = require('dotenv');
22
+ const sharedRoot = path.resolve(__dirname, '..');
23
+ const reportsEnv = path.resolve(sharedRoot, '..', 'Reports_Service', '.env');
24
+ const sharedEnv = path.join(sharedRoot, '.env');
25
+
26
+ if (fs.existsSync(sharedEnv)) {
27
+ dotenv.config({ path: sharedEnv });
28
+ }
29
+ if (fs.existsSync(reportsEnv)) {
30
+ dotenv.config({ path: reportsEnv, override: true });
31
+ }
32
+ }
33
+
34
+ function getDbConfig() {
35
+ const host = (process.env.TYPEORM_HOST || process.env.DB_HOST || '').trim();
36
+ const port = parseInt(process.env.TYPEORM_PORT || process.env.DB_PORT || '5432', 10);
37
+ const user = (process.env.TYPEORM_USERNAME || process.env.DB_USER || '').trim();
38
+ const password = process.env.TYPEORM_PASSWORD || process.env.DB_PASS || '';
39
+ const database = (process.env.TYPEORM_DATABASE || process.env.DB_NAME || '').trim();
40
+
41
+ if (!host || !user || !database) {
42
+ throw new Error(
43
+ 'Missing DB config. Set TYPEORM_* in Reports_Service/.env or DB_* in shared_models/.env'
44
+ );
45
+ }
46
+
47
+ return { host, port, user, password, database };
48
+ }
49
+
50
+ async function main() {
51
+ const skipSync = process.argv.includes('--skip');
52
+ if (skipSync || process.env.SKIP_SLA_SQL_SYNC === 'true') {
53
+ console.log('[sync-sla-reports-sql] skipped (SKIP_SLA_SQL_SYNC or --skip)');
54
+ return;
55
+ }
56
+
57
+ loadEnvFiles();
58
+ const config = getDbConfig();
59
+ const sqlDir = path.join(__dirname, '..', 'sql');
60
+ const manifestPath = path.join(sqlDir, 'sla-reports-sync.manifest.json');
61
+
62
+ if (!fs.existsSync(manifestPath)) {
63
+ throw new Error(`Manifest not found: ${manifestPath}`);
64
+ }
65
+
66
+ const files = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
67
+ if (!Array.isArray(files) || files.length === 0) {
68
+ throw new Error('sla-reports-sync.manifest.json must be a non-empty array');
69
+ }
70
+
71
+ const client = new Client(config);
72
+ await client.connect();
73
+ console.log(`[sync-sla-reports-sql] connected to ${config.host}/${config.database}`);
74
+
75
+ try {
76
+ for (const file of files) {
77
+ const filePath = path.join(sqlDir, file);
78
+ if (!fs.existsSync(filePath)) {
79
+ throw new Error(`SQL file missing: ${filePath}`);
80
+ }
81
+ const sql = fs.readFileSync(filePath, 'utf8');
82
+ await client.query(sql);
83
+ console.log(`[sync-sla-reports-sql] applied ${file}`);
84
+ }
85
+ console.log('[sync-sla-reports-sql] done');
86
+ } finally {
87
+ await client.end();
88
+ }
89
+ }
90
+
91
+ main().catch((err) => {
92
+ console.error('[sync-sla-reports-sql] failed:', err.message);
93
+ process.exit(1);
94
+ });
package/sql/README.md ADDED
@@ -0,0 +1,25 @@
1
+ # SLA reports — database objects
2
+
3
+ Applied in order by `npm run sync:sla-sql` (see `sla-reports-sync.manifest.json`).
4
+
5
+ | File | Description |
6
+ |------|-------------|
7
+ | `vw_sla_my_requests.sql` | My-requests view |
8
+ | `vw_sla_approvals.sql` | Approvals view |
9
+ | `sla_reports_procedures.sql` | User SLA report functions + Excel helpers |
10
+ | `sla_reports_approvals_workbook.sql` | Approvals Excel workbook (Requests + Approvals per sub-service) |
11
+ | `sla_reports_admin_procedures.sql` | Admin SLA report functions (requires procedures file) |
12
+
13
+ From **shared_models**:
14
+
15
+ ```bash
16
+ npm run sync:sla-sql
17
+ ```
18
+
19
+ From **Reports_Service** (uses `Reports_Service/.env`, then `shared_models/.env`):
20
+
21
+ ```bash
22
+ npm run sync:sla-sql
23
+ ```
24
+
25
+ `npm run dev` in Reports_Service runs sync automatically before nodemon.
@@ -0,0 +1,7 @@
1
+ [
2
+ "vw_sla_my_requests.sql",
3
+ "vw_sla_approvals.sql",
4
+ "sla_reports_procedures.sql",
5
+ "sla_reports_approvals_workbook.sql",
6
+ "sla_reports_admin_procedures.sql"
7
+ ]
@@ -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
+ $$;