@opengis/admin 0.1.75 → 0.1.76

Sign up to get free protection for your applications and to get access to all the features.
Files changed (48) hide show
  1. package/dist/{IconChevronDown-DpFXvgwL.js → IconChevronDown-BMnTiJIv.js} +1 -1
  2. package/dist/{add-page-B34T9kD5.js → add-page-BxK4iJe4.js} +1 -1
  3. package/dist/{admin-interface-B7RH9jC0.js → admin-interface-D9vWpTqM.js} +591 -534
  4. package/dist/{admin-view-CB6GnWqy.js → admin-view-DPtlpZue.js} +144 -139
  5. package/dist/admin.js +1 -1
  6. package/dist/admin.umd.cjs +51 -51
  7. package/dist/assets/favicon.svg +1 -0
  8. package/dist/assets/logo.svg +42 -0
  9. package/dist/{card-page-B2ZFa7Pi.js → card-page-KnsaeAPW.js} +3 -3
  10. package/dist/{card-view-BRWlfh-Z.js → card-view-BAC53pSQ.js} +1 -1
  11. package/dist/{edit-page-BUhpjmtA.js → edit-page-D2KM1hue.js} +1 -1
  12. package/dist/{import-file-C08TESFR.js → import-file-C3Slz6rJ.js} +4770 -4754
  13. package/dist/style.css +1 -1
  14. package/module/settings/select/core.user_mentioned.sql +2 -0
  15. package/package.json +4 -4
  16. package/plugin.js +3 -0
  17. package/server/helpers/index.mjs +7 -5
  18. package/server/routes/calendar/controllers/calendar.data.js +3 -3
  19. package/server/routes/data/controllers/cardData.js +5 -5
  20. package/server/routes/data/controllers/cardTabData.js +5 -5
  21. package/server/routes/data/controllers/funcs/getFilterSQL/index.js +3 -3
  22. package/server/routes/data/controllers/tableData.js +4 -3
  23. package/server/routes/data/controllers/tableDataId.js +3 -1
  24. package/server/routes/data/controllers/tableFilter.js +3 -3
  25. package/server/routes/data/controllers/utils/assignTokens.js +6 -6
  26. package/server/routes/notifications/controllers/readNotifications.js +9 -5
  27. package/server/routes/notifications/controllers/testEmail.js +35 -0
  28. package/server/routes/notifications/controllers/userNotifications.js +1 -1
  29. package/server/routes/notifications/funcs/addNotification.js +21 -0
  30. package/server/routes/notifications/funcs/sendNotification.js +105 -0
  31. package/server/routes/notifications/funcs/utils/sendEmail.js +39 -0
  32. package/server/routes/notifications/hook/onWidgetSet.js +4 -4
  33. package/server/routes/notifications/index.mjs +14 -2
  34. package/server/routes/notifications/schema.js +7 -1
  35. package/server/routes/properties/controllers/admin.properties.get.js +2 -2
  36. package/server/routes/user/controllers/user.cls.id.js +14 -0
  37. package/server/routes/user/controllers/user.cls.js +71 -0
  38. package/server/routes/user/controllers/user.cls.post.js +52 -0
  39. package/server/routes/user/controllers/user.info.js +17 -0
  40. package/server/routes/user/index.mjs +15 -0
  41. package/server/routes/user/schema.js +14 -0
  42. package/server/routes/widget/controllers/utils/historyFormat.js +75 -0
  43. package/server/routes/widget/controllers/utils/obj2db.js +13 -0
  44. package/server/routes/widget/controllers/widget.del.js +41 -0
  45. package/server/routes/widget/controllers/widget.get.js +96 -0
  46. package/server/routes/widget/controllers/widget.set.js +76 -0
  47. package/server/routes/widget/index.mjs +39 -0
  48. package/utils.js +5 -0
@@ -1,14 +1,26 @@
1
- import { addHook } from '@opengis/fastify-table/utils.js';
1
+ import { config, addHook } from '@opengis/fastify-table/utils.js';
2
2
 
3
3
  // api
4
+ import testEmail from './controllers/testEmail.js';
4
5
  import readNotifications from './controllers/readNotifications.js'; // mark as read
5
6
  import userNotifications from './controllers/userNotifications.js'; // check all, backend pagination
7
+
6
8
  // hook
7
9
  import onWidgetSet from './hook/onWidgetSet.js'; // send notification on comment
8
10
 
9
- import { notificationSchema } from './schema.js';
11
+ // funcs
12
+ // import addNotification from '../notifications/funcs/addNotification.js'; // add to db
13
+ // import notification from '../notifications/funcs/sendNotification.js'; // send notification
14
+
15
+ import { notificationSchema, emailSchema } from './schema.js';
10
16
 
11
17
  export default async function route(fastify) {
18
+ const prefix = config.prefix || '/api';
19
+
20
+ // fastify.decorate('addNotification', addNotification);
21
+ // fastify.decorate('notification', notification);
22
+
23
+ fastify.get(`${prefix}/test-email`, { config: { policy: ['user'] }, schema: emailSchema }, testEmail);
12
24
  fastify.get(`/notification`, { config: { policy: ['user'] }, schema: notificationSchema }, userNotifications);
13
25
  fastify.get(`/notification-read/:id?`, { config: { policy: ['user'] }, schema: notificationSchema }, readNotifications);
14
26
  addHook('onWidgetSet', onWidgetSet);
@@ -7,5 +7,11 @@ const notificationSchema = {
7
7
  },
8
8
  };
9
9
 
10
+ const emailSchema = {
11
+ quertstring: {
12
+ to: { type: 'string', pattern: '^((?!\\.)[\\w\\-_.]*[^.])(@\\w+)(\\.\\w+(\\.\\w+)?[^.\\W])$' },
13
+ },
14
+ };
15
+
10
16
  export default null;
11
- export { notificationSchema }
17
+ export { notificationSchema, emailSchema }
@@ -1,6 +1,6 @@
1
1
  const table = 'admin.properties';
2
2
 
3
- import { getRedis } from '@opengis/fastify-table/utils.js';
3
+ import { config, getRedis } from '@opengis/fastify-table/utils.js';
4
4
 
5
5
  import { getSettings } from '../../../../utils.js';
6
6
 
@@ -15,7 +15,7 @@ export default async function getSettingsAPI({
15
15
  const keyCache = `${pg.options?.database}:settings:${params?.key || 'all'}:${query?.json ? 'json' : 'plain'}:${table}`;
16
16
 
17
17
  const cache = await redis.get(keyCache);
18
- if (cache && !funcs.config?.local) return JSON.parse(cache);
18
+ if (cache && !config?.local) return JSON.parse(cache);
19
19
 
20
20
  try {
21
21
  const res = await getSettings({
@@ -0,0 +1,14 @@
1
+ import userCls from './user.cls.js';
2
+
3
+ export default async function userClsId(req) {
4
+ const { id } = req.params || {};
5
+ const res = await userCls(req);
6
+ if (req.query?.sql || res?.error) {
7
+ return res;
8
+ }
9
+ const { rows = [] } = res?.message || {};
10
+ if (!rows?.length) {
11
+ return { message: `cls not found: ${id}`, status: 404 };
12
+ }
13
+ return { message: rows?.[0], status: 200 };
14
+ }
@@ -0,0 +1,71 @@
1
+ import { readFile } from 'fs/promises';
2
+ import { getTemplatePath } from '@opengis/fastify-table/utils.js';
3
+
4
+ export default async function userCls(req) {
5
+ const {
6
+ pg, params = {}, query = {}, session = {},
7
+ } = req;
8
+ const { uid } = session.passport?.user || {};
9
+
10
+ if (!uid) {
11
+ return { message: 'access restricted', status: 403 };
12
+ }
13
+
14
+ if (query.type && !['json', 'sql'].includes(query.type)) {
15
+ return { message: 'param type is invalid', status: 400 };
16
+ }
17
+
18
+ const q = `select user_clsid as id, name, type,
19
+ case when type='json' then (
20
+ ${params?.id
21
+ ? `(with recursive rows as (
22
+ select user_clsid, code as id, name as text, icon, color, parent
23
+ from admin.user_cls a
24
+ where name=u.name
25
+ union all
26
+ select a.user_clsid, a.code, a.name, a.icon, a.color, a.parent
27
+ from admin.user_cls a
28
+ join rows b on a.parent=b.text
29
+ ) select json_agg(row_to_json(q)) from rows q where text<>u.name
30
+ )`
31
+ : 'select count(*)::int from admin.user_cls where parent=u.name'} ) else null end as children,
32
+ case when type='sql' then data else null end as sql from admin.user_cls u
33
+ where (case when type='json' then parent is null else true end)
34
+ and uid=(select uid from admin.users where $1 in (login,uid) limit 1)
35
+ and parent is null and ${query.type ? `type='${query.type}'` : '1=1'} and ${params?.id ? 'u.name=$2' : '1=1'}`;
36
+
37
+ if (query?.sql) return q;
38
+
39
+
40
+ const { rows = [] } = await pg.query(q, [uid, params.id].filter((el) => el));
41
+
42
+ rows.forEach((row) => {
43
+ if (row.type === 'sql') delete row.children;
44
+ if (row.type === 'json') { delete row.sql; Object.assign(row, { children: row.children || (params?.id ? [] : 0) }); }
45
+ });
46
+
47
+ const clsList = query.type === 'sql' ? [] : getTemplatePath('cls'); // skip cls if type sql
48
+ const selectList = query.type === 'json' ? [] : getTemplatePath('select'); // skip sql if type json
49
+
50
+ const userClsNames = rows.map((el) => el.name);
51
+ const selectNames = selectList.map((el) => el[0]);
52
+
53
+ const arr = (clsList || []).concat(selectList || []).map((el) => ({ name: el[0], path: el[1], type: el[2] }))
54
+ ?.filter((el) => (params?.id ? el.name === params?.id : true))
55
+ ?.filter((el) => ['sql', 'json'].includes(el?.type) && !userClsNames.includes(el.name))
56
+ ?.filter((el) => (el?.type === 'json' ? !selectNames?.includes(el?.name) : true));
57
+
58
+ const res = await Promise.all(arr?.map(async (el) => {
59
+ const type = { json: 'cls', sql: 'select' }[el.type] || el.type;
60
+ // const clsData = await getSelect(type, el.name);
61
+ const str = await readFile(el.path, 'utf-8');
62
+ const clsData = type === 'cls' ? JSON.parse(str) : str;
63
+ if (type === 'cls') {
64
+ const children = params?.id ? clsData : clsData?.length || 0;
65
+ return { name: el.name, type: el.type, children };
66
+ }
67
+ return { name: el.name, type: el.type, sql: clsData?.sql || clsData };
68
+ }));
69
+
70
+ return { message: { rows: rows.concat(res) }, status: 200 };
71
+ }
@@ -0,0 +1,52 @@
1
+ export default async function userClsPost({
2
+ pg, body = {}, session = {},
3
+ }) {
4
+ const { uid } = session.passport?.user || {};
5
+ const {
6
+ name, type = 'json', children = [], sql,
7
+ } = body;
8
+
9
+ if (!uid) {
10
+ return { message: 'access restricted', status: 403 };
11
+ }
12
+
13
+ if (!name) {
14
+ return { message: 'not enough params: name', status: 400 };
15
+ }
16
+
17
+ if (type === 'json' && (!Array.isArray(children) || !children.length || !children?.[0]?.id)) {
18
+ return { message: 'invalid params: children (array of objects)', status: 400 };
19
+ }
20
+
21
+ if (type === 'sql' && (!sql || typeof sql !== 'string')) {
22
+ return { message: 'invalid params: sql (string)', status: 400 };
23
+ }
24
+
25
+
26
+ const { rowCount = 0 } = await pg.query(`with recursive rows as (
27
+ select user_clsid, name, code, icon, color, parent
28
+ from admin.user_cls a
29
+ where name=$1
30
+ union all
31
+ select a.user_clsid, a.name, a.code, a.icon, a.color, a.parent
32
+ from admin.user_cls a
33
+ join rows b on a.parent=b.name
34
+ ) delete from admin.user_cls where user_clsid in (select user_clsid from rows )`, [name]);
35
+ console.log('delete old user cls', name, rowCount);
36
+
37
+ const { id, data } = await pg.query('insert into admin.user_cls(name,type,data,uid) values($1,$2,$3,$4) returning user_clsid as id, data', [name, type, sql, uid])
38
+ .then((res) => res.rows?.[0] || {});
39
+ if (type === 'json') {
40
+ if (!id) { return { error: 'insert user cls error', status: 500 }; }
41
+ const q1 = `insert into admin.user_cls(code,name,color,icon,parent,uid)
42
+
43
+ select value->>'id',value->>'text',value->>'color',value->>'icon', '${name.replace(/'/g, "''")}', '${uid}'
44
+ from json_array_elements('${JSON.stringify(children).replace(/'/g, "''")}'::json)
45
+
46
+ returning user_clsid as id, code, name, parent`;
47
+ const { rows = [] } = await pg.query(q1);
48
+ return { id, children: rows };
49
+ }
50
+ return { id, data };
51
+
52
+ }
@@ -0,0 +1,17 @@
1
+ export default async function userInfo({
2
+ pg, session = {},
3
+ }) {
4
+ const { uid } = session.passport?.user || {};
5
+
6
+ if (!uid) {
7
+ return { message: 'access restricted', status: 403 };
8
+ }
9
+
10
+ const data = await pg.query(`select user_name, sur_name, father_name, user_rnokpp, user_type, email, login from admin.users
11
+ where uid=$1`, [uid]).then((res) => res.rows?.[0] || {});
12
+
13
+ const { notifications = 0 } = await pg.query(`select count(*)::int as notifications from crm.notifications
14
+ where addressee_id=$1 and read is not true`, [uid]).then((res) => res.rows?.[0] || {});
15
+ return { uid, ...data, notifications };
16
+
17
+ }
@@ -0,0 +1,15 @@
1
+ import userCls from './controllers/user.cls.js';
2
+ import userClsId from './controllers/user.cls.id.js';
3
+ import userInfo from './controllers/user.info.js';
4
+ import userClsPost from './controllers/user.cls.post.js';
5
+
6
+ import { userClsSchema, userClsIdSchema } from './schema.js';
7
+
8
+ async function plugin(fastify, opts) {
9
+ fastify.get('/user-cls', { config: { policy: ['user'] }, schema: userClsSchema }, userCls);
10
+ fastify.get('/user-cls/:id', { config: { policy: ['user'] }, schema: userClsIdSchema }, userClsId);
11
+ fastify.get('/user-info', { config: { policy: ['user'] } }, userInfo);
12
+ fastify.post('/user-cls', { config: { policy: ['user'] }, schema: userClsIdSchema }, userClsPost);
13
+ }
14
+
15
+ export default plugin;
@@ -0,0 +1,14 @@
1
+ const userClsSchema = {
2
+ quertstring: {
3
+ sql: { type: 'string', pattern: '^(\\d)$' },
4
+ },
5
+ };
6
+
7
+ const userClsIdSchema = {
8
+ params: {
9
+ id: { type: 'string', pattern: '^([\\d\\w\\.]+)$' },
10
+ },
11
+ };
12
+
13
+ export { userClsSchema, userClsIdSchema };
14
+ export default null;
@@ -0,0 +1,75 @@
1
+ import { readdir } from 'fs/promises';
2
+ import { existsSync, readFileSync } from 'fs';
3
+
4
+ import { getSelect } from '@opengis/fastify-table/utils.js';
5
+
6
+ const dbData = {};
7
+
8
+ // from config??
9
+ const allTemplates = { table: {} };
10
+
11
+ const historyQ = `select nspname||'.'||relname as table_name, json_agg(json_build_object('name',attname, 'title',coalesce(col_description(attrelid, attnum),attname))) as columns
12
+ from pg_attribute a
13
+ left join pg_catalog.pg_attrdef d ON (a.attrelid, a.attnum) = (d.adrelid, d.adnum)
14
+ JOIN pg_class AS i
15
+ ON i.oid = a.attrelid
16
+ JOIN pg_namespace AS NS ON i.relnamespace = NS.OID
17
+ where a.attnum > 0
18
+ and not a.attisdropped
19
+ group by nspname||'.'||relname`;
20
+
21
+ export default async function historyFormat(rows, table, pg) {
22
+ if (!rows?.[0]?.changes) return rows; // old structure
23
+ // on startup
24
+ if (!allTemplates.table.length) {
25
+ const templateDir = './server/templates/table';
26
+ const templates = existsSync(templateDir) ? await readdir(templateDir) : [];
27
+ templates.forEach((template) => {
28
+ const body = JSON.parse(readFileSync(`${templateDir}/${template}`) || '{}');
29
+ Object.assign(allTemplates.table, { [template]: body });
30
+ });
31
+ }
32
+
33
+ const progrid = Object.keys(allTemplates.table).find((key) => key.replace('.json', '') === table);
34
+ if (!progrid) return rows;
35
+ // const body = await getTemplate('table', progrid);
36
+ const body = allTemplates.table[progrid];
37
+ const tableName = body?.table || table;
38
+ if (!tableName) return rows;
39
+
40
+ // get DB column description
41
+ if (!dbData?.[body.table]?.length) {
42
+ const { rows: rows1 } = await pg.query(historyQ);
43
+ rows1.forEach((row) => {
44
+ dbData[row.table_name] = row.columns;
45
+ });
46
+ }
47
+
48
+ // rewrite!!!
49
+ await Promise.all(rows?.map(async (op) => {
50
+ op.changes?.map(async (el) => {
51
+ const col = body.colModel.filter((col1) => col1.name === el.attr)[0] || dbData[body?.table]?.find((col1) => col1.name === el.attr);
52
+ if (el.attr === 'geom') {
53
+ el.title = 'Геометрія';
54
+ }
55
+ el.title = col?.ua || col?.title || el.attr;
56
+ if (!col) return;
57
+ el.type = col.type;
58
+ el.format = col.format;
59
+ const select = col.data || col.option || col.select;
60
+
61
+ // getSelect not equals to node
62
+ if (select && false) {
63
+ el.select = select;
64
+ const cls = await getSelect(select, {
65
+ val: [el.old, el.new],
66
+ });
67
+
68
+ el.oldf = cls[0] || el.old;
69
+ el.newf = cls[1] || el.new;
70
+ }
71
+ });
72
+ }));
73
+
74
+ return rows;
75
+ }
@@ -0,0 +1,13 @@
1
+ export default function obj2db(data, nonexistCol) {
2
+ if (
3
+ typeof data !== 'object'
4
+ || Array.isArray(data)
5
+ || !Object.keys(data || {}).length
6
+ ) return { error: 'invalid data type' };
7
+
8
+ const columns = Object.keys(data)?.filter((key) => !nonexistCol.includes(key));
9
+ // const columns = existColumns?.filter((col) => data[col] || data?.[col] === 0);
10
+ const args = columns?.map((col) => data[col]);
11
+
12
+ return { columns, args, error: !columns?.length ? 'nothing to process' : undefined };
13
+ }
@@ -0,0 +1,41 @@
1
+ /**
2
+ * Дістає CRM дані для vue хешує ідентифікатори, підтягує селекти
3
+ *
4
+ * @method DELETE
5
+ * @summary CRM дані для обраного віджета.
6
+ * @priority 2
7
+ * @tag table
8
+ * @type api
9
+ * @requires setTokenById
10
+ * @requires getSelect
11
+ * @param {String} id Ідентифікатор для хешування
12
+ * @param {Any} sql Використовується для повернення sql запиту
13
+ * @param {String} type Тип для хешування даних
14
+ * @errors 400, 500
15
+ * @returns {Number} status Номер помилки
16
+ * @returns {String|Object} error Опис помилки
17
+ * @returns {String|Object} message Повідомлення про успішне виконання або об'єкт з параметрами
18
+ */
19
+
20
+ export default async function widgetDel({
21
+ pg, params = {}, session = {},
22
+ }) {
23
+ const { user = {} } = session.passport || {};
24
+ if (!user.uid) return { error: 'access restricted', status: 403 };
25
+ const { type, objectid, id } = params;
26
+
27
+ if (!objectid) return { error: 'id required', status: 400 };
28
+
29
+ const sqls = {
30
+ comment: 'delete from crm.communications where entity_id=$1 and uid=$2 and communication_id=$3',
31
+ checklist: 'delete from crm.checklists where entity_id=$1 and uid=$2 and checklist_id=$3',
32
+ file: 'update crm.files set file_status=3 where entity_id=$1 and uid=$2 and file_id=$3',
33
+ gallery: 'update crm.files set file_status=3 where entity_id=$1 and uid=$2 and file_id=$3',
34
+ };
35
+ const sql = sqls[type];
36
+ if (!sql) return { error: 'type not valid', status: 401 };
37
+
38
+ await pg.query(sql, [objectid, user.uid, id]);
39
+ return { data: { id }, user: { uid: user.uid, name: user.user_name } };
40
+
41
+ }
@@ -0,0 +1,96 @@
1
+ import { getMeta, getToken } from '@opengis/fastify-table/utils.js';
2
+
3
+ import historyFormat from './utils/historyFormat.js';
4
+
5
+ const galleryExtList = ['png', 'svg', 'jpg', 'jpeg', 'gif', 'mp4', 'mov', 'avi'];
6
+
7
+ /**
8
+ * Дістає CRM для widget
9
+ *
10
+ */
11
+
12
+ export default async function widgetGet({
13
+ pg, session = {}, params = {}, query = {},
14
+ }) {
15
+ const { user = {} } = session.passport || {};
16
+
17
+ const param = user?.uid ? await getToken({
18
+ token: params.objectid, mode: 'w', uid: user.uid,
19
+ }) : null;
20
+
21
+ const objectid = param ? JSON.parse(param)?.id : params.objectid;
22
+
23
+ if (!objectid) return { error: 'id required', status: 400 };
24
+
25
+ const sqls = {
26
+ comment: pg.pk['admin.users']
27
+ ? `select communication_id, entity_id, body, subject, c.cdate, c.uid,
28
+ coalesce(user_name,' ')||' '||coalesce(sur_name,'') as username, avatar
29
+ from crm.communications c left join admin.users u on u.uid=c.uid where entity_id=$1 order by cdate desc`
30
+ : 'select communication_id, entity_id, body, subject, cdate, uid from crm.communications where entity_id=$1 order by cdate desc',
31
+
32
+ history: `SELECT b.change_data_id, change_id, entity_id, entity_type, change_type, change_date, uid, cdate, b.entity_key, b.value_new, b.value_old FROM log.table_changes a
33
+ left join lateral(
34
+ select change_data_id, entity_key, value_new, value_old from log.table_changes_data where change_id=a.change_id
35
+ )b on 1=1
36
+ where (entity_id=$1 or entity_id in (
37
+ select communication_id as comments from crm.communications where entity_id=$1
38
+ union all select file_id from crm.files where entity_id=$1
39
+ union all select checklist_id from crm.checklists where entity_id=$1)
40
+ ) and b.change_data_id is not null order by cdate desc limit 100`,
41
+
42
+ checklist: pg.pk['admin.users']
43
+ ? `SELECT checklist_id, entity_id, subject, is_done, done_date, c.uid, c.cdate, coalesce(user_name,' ')||' '||coalesce(sur_name,'') as username,
44
+ avatar FROM crm.checklists c left join admin.users u on u.uid=c.uid where entity_id=$1 order by cdate desc`
45
+ : 'SELECT checklist_id, entity_id, subject, is_done, done_date, uid, cdate FROM crm.checklists where entity_id=$1 order by cdate desc',
46
+
47
+ file: pg.pk['admin.users']
48
+ ? `SELECT file_id, entity_id, entity_type, file_path, uploaded_name, ext, size, c.uid, c.cdate, file_type, c.ismain,
49
+ coalesce(user_name,' ')||' '||coalesce(sur_name,'') as username, isverified,
50
+ avatar, c.uid as author, file_status FROM crm.files c left join admin.users u on u.uid=c.uid
51
+ where entity_id=$1 and file_status<>3 order by cdate desc`
52
+ : `SELECT file_id, entity_id, entity_type, file_path, uploaded_name, ext, size, uid, cdate, file_type, ismain,
53
+ isverified, uid as author, file_status FROM crm.files c where entity_id=$1 and file_status<>3 order by cdate desc`,
54
+ gallery: pg.pk['admin.users']
55
+ ? `SELECT file_id, entity_id, entity_type, file_path, uploaded_name, ext, size, c.uid, c.cdate, file_type, c.ismain,
56
+ coalesce(user_name,' ')||' '||coalesce(sur_name,'') as username, isverified,
57
+ avatar, c.uid as author, file_status FROM crm.files c left join admin.users u on u.uid=c.uid
58
+ where entity_id=$1 and file_status<>3 and ext = any($2) order by cdate desc`
59
+ : `SELECT file_id, entity_id, entity_type, file_path, uploaded_name, ext, size, c.uid, c.cdate, file_type, ismain,
60
+ isverified, uid as author, file_status FROM crm.files c where entity_id=$1 and file_status<>3 and ext = any($2) order by cdate desc`,
61
+
62
+ };
63
+ const sql = sqls[params.type];
64
+ if (!sql) {
65
+ return { error: 'param type not valid', status: 400 };
66
+ }
67
+
68
+
69
+ /* data */
70
+ const time = [Date.now()];
71
+ const { rows } = await pg.query(sql, [objectid, params.type === 'gallery' ? galleryExtList : null].filter((el) => el));
72
+ time.push(Date.now());
73
+
74
+ /* Object info */
75
+ const { tableName } = pg.pk['log.table_changes'] ? await pg.one('select entity_type as "tableName" from log.table_changes where entity_id=$1 limit 1', [objectid]) : {};
76
+ const { pk } = await getMeta({ table: tableName });
77
+
78
+ const q = tableName && pg.pk['admin.users'] ? `select coalesce(b.user_name,'')||coalesce(' '||b.sur_name,'') as author, a.cdate, a.editor_date from ${tableName} a
79
+ left join admin.users b on a.uid=b.uid where a.${pk}=$1 limit 1` : undefined;
80
+ const data = pk && q ? await pg.one(q, [objectid]) : {};
81
+
82
+ if (query.debug && user?.user_type === 'admin') {
83
+ return {
84
+ sql, type: params.type, q, id: objectid, data,
85
+ };
86
+ }
87
+
88
+ time.push(Date.now());
89
+ return {
90
+ time: { data: time[1] - time[0], format: time[2] - time[1] },
91
+ rows: params.type === 'history' ? await historyFormat(rows, tableName, pg) : rows,
92
+ user: { uid: user?.uid, name: user?.user_name },
93
+ data: { author: data?.author, cdate: data?.cdate, edate: data?.editor_date },
94
+ objectid: params.objectid,
95
+ };
96
+ }
@@ -0,0 +1,76 @@
1
+ import path from 'path';
2
+
3
+ import {
4
+ getMeta, dataInsert, dataUpdate, applyHook,
5
+ } from '@opengis/fastify-table/utils.js';
6
+
7
+ import { uploadMultiPart } from '@opengis/fastify-file/utils.js';
8
+
9
+ const tableList = {
10
+ comment: 'crm.communications',
11
+ checklist: 'crm.checklists',
12
+ };
13
+ const pkList = {
14
+ comment: 'communication_id',
15
+ checklist: 'checklist_id',
16
+ };
17
+
18
+ const galleryExtList = ['png', 'svg', 'jpg', 'jpeg', 'gif', 'mp4', 'mov', 'avi'];
19
+
20
+ export default async function widgetSet(req) {
21
+ const {
22
+ pg, params = {}, session = {}, body = {},
23
+ } = req;
24
+ const { user = {} } = session.passport || {};
25
+ const { type, id, objectid } = params;
26
+ if (!['comment', 'checklist', 'file', 'gallery'].includes(type)) return { message: 'param type not valid', status: 400 };
27
+ if (!objectid) return { message: 'id required', status: 400 };
28
+
29
+ const table = tableList[type];
30
+
31
+
32
+ if (['gallery', 'file'].includes(type)) {
33
+ const file = await uploadMultiPart(req);
34
+ const extName = path.extname(file.filepath).slice(1).toLowerCase();
35
+
36
+ const data = {
37
+ uploaded_name: file?.originalFilename?.toLocaleLowerCase()?.replace(/'/g, '\'\''),
38
+ file_path: file?.relativeFilepath?.replace(/\\/g, '/'),
39
+ ext: extName,
40
+ size: file?.size,
41
+ file_status: 1,
42
+ uid: user?.uid || 1,
43
+ entity_id: objectid,
44
+ };
45
+
46
+ if (type === 'gallery' && !galleryExtList.includes(extName.toLowerCase())) {
47
+ return { message: 'invalid file extension', status: 400 };
48
+ }
49
+
50
+ const { rows = [] } = await dataInsert({
51
+ table: 'crm.files', data, uid: user?.uid,
52
+ });
53
+ return {
54
+ rowCount: 1, data: 'ok', command: 'UPLOAD', id: rows[0]?.file_id, entity_id: rows[0]?.entity_id,
55
+ };
56
+ }
57
+ const { pk } = await getMeta({ pg, table });
58
+ if (!pk) return { message: 'table not found', status: 404 };
59
+
60
+ const data = { ...body, uid: user?.uid, entity_id: objectid };
61
+
62
+ await applyHook('onWidgetSet', {
63
+ link: req.path, session, type, payload: data,
64
+ });
65
+ const result = id
66
+ ? await dataUpdate({
67
+ table, data, id, uid: user?.uid,
68
+ })
69
+ : await dataInsert({
70
+ table, data, uid: user?.uid,
71
+ });
72
+
73
+ return {
74
+ rowCount: result.rowCount, data: 'ok', command: result.command, id: result.rows?.[0]?.[pkList[type]] || result?.[pkList[type]],
75
+ };
76
+ }
@@ -0,0 +1,39 @@
1
+ import widgetDel from './controllers/widget.del.js';
2
+ import widgetSet from './controllers/widget.set.js';
3
+ import widgetGet from './controllers/widget.get.js';
4
+
5
+ const tableSchema = {
6
+ params: {
7
+ // type: { type: 'string', pattern: '^([\\d\\w]+)$' },
8
+ objectid: { type: 'string', pattern: '^([\\d\\w]+)$' },
9
+ id: { type: 'string', pattern: '^([\\d\\w]+)$' },
10
+ },
11
+ querystring: {
12
+ debug: { type: 'string', pattern: '^(\\d+)$' },
13
+ },
14
+ };
15
+
16
+ async function route(fastify, opt) {
17
+ fastify.route({
18
+ method: 'DELETE',
19
+ url: '/widget/:type/:objectid/:id',
20
+ schema: tableSchema,
21
+ handler: widgetDel,
22
+ });
23
+ fastify.route({
24
+ method: 'POST',
25
+ path: '/widget/:type/:objectid/:id?',
26
+ schema: tableSchema,
27
+ handler: widgetSet,
28
+ });
29
+ fastify.route({
30
+ method: 'GET',
31
+ path: '/widget/:type/:objectid',
32
+ config: {
33
+ policy: ['public'],
34
+ },
35
+ schema: tableSchema,
36
+ handler: widgetGet,
37
+ });
38
+ }
39
+ export default route;
package/utils.js CHANGED
@@ -3,6 +3,9 @@ import yamlSafe from 'js-yaml';
3
3
  import getSettings from './server/routes/properties/funcs/getSettings.js';
4
4
  import setSettings from './server/routes/properties/funcs/setSettings.js';
5
5
 
6
+ import addNotification from './server/routes/notifications/funcs/addNotification.js';
7
+ import sendNotification from './server/routes/notifications/funcs/sendNotification.js';
8
+
6
9
  function loadSafe (yml) {
7
10
  try {
8
11
  return yamlSafe.load(yml);
@@ -18,4 +21,6 @@ export {
18
21
  yamlSafe,
19
22
  getSettings,
20
23
  setSettings,
24
+ addNotification,
25
+ sendNotification,
21
26
  };