@opengis/admin 0.1.58 → 0.1.59

Sign up to get free protection for your applications and to get access to all the features.
Files changed (111) hide show
  1. package/README.md +29 -29
  2. package/config.js +4 -4
  3. package/dist/{add-page-d6TCPawD.js → add-page-BOXd-uV_.js} +1 -1
  4. package/dist/{admin-interface-w6YuIWM3.js → admin-interface-DbfKIAnw.js} +211 -207
  5. package/dist/{admin-view-3Sz3UF_y.js → admin-view-DDDd_P1v.js} +2 -2
  6. package/dist/admin.js +1 -1
  7. package/dist/admin.umd.cjs +12 -12
  8. package/dist/{card-page-BsQts5XS.js → card-page-Bq7fUm5C.js} +2 -2
  9. package/dist/{card-view-UYkkl_Pu.js → card-view-NLu0N8TA.js} +1 -1
  10. package/dist/{edit-page-CtKHElG1.js → edit-page-z7ozrw2C.js} +1 -1
  11. package/dist/{import-file-CsCQ5xvs.js → import-file-BBA7riUK.js} +13 -20
  12. package/dist/style.css +1 -1
  13. package/module/settings/card/admin.roles.table/access.hbs +27 -27
  14. package/module/settings/card/admin.roles.table/general_info.hbs +16 -16
  15. package/module/settings/card/admin.roles.table/index.yml +14 -14
  16. package/module/settings/card/admin.roles.table/users.hbs +27 -27
  17. package/module/settings/card/admin.routes.table/general_info.hbs +40 -40
  18. package/module/settings/card/admin.routes.table/index.yml +8 -8
  19. package/module/settings/card/admin.routes.table/users.hbs +33 -33
  20. package/module/settings/card/admin.users.table/general_info.hbs +25 -25
  21. package/module/settings/card/admin.users.table/index.yml +12 -12
  22. package/module/settings/card/admin.users.table/logs.hbs +30 -30
  23. package/module/settings/card/admin.users.table/user_roles.hbs +24 -24
  24. package/module/settings/cls/core.actions.json +13 -13
  25. package/module/settings/cls/core.scope.json +13 -13
  26. package/module/settings/cls/properties.site_status.json +13 -13
  27. package/module/settings/cls/properties.widget_status.json +13 -13
  28. package/module/settings/cls/users.user_type.json +13 -13
  29. package/module/settings/form/admin.access.form.json +36 -36
  30. package/module/settings/form/admin.custom_column.form.json +71 -71
  31. package/module/settings/form/admin.properties.form.json +15 -15
  32. package/module/settings/form/admin.roles.form.json +19 -19
  33. package/module/settings/form/admin.routes.form.json +25 -25
  34. package/module/settings/form/admin.user_properties.form.json +15 -15
  35. package/module/settings/form/admin.user_roles.form.json +21 -21
  36. package/module/settings/form/admin.users.form.json +150 -150
  37. package/module/settings/form/user.user_roles.form.json +13 -13
  38. package/module/settings/interface/admin.properties.json +4 -4
  39. package/module/settings/interface/admin.roles.json +4 -4
  40. package/module/settings/interface/admin.routes.json +4 -4
  41. package/module/settings/interface/admin.users.json +4 -4
  42. package/module/settings/menu.json +50 -50
  43. package/module/settings/select/core.roles.json +2 -2
  44. package/module/settings/select/core.routes.sql +1 -1
  45. package/module/settings/select/core.user_uid.sql +1 -1
  46. package/module/settings/table/admin.access.table.json +77 -77
  47. package/module/settings/table/admin.custom_column.table.json +94 -94
  48. package/module/settings/table/admin.properties.table.json +33 -33
  49. package/module/settings/table/admin.roles.table.json +58 -58
  50. package/module/settings/table/admin.routes.table.json +67 -67
  51. package/module/settings/table/admin.user_properties.table.json +28 -28
  52. package/module/settings/table/admin.user_roles.table.json +66 -66
  53. package/module/settings/table/admin.users.table.json +119 -119
  54. package/package.json +79 -81
  55. package/plugin.js +164 -162
  56. package/server/helpers/controllers/badge.js +11 -11
  57. package/server/helpers/controllers/hb.js +2 -2
  58. package/server/helpers/controllers/map.js +2 -2
  59. package/server/helpers/controllers/mls.js +2 -2
  60. package/server/helpers/controllers/vue.js +2 -2
  61. package/server/helpers/index.mjs +13 -13
  62. package/server/plugins/docs.js +28 -28
  63. package/server/plugins/hook.js +228 -228
  64. package/server/plugins/vite.js +69 -69
  65. package/server/routes/calendar/controllers/calendar.data.js +88 -88
  66. package/server/routes/calendar/index.mjs +17 -17
  67. package/server/routes/data/controllers/cardData.js +56 -56
  68. package/server/routes/data/controllers/cardTabData.js +49 -49
  69. package/server/routes/data/controllers/funcs/getFilterSQL/index.js +92 -92
  70. package/server/routes/data/controllers/funcs/getFilterSQL/util/formatValue.js +170 -170
  71. package/server/routes/data/controllers/funcs/getFilterSQL/util/getCustomQuery.js +13 -13
  72. package/server/routes/data/controllers/funcs/getFilterSQL/util/getFilterQuery.js +64 -64
  73. package/server/routes/data/controllers/funcs/getFilterSQL/util/getOptimizedQuery.js +12 -12
  74. package/server/routes/data/controllers/funcs/getFilterSQL/util/getTableSql.js +34 -34
  75. package/server/routes/data/controllers/tableData.js +145 -145
  76. package/server/routes/data/controllers/tableDataId.js +27 -27
  77. package/server/routes/data/controllers/tableFilter.js +63 -63
  78. package/server/routes/data/controllers/utils/assignTokens.js +30 -30
  79. package/server/routes/data/controllers/utils/getColumns.js +21 -21
  80. package/server/routes/data/index.mjs +15 -15
  81. package/server/routes/data/schema.js +7 -7
  82. package/server/routes/menu/controllers/getMenu.js +34 -34
  83. package/server/routes/menu/index.mjs +5 -5
  84. package/server/routes/notifications/controllers/readNotifications.js +30 -30
  85. package/server/routes/notifications/controllers/userNotifications.js +64 -64
  86. package/server/routes/notifications/hook/onWidgetSet.js +63 -63
  87. package/server/routes/notifications/index.mjs +40 -40
  88. package/server/routes/properties/controllers/admin.properties.get.js +29 -29
  89. package/server/routes/properties/controllers/user.properties.get.js +34 -34
  90. package/server/routes/properties/controllers/user.properties.post.js +30 -30
  91. package/server/routes/properties/funcs/getSettings.js +56 -56
  92. package/server/routes/properties/funcs/setSettings.js +44 -44
  93. package/server/routes/properties/funcs/utils/dataInsert.js +26 -26
  94. package/server/routes/properties/index.mjs +26 -26
  95. package/server/routes/root.mjs +3 -3
  96. package/server/routes/templates/controllers/getTemplate.js +16 -16
  97. package/server/routes/templates/index.mjs +14 -14
  98. package/server/templates/cls/itree.recrzone_category.json +73 -73
  99. package/server/templates/cls/test.json +9 -9
  100. package/server/templates/form/admin.user_cls.data.form.json +49 -49
  101. package/server/templates/form/admin.user_group_rel.form.json +21 -21
  102. package/server/templates/form/cp_building.form.json +32 -32
  103. package/server/templates/form/form-user-pass.json +10 -10
  104. package/server/templates/form/form-user_group.json +39 -39
  105. package/server/templates/form/form-users.json +156 -156
  106. package/server/templates/form/user_group_access.form.json +22 -22
  107. package/server/templates/select/account_id.json +2 -2
  108. package/server/templates/table/gis.dataset.table.json +43 -43
  109. package/server/templates/table/management.user_group.table.json +112 -112
  110. package/server/templates/table/management.users.table.json +126 -126
  111. package/utils.js +21 -21
@@ -1,228 +1,228 @@
1
- import fp from 'fastify-plugin';
2
- import fs from 'node:fs';
3
- import path from 'node:path';
4
-
5
- import { addHook, getTemplatePath, getTemplate, pgClients } from '@opengis/fastify-table/utils.js';
6
-
7
- import getMenu from '../routes/menu/controllers/getMenu.js';
8
-
9
- import config from '../../config.js';
10
-
11
- // to export the decorators to the outer scope
12
-
13
- async function plugin(fastify) {
14
- fastify.decorate('config', config);
15
-
16
- addHook('afterTemplate', async ({ req, data = {} }) => {
17
- const { uid } = req.session.passport?.user || {};
18
- if (!uid || !data || req.params?.type !== 'form' || !req.params?.name) return null;
19
- const { pg, params = {} } = req;
20
- const { rows: properties = [] } = await pg.query(`select name, title, format, data from admin.custom_column
21
- where entity=$1 and uid=$2`, [params.name?.replace('.form', '.table'), uid]);
22
- await Promise.all(properties.map(async (el) => {
23
- const clsData = el.data ? await getTemplate(['cls', 'select'], el.data) : undefined;
24
- const type = clsData ? 'Select' : ({ date: 'DatePicker' }[el.format] || 'Text');
25
- Object.assign(data?.schema || data || {}, { [el.name]: { type, ua: el.title, data: el.data, options: type === 'Select' && Array.isArray(clsData) && clsData?.length ? clsData : undefined, extra: 1 } });
26
- }));
27
- });
28
-
29
- addHook('afterUpdate', async ({ req, res = {} }) => {
30
- const {
31
- pg, session = {}, params = {}, body = {},
32
- } = req;
33
- const { uid } = session.passport?.user || {};
34
- if (!uid) return null;
35
-
36
- const loadTable = await getTemplate('table', params.table);
37
- if (!pg.pk[loadTable?.table || params.table]) return null;
38
- const pk = pg.pk[loadTable?.table || params.table];
39
- const id = res[pk];
40
-
41
- const { rows: properties = [] } = await pg.query(`select column_id, name, title, format, data from admin.custom_column
42
- where entity=$1 and uid=$2`, [params.table, uid]);
43
-
44
- if (!id || !properties?.length) return null;
45
-
46
- const q = `delete from crm.extra_data where property_entity='${params.table}' and object_id='${id}';${properties
47
- .filter((el) => Object.keys(body).includes(el.name))
48
- .map((el) => `insert into crm.extra_data(property_id,property_key,property_entity,object_id,${el.format?.toLowerCase() === 'date' ? 'value_date' : 'value_text'})
49
- select '${el.column_id}', '${el.name}', '${params.table}', '${id}', ${el.format?.toLowerCase() === 'date' ? `'${body[el.name]}'::timestamp without time zone` : `'${body[el.name]}'::text`}`)
50
- .join(';\n') || ''}`;
51
- return pg.query(q);
52
- });
53
-
54
- addHook('afterInsert', async ({ req, res = {} }) => {
55
- const {
56
- pg, session = {}, params = {}, body = {},
57
- } = req;
58
- const { uid } = session.passport?.user || {};
59
- if (!uid) return null;
60
-
61
- const loadTable = await getTemplate('table', params.table);
62
- if (!pg.pk[loadTable?.table || params.table]) return null;
63
- const pk = pg.pk[loadTable?.table || params.table];
64
- const id = res.rows?.[0]?.[pk];
65
-
66
- const { rows: properties = [] } = await pg.query(`select column_id, name, title, format, data from admin.custom_column
67
- where entity=$1 and uid=$2`, [params.table, uid]);
68
-
69
- if (!id || !properties?.length) return null;
70
-
71
- const q = properties
72
- .filter((el) => Object.keys(body).includes(el.name))
73
- .map((el) => `insert into crm.extra_data(property_id,property_key,property_entity,object_id,${el.format?.toLowerCase() === 'date' ? 'value_date' : 'value_text'})
74
- select '${el.column_id}', '${el.name}', '${params.table}', '${id}', ${el.format?.toLowerCase() === 'date' ? `'${body[el.name]}'::timestamp without time zone` : `'${body[el.name]}'::text`}`)
75
- .join(';\n');
76
- return pg.query(q);
77
- });
78
-
79
- fastify.addHook('onListen', async () => {
80
- const { client } = pgClients;
81
- const json = await getMenu();
82
- // insert interface list to db (user access management)
83
- if (client?.pk?.['admin.routes'] && json?.length) {
84
- const menuList = json.filter((el) => el?.menu?.length && el?.ua || el?.en || el?.name);
85
- const interfaces = menuList.reduce((acc, curr) => { curr.menu.forEach((el) => acc.push(el.path)); return acc; }, []);
86
- await client.query('update admin.routes set enabled=false where not array[route_id] <@ $1::text[]', [interfaces]);
87
-
88
- const q = `insert into admin.menu(name, ord) values${menuList.map((el, i) => `('${(el?.ua || el?.en || el?.name).replace(/'/g, '’')}', ${i}) `).join(',')
89
- } on conflict (name) do update set ord=excluded.ord, enabled=true returning name, menu_id`;
90
- const { rows = [] } = await client.query(q);
91
- await client.query('update admin.menu set enabled=false where not array[menu_id] <@ $1::text[]', [rows.map((el) => el.menu_id)]);
92
-
93
- const menus = rows.reduce((acc, curr) => Object.assign(acc, { [curr.menu_id]: menuList.find((item) => (item?.ua || item?.en || item?.name) === curr.name) }), {});
94
- const values = Object.entries(menus).reduce((acc, curr) => { curr[1]?.menu?.forEach((el) => acc.push({ ...el, menuId: curr[0] })); return acc; }, []);
95
-
96
- await Promise.all(values.filter((el) => el?.table).map(async (el) => Object.assign(el, { table: (await getTemplate('table', el.table))?.table || el.table })));
97
-
98
- const q1 = `insert into admin.routes(route_id, alias, title, menu_id, table_name) values ${values.map((el) => `('${el.path}', '${el.path}', '${el.title}', '${el.menuId}', '${el.table}')`).join(',')}
99
- on conflict (route_id) do update set menu_id=excluded.menu_id, title=excluded.title, enabled=true,
100
- table_name=excluded.table_name returning route_id, table_name`;
101
- try {
102
- const { rowCount } = await client.query(q1);
103
- // console.log('interface insert ok', values, rowCount);
104
- } catch (err) {
105
- console.log('interface insert error', values, q1, err);
106
- }
107
- }
108
- });
109
-
110
- fastify.addHook('onListen', async () => {
111
- const { client: pg } = pgClients;
112
- const clsQuery = [];
113
- if (!pg.pk?.['admin.cls']) return;
114
-
115
- const selectList = await getTemplatePath('select');
116
- const clsList = (await getTemplatePath('cls'))?.filter((el) => !(selectList?.map((el) => el?.[0]) || []).includes(el[0]));
117
- const cls = (selectList || []).concat(clsList || [])
118
- ?.map((el) => ({ name: el[0], module: path.basename(path.dirname(path.dirname(el[1]))), type: { 'json': 'cls', 'sql': 'select' }[el[2]] }))
119
- if (!cls?.length) return;
120
-
121
- const dupes = cls.filter((el, idx, arr) => arr.map((item) => item.name).indexOf(el.name) !== idx);
122
- // console.log('cls insert skip dupes', dupes.map((el) => el.name));
123
-
124
- try {
125
- await Promise.all(cls.filter((el, idx, arr) => arr.map((item) => item.name).indexOf(el.name) === idx).map(async (el) => {
126
- const { name, module, type } = el;
127
- const loadTemplate = await getTemplate(type, name);
128
- //console.log(name, type);
129
- if (type === 'select') {
130
- clsQuery.push(`insert into admin.cls(name,type,data,module) values('${name}','sql','${(loadTemplate?.sql || loadTemplate)?.replace(/'/g, "''")}', '${module?.replace(/'/g, "''")}')`);
131
- } else if (type === 'cls' && loadTemplate?.length) {
132
- clsQuery.push(`insert into admin.cls(name,type, module) values('${name}','json', '${module?.replace(/'/g, "''")}');
133
- insert into admin.cls(code,name,parent,icon)
134
- select value->>'id',value->>'text','${name}',value->>'icon'
135
- from json_array_elements('${JSON.stringify(loadTemplate).replace(/'/g, "''")}'::json)`);
136
- } else {
137
- console.log(name, type, 'empty');
138
- }
139
- }));
140
-
141
- await pg.query('truncate admin.cls');
142
- if (clsQuery.filter((el) => el).length) {
143
- await pg.query(clsQuery.filter((el) => el).join(';'));
144
- console.log('cls insert ok', clsQuery?.length);
145
- }
146
- } catch (err) {
147
- console.error('cls insert error', err.toString());
148
- }
149
- });
150
-
151
- // pre Request
152
- fastify.addHook('onRequest', async (req) => {
153
- req.funcs = fastify;
154
- const { user } = req.session?.passport || {};
155
- req.user = user;
156
- });
157
-
158
- // preSerialization
159
- fastify.addHook('preSerialization', async (req, reply, payload) => {
160
- if (req.url.includes('/suggest/') && !req.query.json) {
161
- return payload?.data;
162
- }
163
- if (payload?.redirect) {
164
- return reply.redirect(payload.redirect);
165
- }
166
- if (reply.sent) {
167
- return null;
168
- }
169
-
170
- if (['200', '400', '500', '403', '404'].includes(payload?.status)) {
171
- reply.status(payload.status);
172
- }
173
- /* if (payload.headers) {
174
- reply.headers(payload.headers);
175
- } */
176
- if (payload?.buffer) {
177
- return payload.buffer;
178
- }
179
- if (payload?.file) {
180
- // const buffer = await readFile(payload.file);
181
- // return reply.send(buffer);
182
- const stream = fs.createReadStream(payload.file);
183
- return stream;
184
- // return reply.send(stream);
185
- }
186
-
187
- if (payload?.message) {
188
- return payload.message;
189
- }
190
- return payload;
191
- });
192
-
193
- // preValidation
194
- fastify.addHook('preValidation', async (req) => {
195
- const parseRawBody = ['POST', 'PUT'].includes(req.method) && req.body && typeof req.body === 'string'
196
- && req.body.trim(/\r\n/g).startsWith('{')
197
- && req.body.trim(/\r\n/g).endsWith('}');
198
- if (parseRawBody) {
199
- try {
200
- req.body = JSON.parse(req.body || '{}');
201
- }
202
- catch (err) {
203
- // throw new Error('invalid body');
204
- // return { error: 'invalid body', status: 400 };
205
- }
206
- }
207
- });
208
-
209
- // allow upload file
210
- const kIsMultipart = Symbol.for('[FastifyMultipart.isMultipart]');
211
- fastify.addContentTypeParser('multipart', (request, _, done) => {
212
- request[kIsMultipart] = true;
213
- done(null);
214
- });
215
-
216
- // parse Body
217
- function contentParser(req, body, done) {
218
- const parseBody = decodeURIComponent(body.toString()).split('&').reduce((acc, el) => {
219
- const [key, val] = el.split('=');
220
- return { ...acc, [key]: val };
221
- }, {});
222
- done(null, parseBody);
223
- }
224
-
225
- fastify.addContentTypeParser('application/x-www-form-urlencoded', { parseAs: 'buffer' }, contentParser);
226
- }
227
-
228
- export default fp(plugin);
1
+ import fp from 'fastify-plugin';
2
+ import fs from 'node:fs';
3
+ import path from 'node:path';
4
+
5
+ import { addHook, getTemplatePath, getTemplate, pgClients } from '@opengis/fastify-table/utils.js';
6
+
7
+ import getMenu from '../routes/menu/controllers/getMenu.js';
8
+
9
+ import config from '../../config.js';
10
+
11
+ // to export the decorators to the outer scope
12
+
13
+ async function plugin(fastify) {
14
+ fastify.decorate('config', config);
15
+
16
+ addHook('afterTemplate', async ({ req, data = {} }) => {
17
+ const { uid } = req.session?.passport?.user || {};
18
+ if (!uid || !data || req.params?.type !== 'form' || !req.params?.name) return null;
19
+ const { pg, params = {} } = req;
20
+ const { rows: properties = [] } = await pg.query(`select name, title, format, data from admin.custom_column
21
+ where entity=$1 and uid=$2`, [params.name?.replace('.form', '.table'), uid]);
22
+ await Promise.all(properties.map(async (el) => {
23
+ const clsData = el.data ? await getTemplate(['cls', 'select'], el.data) : undefined;
24
+ const type = clsData ? 'Select' : ({ date: 'DatePicker' }[el.format] || 'Text');
25
+ Object.assign(data?.schema || data || {}, { [el.name]: { type, ua: el.title, data: el.data, options: type === 'Select' && Array.isArray(clsData) && clsData?.length ? clsData : undefined, extra: 1 } });
26
+ }));
27
+ });
28
+
29
+ addHook('afterUpdate', async ({ req, res = {} }) => {
30
+ const {
31
+ pg, session = {}, params = {}, body = {},
32
+ } = req;
33
+ const { uid } = session?.passport?.user || {};
34
+ if (!uid) return null;
35
+
36
+ const loadTable = await getTemplate('table', params.table);
37
+ if (!pg.pk[loadTable?.table || params.table]) return null;
38
+ const pk = pg.pk[loadTable?.table || params.table];
39
+ const id = res[pk];
40
+
41
+ const { rows: properties = [] } = await pg.query(`select column_id, name, title, format, data from admin.custom_column
42
+ where entity=$1 and uid=$2`, [params.table, uid]);
43
+
44
+ if (!id || !properties?.length) return null;
45
+
46
+ const q = `delete from crm.extra_data where property_entity='${params.table}' and object_id='${id}';${properties
47
+ .filter((el) => Object.keys(body).includes(el.name))
48
+ .map((el) => `insert into crm.extra_data(property_id,property_key,property_entity,object_id,${el.format?.toLowerCase() === 'date' ? 'value_date' : 'value_text'})
49
+ select '${el.column_id}', '${el.name}', '${params.table}', '${id}', ${el.format?.toLowerCase() === 'date' ? `'${body[el.name]}'::timestamp without time zone` : `'${body[el.name]}'::text`}`)
50
+ .join(';\n') || ''}`;
51
+ return pg.query(q);
52
+ });
53
+
54
+ addHook('afterInsert', async ({ req, res = {} }) => {
55
+ const {
56
+ pg, session = {}, params = {}, body = {},
57
+ } = req;
58
+ const { uid } = session?.passport?.user || {};
59
+ if (!uid) return null;
60
+
61
+ const loadTable = await getTemplate('table', params.table);
62
+ if (!pg.pk[loadTable?.table || params.table]) return null;
63
+ const pk = pg.pk[loadTable?.table || params.table];
64
+ const id = res.rows?.[0]?.[pk];
65
+
66
+ const { rows: properties = [] } = await pg.query(`select column_id, name, title, format, data from admin.custom_column
67
+ where entity=$1 and uid=$2`, [params.table, uid]);
68
+
69
+ if (!id || !properties?.length) return null;
70
+
71
+ const q = properties
72
+ .filter((el) => Object.keys(body).includes(el.name))
73
+ .map((el) => `insert into crm.extra_data(property_id,property_key,property_entity,object_id,${el.format?.toLowerCase() === 'date' ? 'value_date' : 'value_text'})
74
+ select '${el.column_id}', '${el.name}', '${params.table}', '${id}', ${el.format?.toLowerCase() === 'date' ? `'${body[el.name]}'::timestamp without time zone` : `'${body[el.name]}'::text`}`)
75
+ .join(';\n');
76
+ return pg.query(q);
77
+ });
78
+
79
+ fastify.addHook('onListen', async () => {
80
+ const { client } = pgClients;
81
+ const json = await getMenu();
82
+ // insert interface list to db (user access management)
83
+ if (client?.pk?.['admin.routes'] && json?.length) {
84
+ const menuList = json.filter((el) => el?.menu?.length && el?.ua || el?.en || el?.name);
85
+ const interfaces = menuList.reduce((acc, curr) => { curr.menu.forEach((el) => acc.push(el.path)); return acc; }, []);
86
+ await client.query('update admin.routes set enabled=false where not array[route_id] <@ $1::text[]', [interfaces]);
87
+
88
+ const q = `insert into admin.menu(name, ord) values${menuList.map((el, i) => `('${(el?.ua || el?.en || el?.name).replace(/'/g, '’')}', ${i}) `).join(',')
89
+ } on conflict (name) do update set ord=excluded.ord, enabled=true returning name, menu_id`;
90
+ const { rows = [] } = await client.query(q);
91
+ await client.query('update admin.menu set enabled=false where not array[menu_id] <@ $1::text[]', [rows.map((el) => el.menu_id)]);
92
+
93
+ const menus = rows.reduce((acc, curr) => Object.assign(acc, { [curr.menu_id]: menuList.find((item) => (item?.ua || item?.en || item?.name) === curr.name) }), {});
94
+ const values = Object.entries(menus).reduce((acc, curr) => { curr[1]?.menu?.forEach((el) => acc.push({ ...el, menuId: curr[0] })); return acc; }, []);
95
+
96
+ await Promise.all(values.filter((el) => el?.table).map(async (el) => Object.assign(el, { table: (await getTemplate('table', el.table))?.table || el.table })));
97
+
98
+ const q1 = `insert into admin.routes(route_id, alias, title, menu_id, table_name) values ${values.map((el) => `('${el.path}', '${el.path}', '${el.title}', '${el.menuId}', '${el.table}')`).join(',')}
99
+ on conflict (route_id) do update set menu_id=excluded.menu_id, title=excluded.title, enabled=true,
100
+ table_name=excluded.table_name returning route_id, table_name`;
101
+ try {
102
+ const { rowCount } = await client.query(q1);
103
+ // console.log('interface insert ok', values, rowCount);
104
+ } catch (err) {
105
+ console.log('interface insert error', values, q1, err);
106
+ }
107
+ }
108
+ });
109
+
110
+ fastify.addHook('onListen', async () => {
111
+ const { client: pg } = pgClients;
112
+ const clsQuery = [];
113
+ if (!pg.pk?.['admin.cls']) return;
114
+
115
+ const selectList = await getTemplatePath('select');
116
+ const clsList = (await getTemplatePath('cls'))?.filter((el) => !(selectList?.map((el) => el?.[0]) || []).includes(el[0]));
117
+ const cls = (selectList || []).concat(clsList || [])
118
+ ?.map((el) => ({ name: el[0], module: path.basename(path.dirname(path.dirname(el[1]))), type: { 'json': 'cls', 'sql': 'select' }[el[2]] }))
119
+ if (!cls?.length) return;
120
+
121
+ const dupes = cls.filter((el, idx, arr) => arr.map((item) => item.name).indexOf(el.name) !== idx);
122
+ // console.log('cls insert skip dupes', dupes.map((el) => el.name));
123
+
124
+ try {
125
+ await Promise.all(cls.filter((el, idx, arr) => arr.map((item) => item.name).indexOf(el.name) === idx).map(async (el) => {
126
+ const { name, module, type } = el;
127
+ const loadTemplate = await getTemplate(type, name);
128
+ //console.log(name, type);
129
+ if (type === 'select') {
130
+ clsQuery.push(`insert into admin.cls(name,type,data,module) values('${name}','sql','${(loadTemplate?.sql || loadTemplate)?.replace(/'/g, "''")}', '${module?.replace(/'/g, "''")}')`);
131
+ } else if (type === 'cls' && loadTemplate?.length) {
132
+ clsQuery.push(`insert into admin.cls(name,type, module) values('${name}','json', '${module?.replace(/'/g, "''")}');
133
+ insert into admin.cls(code,name,parent,icon)
134
+ select value->>'id',value->>'text','${name}',value->>'icon'
135
+ from json_array_elements('${JSON.stringify(loadTemplate).replace(/'/g, "''")}'::json)`);
136
+ } else {
137
+ console.log(name, type, 'empty');
138
+ }
139
+ }));
140
+
141
+ await pg.query('truncate admin.cls');
142
+ if (clsQuery.filter((el) => el).length) {
143
+ await pg.query(clsQuery.filter((el) => el).join(';'));
144
+ console.log('cls insert ok', clsQuery?.length);
145
+ }
146
+ } catch (err) {
147
+ console.error('cls insert error', err.toString());
148
+ }
149
+ });
150
+
151
+ // pre Request
152
+ fastify.addHook('onRequest', async (req) => {
153
+ req.funcs = fastify;
154
+ const { user } = req.session?.passport || {};
155
+ req.user = user;
156
+ });
157
+
158
+ // preSerialization
159
+ fastify.addHook('preSerialization', async (req, reply, payload) => {
160
+ if (req.url.includes('/suggest/') && !req.query.json) {
161
+ return payload?.data;
162
+ }
163
+ if (payload?.redirect) {
164
+ return reply.redirect(payload.redirect);
165
+ }
166
+ if (reply.sent) {
167
+ return null;
168
+ }
169
+
170
+ if (['200', '400', '500', '403', '404'].includes(payload?.status)) {
171
+ reply.status(payload.status);
172
+ }
173
+ /* if (payload.headers) {
174
+ reply.headers(payload.headers);
175
+ } */
176
+ if (payload?.buffer) {
177
+ return payload.buffer;
178
+ }
179
+ if (payload?.file) {
180
+ // const buffer = await readFile(payload.file);
181
+ // return reply.send(buffer);
182
+ const stream = fs.createReadStream(payload.file);
183
+ return stream;
184
+ // return reply.send(stream);
185
+ }
186
+
187
+ if (payload?.message) {
188
+ return payload.message;
189
+ }
190
+ return payload;
191
+ });
192
+
193
+ // preValidation
194
+ fastify.addHook('preValidation', async (req) => {
195
+ const parseRawBody = ['POST', 'PUT'].includes(req.method) && req.body && typeof req.body === 'string'
196
+ && req.body.trim(/\r\n/g).startsWith('{')
197
+ && req.body.trim(/\r\n/g).endsWith('}');
198
+ if (parseRawBody) {
199
+ try {
200
+ req.body = JSON.parse(req.body || '{}');
201
+ }
202
+ catch (err) {
203
+ // throw new Error('invalid body');
204
+ // return { error: 'invalid body', status: 400 };
205
+ }
206
+ }
207
+ });
208
+
209
+ // allow upload file
210
+ const kIsMultipart = Symbol.for('[FastifyMultipart.isMultipart]');
211
+ fastify.addContentTypeParser('multipart', (request, _, done) => {
212
+ request[kIsMultipart] = true;
213
+ done(null);
214
+ });
215
+
216
+ // parse Body
217
+ function contentParser(req, body, done) {
218
+ const parseBody = decodeURIComponent(body.toString()).split('&').reduce((acc, el) => {
219
+ const [key, val] = el.split('=');
220
+ return { ...acc, [key]: val };
221
+ }, {});
222
+ done(null, parseBody);
223
+ }
224
+
225
+ fastify.addContentTypeParser('application/x-www-form-urlencoded', { parseAs: 'buffer' }, contentParser);
226
+ }
227
+
228
+ export default fp(plugin);
@@ -1,69 +1,69 @@
1
- import fs from 'fs';
2
- import path, { dirname } from 'path';
3
- import { fileURLToPath } from 'url';
4
-
5
- const dir = dirname(fileURLToPath(import.meta.url));
6
- const root = `${dir}/../..`;
7
-
8
- const isProduction = process.env.NODE_ENV === 'production';
9
-
10
- async function plugin(fastify) {
11
- // vite server
12
- if (!isProduction) {
13
- const vite = await import('vite');
14
-
15
- const viteServer = await vite.createServer({
16
- server: {
17
- middlewareMode: true,
18
- },
19
- });
20
- // hot reload
21
- viteServer.watcher.on('all', function (d, t) {
22
- if (!t.includes('module')) return;
23
- console.log(d, t);
24
- viteServer.ws.send({ type: 'full-reload' });
25
- });
26
-
27
- // this is middleware for vite's dev servert
28
- fastify.addHook('onRequest', async (req, reply) => {
29
- const { user } = req.session?.passport || {};
30
- const { disableAuth } = req.funcs?.config || {};
31
- if (!user && !disableAuth) {
32
- // return reply.send('access restricted');
33
- }
34
-
35
- const next = () => new Promise((resolve) => {
36
- viteServer.middlewares(req.raw, reply.raw, () => resolve());
37
- });
38
- await next();
39
- });
40
- fastify.get('*', async () => { });
41
- return;
42
- }
43
-
44
- // From Build
45
- fastify.get('*', async (req, reply) => {
46
- const { user } = req.session?.passport || {};
47
-
48
- // if (!user) return reply.redirect('/login');
49
- const stream = fs.createReadStream('dist/index.html');
50
- return reply.type('text/html').send(stream);
51
- });
52
-
53
- async function staticFile(req, reply) {
54
- const assetsDir = 'dist';
55
- const filePath = path.join(root, assetsDir, req.url);
56
- const ext = path.extname(filePath);
57
- const mime = {
58
- '.js': 'text/javascript', '.css': 'text/css', '.woff2': 'application/font-woff', '.png': 'image/png', '.svg': 'image/svg+xml', '.jpg': 'image/jpg',
59
- }[ext];
60
- reply.headers({ 'Cache-control': 'max-age=3600, public' });
61
- const stream = fs.createReadStream(filePath);
62
- return mime ? reply.type(mime).send(stream) : stream;
63
- }
64
-
65
- fastify.get('/assets/*', staticFile);
66
- fastify.get('/public/*', staticFile);
67
- }
68
-
69
- export default plugin;
1
+ import fs from 'fs';
2
+ import path, { dirname } from 'path';
3
+ import { fileURLToPath } from 'url';
4
+
5
+ const dir = dirname(fileURLToPath(import.meta.url));
6
+ const root = `${dir}/../..`;
7
+
8
+ const isProduction = process.env.NODE_ENV === 'production';
9
+
10
+ async function plugin(fastify) {
11
+ // vite server
12
+ if (!isProduction) {
13
+ const vite = await import('vite');
14
+
15
+ const viteServer = await vite.createServer({
16
+ server: {
17
+ middlewareMode: true,
18
+ },
19
+ });
20
+ // hot reload
21
+ viteServer.watcher.on('all', function (d, t) {
22
+ if (!t.includes('module')) return;
23
+ console.log(d, t);
24
+ viteServer.ws.send({ type: 'full-reload' });
25
+ });
26
+
27
+ // this is middleware for vite's dev servert
28
+ fastify.addHook('onRequest', async (req, reply) => {
29
+ const { user } = req.session?.passport || {};
30
+ const { disableAuth } = req.funcs?.config || {};
31
+ if (!user && !disableAuth) {
32
+ // return reply.send('access restricted');
33
+ }
34
+
35
+ const next = () => new Promise((resolve) => {
36
+ viteServer.middlewares(req.raw, reply.raw, () => resolve());
37
+ });
38
+ await next();
39
+ });
40
+ fastify.get('*', async () => { });
41
+ return;
42
+ }
43
+
44
+ // From Build
45
+ fastify.get('*', async (req, reply) => {
46
+ const { user } = req.session?.passport || {};
47
+
48
+ if (!user) return reply.redirect('/login');
49
+ const stream = fs.createReadStream('dist/index.html');
50
+ return reply.type('text/html').send(stream);
51
+ });
52
+
53
+ async function staticFile(req, reply) {
54
+ const assetsDir = 'dist';
55
+ const filePath = path.join(root, assetsDir, req.url);
56
+ const ext = path.extname(filePath);
57
+ const mime = {
58
+ '.js': 'text/javascript', '.css': 'text/css', '.woff2': 'application/font-woff', '.png': 'image/png', '.svg': 'image/svg+xml', '.jpg': 'image/jpg',
59
+ }[ext];
60
+ reply.headers({ 'Cache-control': 'max-age=3600, public' });
61
+ const stream = fs.createReadStream(filePath);
62
+ return mime ? reply.type(mime).send(stream) : stream;
63
+ }
64
+
65
+ fastify.get('/assets/*', staticFile);
66
+ fastify.get('/public/*', staticFile);
67
+ }
68
+
69
+ export default plugin;