@platform-modules/foreign-ministry 1.3.322 → 1.3.326

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,8 +1,18 @@
1
1
  {
2
2
  "name": "@platform-modules/foreign-ministry",
3
- "version": "1.3.322",
3
+ "version": "1.3.326",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/index.d.ts",
6
+ "exports": {
7
+ ".": {
8
+ "types": "./dist/index.d.ts",
9
+ "default": "./dist/index.js"
10
+ },
11
+ "./sla-report-views": {
12
+ "types": "./dist/sla-report-views.d.ts",
13
+ "default": "./dist/sla-report-views.js"
14
+ }
15
+ },
6
16
  "scripts": {
7
17
  "build": "tsc",
8
18
  "dev": "ts-node src/scripts.ts",
@@ -1,38 +1,38 @@
1
- const dotenv = require('dotenv');
2
- const { Client } = require('pg');
3
-
4
- dotenv.config({ path: '../Reports_Service/.env' });
5
- dotenv.config({ path: '.env', override: true });
6
-
7
- async function main() {
8
- const client = new Client({
9
- host: process.env.DB_HOST.trim(),
10
- port: 5432,
11
- user: process.env.DB_USER.trim(),
12
- password: process.env.DB_PASS,
13
- database: process.env.DB_NAME.trim(),
14
- });
15
- await client.connect();
16
-
17
- const tables = ['sla_requests', 'users', 'departments', 'sections', 'fm_services', 'fm_sub_services'];
18
- for (const table of tables) {
19
- const { rows } = await client.query(
20
- `SELECT column_name, data_type
21
- FROM information_schema.columns
22
- WHERE table_schema = 'public' AND table_name = $1
23
- ORDER BY ordinal_position`,
24
- [table]
25
- );
26
- console.log(`\n${table}:`);
27
- for (const row of rows) {
28
- console.log(` ${row.column_name}: ${row.data_type}`);
29
- }
30
- }
31
-
32
- await client.end();
33
- }
34
-
35
- main().catch((err) => {
36
- console.error(err.message);
37
- process.exit(1);
38
- });
1
+ const dotenv = require('dotenv');
2
+ const { Client } = require('pg');
3
+
4
+ dotenv.config({ path: '../Reports_Service/.env' });
5
+ dotenv.config({ path: '.env', override: true });
6
+
7
+ async function main() {
8
+ const client = new Client({
9
+ host: process.env.DB_HOST.trim(),
10
+ port: 5432,
11
+ user: process.env.DB_USER.trim(),
12
+ password: process.env.DB_PASS,
13
+ database: process.env.DB_NAME.trim(),
14
+ });
15
+ await client.connect();
16
+
17
+ const tables = ['sla_requests', 'users', 'departments', 'sections', 'fm_services', 'fm_sub_services'];
18
+ for (const table of tables) {
19
+ const { rows } = await client.query(
20
+ `SELECT column_name, data_type
21
+ FROM information_schema.columns
22
+ WHERE table_schema = 'public' AND table_name = $1
23
+ ORDER BY ordinal_position`,
24
+ [table]
25
+ );
26
+ console.log(`\n${table}:`);
27
+ for (const row of rows) {
28
+ console.log(` ${row.column_name}: ${row.data_type}`);
29
+ }
30
+ }
31
+
32
+ await client.end();
33
+ }
34
+
35
+ main().catch((err) => {
36
+ console.error(err.message);
37
+ process.exit(1);
38
+ });
@@ -1,42 +1,42 @@
1
- const dotenv = require('dotenv');
2
- const path = require('path');
3
- const { Client } = require('pg');
4
-
5
- async function listTables(label, config) {
6
- const client = new Client(config);
7
- await client.connect();
8
- const { rows } = await client.query(`
9
- SELECT table_name
10
- FROM information_schema.tables
11
- WHERE table_schema = 'public'
12
- AND (table_name LIKE '%service%' OR table_name LIKE 'sla%')
13
- ORDER BY table_name
14
- `);
15
- console.log(`${label} (${config.database}):`, rows.map((r) => r.table_name).join(', ') || '(none)');
16
- await client.end();
17
- }
18
-
19
- async function main() {
20
- dotenv.config({ path: path.join(__dirname, '..', '.env') });
21
- await listTables('FM shared_models/.env', {
22
- host: process.env.DB_HOST.trim(),
23
- port: 5432,
24
- user: process.env.DB_USER.trim(),
25
- password: process.env.DB_PASS,
26
- database: process.env.DB_NAME.trim(),
27
- });
28
-
29
- dotenv.config({ path: path.join(__dirname, '..', '..', 'Reports_Service', '.env') });
30
- await listTables('Reports_Service/.env', {
31
- host: (process.env.TYPEORM_HOST || '').trim(),
32
- port: 5432,
33
- user: (process.env.TYPEORM_USERNAME || '').trim(),
34
- password: process.env.TYPEORM_PASSWORD,
35
- database: (process.env.TYPEORM_DATABASE || '').trim(),
36
- });
37
- }
38
-
39
- main().catch((err) => {
40
- console.error(err.message);
41
- process.exit(1);
42
- });
1
+ const dotenv = require('dotenv');
2
+ const path = require('path');
3
+ const { Client } = require('pg');
4
+
5
+ async function listTables(label, config) {
6
+ const client = new Client(config);
7
+ await client.connect();
8
+ const { rows } = await client.query(`
9
+ SELECT table_name
10
+ FROM information_schema.tables
11
+ WHERE table_schema = 'public'
12
+ AND (table_name LIKE '%service%' OR table_name LIKE 'sla%')
13
+ ORDER BY table_name
14
+ `);
15
+ console.log(`${label} (${config.database}):`, rows.map((r) => r.table_name).join(', ') || '(none)');
16
+ await client.end();
17
+ }
18
+
19
+ async function main() {
20
+ dotenv.config({ path: path.join(__dirname, '..', '.env') });
21
+ await listTables('FM shared_models/.env', {
22
+ host: process.env.DB_HOST.trim(),
23
+ port: 5432,
24
+ user: process.env.DB_USER.trim(),
25
+ password: process.env.DB_PASS,
26
+ database: process.env.DB_NAME.trim(),
27
+ });
28
+
29
+ dotenv.config({ path: path.join(__dirname, '..', '..', 'Reports_Service', '.env') });
30
+ await listTables('Reports_Service/.env', {
31
+ host: (process.env.TYPEORM_HOST || '').trim(),
32
+ port: 5432,
33
+ user: (process.env.TYPEORM_USERNAME || '').trim(),
34
+ password: process.env.TYPEORM_PASSWORD,
35
+ database: (process.env.TYPEORM_DATABASE || '').trim(),
36
+ });
37
+ }
38
+
39
+ main().catch((err) => {
40
+ console.error(err.message);
41
+ process.exit(1);
42
+ });
@@ -1,95 +1,95 @@
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
- // Reports_Service defaults first; shared_models/.env wins (FM DB for sync:sla-sql).
27
- if (fs.existsSync(reportsEnv)) {
28
- dotenv.config({ path: reportsEnv });
29
- }
30
- if (fs.existsSync(sharedEnv)) {
31
- dotenv.config({ path: sharedEnv, override: true });
32
- }
33
- }
34
-
35
- function getDbConfig() {
36
- const host = (process.env.TYPEORM_HOST || process.env.DB_HOST || '').trim();
37
- const port = parseInt(process.env.TYPEORM_PORT || process.env.DB_PORT || '5432', 10);
38
- const user = (process.env.TYPEORM_USERNAME || process.env.DB_USER || '').trim();
39
- const password = process.env.TYPEORM_PASSWORD || process.env.DB_PASS || '';
40
- const database = (process.env.TYPEORM_DATABASE || process.env.DB_NAME || '').trim();
41
-
42
- if (!host || !user || !database) {
43
- throw new Error(
44
- 'Missing DB config. Set TYPEORM_* in Reports_Service/.env or DB_* in shared_models/.env'
45
- );
46
- }
47
-
48
- return { host, port, user, password, database };
49
- }
50
-
51
- async function main() {
52
- const skipSync = process.argv.includes('--skip');
53
- if (skipSync || process.env.SKIP_SLA_SQL_SYNC === 'true') {
54
- console.log('[sync-sla-reports-sql] skipped (SKIP_SLA_SQL_SYNC or --skip)');
55
- return;
56
- }
57
-
58
- loadEnvFiles();
59
- const config = getDbConfig();
60
- const sqlDir = path.join(__dirname, '..', 'sql');
61
- const manifestPath = path.join(sqlDir, 'sla-reports-sync.manifest.json');
62
-
63
- if (!fs.existsSync(manifestPath)) {
64
- throw new Error(`Manifest not found: ${manifestPath}`);
65
- }
66
-
67
- const files = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
68
- if (!Array.isArray(files) || files.length === 0) {
69
- throw new Error('sla-reports-sync.manifest.json must be a non-empty array');
70
- }
71
-
72
- const client = new Client(config);
73
- await client.connect();
74
- console.log(`[sync-sla-reports-sql] connected to ${config.host}/${config.database}`);
75
-
76
- try {
77
- for (const file of files) {
78
- const filePath = path.join(sqlDir, file);
79
- if (!fs.existsSync(filePath)) {
80
- throw new Error(`SQL file missing: ${filePath}`);
81
- }
82
- const sql = fs.readFileSync(filePath, 'utf8');
83
- await client.query(sql);
84
- console.log(`[sync-sla-reports-sql] applied ${file}`);
85
- }
86
- console.log('[sync-sla-reports-sql] done');
87
- } finally {
88
- await client.end();
89
- }
90
- }
91
-
92
- main().catch((err) => {
93
- console.error('[sync-sla-reports-sql] failed:', err.message);
94
- process.exit(1);
95
- });
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
+ // Reports_Service defaults first; shared_models/.env wins (FM DB for sync:sla-sql).
27
+ if (fs.existsSync(reportsEnv)) {
28
+ dotenv.config({ path: reportsEnv });
29
+ }
30
+ if (fs.existsSync(sharedEnv)) {
31
+ dotenv.config({ path: sharedEnv, override: true });
32
+ }
33
+ }
34
+
35
+ function getDbConfig() {
36
+ const host = (process.env.TYPEORM_HOST || process.env.DB_HOST || '').trim();
37
+ const port = parseInt(process.env.TYPEORM_PORT || process.env.DB_PORT || '5432', 10);
38
+ const user = (process.env.TYPEORM_USERNAME || process.env.DB_USER || '').trim();
39
+ const password = process.env.TYPEORM_PASSWORD || process.env.DB_PASS || '';
40
+ const database = (process.env.TYPEORM_DATABASE || process.env.DB_NAME || '').trim();
41
+
42
+ if (!host || !user || !database) {
43
+ throw new Error(
44
+ 'Missing DB config. Set TYPEORM_* in Reports_Service/.env or DB_* in shared_models/.env'
45
+ );
46
+ }
47
+
48
+ return { host, port, user, password, database };
49
+ }
50
+
51
+ async function main() {
52
+ const skipSync = process.argv.includes('--skip');
53
+ if (skipSync || process.env.SKIP_SLA_SQL_SYNC === 'true') {
54
+ console.log('[sync-sla-reports-sql] skipped (SKIP_SLA_SQL_SYNC or --skip)');
55
+ return;
56
+ }
57
+
58
+ loadEnvFiles();
59
+ const config = getDbConfig();
60
+ const sqlDir = path.join(__dirname, '..', 'sql');
61
+ const manifestPath = path.join(sqlDir, 'sla-reports-sync.manifest.json');
62
+
63
+ if (!fs.existsSync(manifestPath)) {
64
+ throw new Error(`Manifest not found: ${manifestPath}`);
65
+ }
66
+
67
+ const files = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
68
+ if (!Array.isArray(files) || files.length === 0) {
69
+ throw new Error('sla-reports-sync.manifest.json must be a non-empty array');
70
+ }
71
+
72
+ const client = new Client(config);
73
+ await client.connect();
74
+ console.log(`[sync-sla-reports-sql] connected to ${config.host}/${config.database}`);
75
+
76
+ try {
77
+ for (const file of files) {
78
+ const filePath = path.join(sqlDir, file);
79
+ if (!fs.existsSync(filePath)) {
80
+ throw new Error(`SQL file missing: ${filePath}`);
81
+ }
82
+ const sql = fs.readFileSync(filePath, 'utf8');
83
+ await client.query(sql);
84
+ console.log(`[sync-sla-reports-sql] applied ${file}`);
85
+ }
86
+ console.log('[sync-sla-reports-sql] done');
87
+ } finally {
88
+ await client.end();
89
+ }
90
+ }
91
+
92
+ main().catch((err) => {
93
+ console.error('[sync-sla-reports-sql] failed:', err.message);
94
+ process.exit(1);
95
+ });
@@ -343,8 +343,6 @@ import { EmbassyEvaluationRequestAttachment } from './models/EmbassyEvaluationAt
343
343
  import { SlaRequest } from './models/SlaRequestModel';
344
344
  import { ServiceSlaApproval } from './models/ServiceSlaApprovalModel';
345
345
  import { SlaConfig } from './models/SlaConfigModel';
346
- import { SlaMyRequestsView } from './models/SlaMyRequestsViewModel';
347
- import { SlaApprovalsView } from './models/SlaApprovalsViewModel';
348
346
 
349
347
  export const AppDataSource = new DataSource({
350
348
  type: 'postgres',
@@ -684,7 +682,5 @@ export const AppDataSource = new DataSource({
684
682
  SlaRequest,
685
683
  ServiceSlaApproval,
686
684
  SlaConfig,
687
- SlaMyRequestsView,
688
- SlaApprovalsView,
689
685
  ],
690
686
  });
@@ -0,0 +1,46 @@
1
+ import type { DataSource } from 'typeorm';
2
+ import { Role } from '../models/role';
3
+ import { UserRole } from '../models/userRolesModel';
4
+
5
+ /** Parse authenticated user id from Fastify `request.meta`. */
6
+ export function parsePortalUserIdFromRequest(request: {
7
+ meta?: { userId?: string | number };
8
+ }): number | null {
9
+ const raw = request.meta?.userId;
10
+ if (raw == null || raw === '') return null;
11
+ const n = Number(raw);
12
+ return Number.isFinite(n) && n > 0 ? n : null;
13
+ }
14
+
15
+ /**
16
+ * True when the user has an active Admin role in `user_role` (role.name = admin, case-insensitive).
17
+ * Same check as Reports_Service `isReportsAdmin`.
18
+ */
19
+ export async function userHasPortalAdminRole(
20
+ dataSource: DataSource,
21
+ userId: number
22
+ ): Promise<boolean> {
23
+ if (!dataSource?.isInitialized || !Number.isFinite(userId) || userId <= 0) {
24
+ return false;
25
+ }
26
+
27
+ return dataSource
28
+ .getRepository(UserRole)
29
+ .createQueryBuilder('ur')
30
+ .innerJoin(Role, 'role', 'role.id = ur.role_id AND COALESCE(role.is_deleted, false) = false')
31
+ .where('ur.user_id = :userId', { userId })
32
+ .andWhere('ur.is_deleted = false')
33
+ .andWhere('ur.is_active = true')
34
+ .andWhere('LOWER(role.name) = :adminRole', { adminRole: 'admin' })
35
+ .getExists();
36
+ }
37
+
38
+ /** Convenience: resolve user id from request and check admin role. */
39
+ export async function isPortalAdminFromRequest(
40
+ dataSource: DataSource,
41
+ request: { meta?: { userId?: string | number } }
42
+ ): Promise<boolean> {
43
+ const userId = parsePortalUserIdFromRequest(request);
44
+ if (!userId) return false;
45
+ return userHasPortalAdminRole(dataSource, userId);
46
+ }