@opengis/admin 0.4.34 → 0.4.35
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/README.md +97 -97
 - package/dist/{add-page-BbGCyn0k.js → add-page-GMWte58m.js} +1 -1
 - package/dist/{admin-interface-BbtDg4Fy.js → admin-interface-TLM61JNF.js} +2 -2
 - package/dist/{admin-view-BT95LpxW.js → admin-view-BKD-rKqQ.js} +1 -1
 - package/dist/admin.js +1 -1
 - package/dist/admin.umd.cjs +57 -57
 - package/dist/assets/logo.svg +41 -41
 - package/dist/{card-view-Cwu20O9Z.js → card-view-CwkJ9Qg2.js} +1 -1
 - package/dist/{edit-page-D1ZN9oak.js → edit-page-C_igWB2a.js} +1 -1
 - package/dist/{import-file-CGrExq_X.js → import-file-2iB1izw1.js} +11497 -11244
 - package/dist/{profile-page-Quk3Xc6g.js → profile-page-Cuy6hkl8.js} +1 -1
 - package/dist/style.css +1 -1
 - package/module/settings/card/admin.accounts.table/index.yml +7 -7
 - package/module/settings/card/admin.accounts.table/rules.hbs +18 -18
 - package/module/settings/card/admin.accounts.table/users.hbs +13 -13
 - package/module/settings/card/admin.routes.table/groups.hbs +11 -11
 - package/module/settings/card/admin.routes.table/users.hbs +16 -16
 - package/module/settings/cls/core.actions.json +17 -17
 - package/module/settings/cls/core.scope.json +13 -13
 - package/module/settings/cls/properties.site_status.json +13 -13
 - package/module/settings/cls/properties.widget_status.json +13 -13
 - package/module/settings/cls/yes_no.json +11 -11
 - package/module/settings/form/admin.accounts.form.json +13 -13
 - package/module/settings/form/admin.properties.form.json +15 -15
 - package/module/settings/form/admin.roles.form.json +21 -21
 - package/module/settings/form/admin.user_properties.form.json +15 -15
 - package/module/settings/form/admin.user_roles_card.form.json +13 -13
 - package/module/settings/interface/admin.properties.json +4 -4
 - package/module/settings/interface/admin.roles.json +4 -4
 - package/module/settings/interface/admin.routes.json +4 -4
 - package/module/settings/interface/admin.users.json +4 -4
 - package/module/settings/select/core.routes.sql +1 -1
 - package/module/settings/select/core.user_mentioned.sql +1 -1
 - package/module/settings/select/core.user_uid.sql +1 -1
 - package/module/settings/table/admin.properties.table.json +39 -39
 - package/module/settings/table/admin.user_properties.table.json +34 -34
 - package/package.json +1 -1
 - package/server/helpers/core/coalesce.js +7 -7
 - package/server/helpers/core/select.js +48 -48
 - package/server/helpers/core/token.js +18 -18
 - package/server/helpers/list/buttonHelper.js +21 -21
 - package/server/helpers/list/utils/button.js +5 -5
 - package/server/helpers/temp/contentList.js +58 -58
 - package/server/helpers/temp/ifCond.js +101 -101
 - package/server/helpers/utils/button.js +5 -5
 - package/server/helpers/utils/mdToHTML.js +17 -17
 - package/server/plugins/cron.js +10 -10
 - package/server/plugins/docs.js +28 -28
 - package/server/plugins/hook.js +4 -4
 - package/server/routes/calendar/controllers/calendar.data.js +124 -124
 - package/server/routes/calendar/index.mjs +7 -7
 - package/server/routes/notifications/controllers/readNotifications.js +18 -18
 - package/server/routes/notifications/controllers/testEmail.js +35 -35
 - package/server/routes/notifications/controllers/userNotifications.js +53 -53
 - package/server/routes/notifications/hook/onWidgetSet.js +56 -56
 - package/server/routes/notifications/index.mjs +26 -26
 - package/server/routes/root.mjs +3 -3
 - package/server/routes/user/controllers/user.cls.id.js +14 -14
 - package/server/routes/user/controllers/user.cls.js +72 -72
 - package/server/routes/user/controllers/user.info.js +17 -17
 - package/server/templates/cls/itree.recrzone_category.json +73 -73
 - package/server/templates/cls/test.json +9 -9
 - package/server/templates/form/admin.user_cls.data.form.json +49 -49
 - package/server/templates/form/admin.user_group_rel.form.json +21 -21
 - package/server/templates/form/form-user-pass.json +10 -10
 - package/server/templates/form/form-user_group.json +39 -39
 - package/server/templates/form/form-users.json +156 -156
 - package/server/templates/form/user_group_access.form.json +22 -22
 - package/server/templates/select/account_id.json +2 -2
 - package/server/templates/table/gis.dataset.table.json +43 -43
 - package/server/templates/table/management.user_group.table.json +112 -112
 - package/server/templates/table/management.users.table.json +126 -126
 - package/server/utils/addNotification.js +21 -21
 - package/server/utils/sendNotification.js +89 -89
 
| 
         @@ -1,48 +1,48 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import { getPG, getSelect } from "@opengis/fastify-table/utils.js";
         
     | 
| 
       2 
     | 
    
         
            -
            const pg = getPG();
         
     | 
| 
       3 
     | 
    
         
            -
             
     | 
| 
       4 
     | 
    
         
            -
            export default async function select(ids, options) {
         
     | 
| 
       5 
     | 
    
         
            -
              if (!ids || (Array.isArray(ids) && ids.length === 0) || ids === '') {
         
     | 
| 
       6 
     | 
    
         
            -
                return '';
         
     | 
| 
       7 
     | 
    
         
            -
              }
         
     | 
| 
       8 
     | 
    
         
            -
             
     | 
| 
       9 
     | 
    
         
            -
              const data = options.hash?.data;
         
     | 
| 
       10 
     | 
    
         
            -
              if (!data && data !== false && data !== 0) return '';
         
     | 
| 
       11 
     | 
    
         
            -
              try {
         
     | 
| 
       12 
     | 
    
         
            -
                const idsArray = Array.isArray(ids) ? ids : [ids];
         
     | 
| 
       13 
     | 
    
         
            -
             
     | 
| 
       14 
     | 
    
         
            -
                const classifier = await getSelect(data);
         
     | 
| 
       15 
     | 
    
         
            -
             
     | 
| 
       16 
     | 
    
         
            -
             
     | 
| 
       17 
     | 
    
         
            -
             
     | 
| 
       18 
     | 
    
         
            -
                if (!classifier) return `Не знайдено класифікатор ${data}`;
         
     | 
| 
       19 
     | 
    
         
            -
             
     | 
| 
       20 
     | 
    
         
            -
                const arr = classifier.arr || []
         
     | 
| 
       21 
     | 
    
         
            -
                if (classifier.sql && typeof classifier.sql === 'string') {
         
     | 
| 
       22 
     | 
    
         
            -
             
     | 
| 
       23 
     | 
    
         
            -
             
     | 
| 
       24 
     | 
    
         
            -
             
     | 
| 
       25 
     | 
    
         
            -
                  const metaQuery = `SELECT * FROM (${classifier.sql})q LIMIT 0`;
         
     | 
| 
       26 
     | 
    
         
            -
                  const meta = await pg.query(metaQuery);
         
     | 
| 
       27 
     | 
    
         
            -
                  const idColumn = meta.fields[0].name;
         
     | 
| 
       28 
     | 
    
         
            -
                  const textColumn = meta.fields[1].name;
         
     | 
| 
       29 
     | 
    
         
            -
             
     | 
| 
       30 
     | 
    
         
            -
                  const q = `SELECT "${idColumn}" AS id, "${textColumn}" AS text FROM (${classifier.sql}) q WHERE "${idColumn}"::text = ANY($1::text[])`;
         
     | 
| 
       31 
     | 
    
         
            -
                  const values = [idsArray.map(id => String(id))];
         
     | 
| 
       32 
     | 
    
         
            -
             
     | 
| 
       33 
     | 
    
         
            -
                  const { rows } = await pg.query(q, values);
         
     | 
| 
       34 
     | 
    
         
            -
                  Object.assign(arr, rows);
         
     | 
| 
       35 
     | 
    
         
            -
             
     | 
| 
       36 
     | 
    
         
            -
                }
         
     | 
| 
       37 
     | 
    
         
            -
                if (!arr.length) return idsArray;
         
     | 
| 
       38 
     | 
    
         
            -
             
     | 
| 
       39 
     | 
    
         
            -
                const results = idsArray.map(id => {
         
     | 
| 
       40 
     | 
    
         
            -
                  const result = arr.find(el => String(el.id) == String(id));
         
     | 
| 
       41 
     | 
    
         
            -
                  return result ? result.text : '';
         
     | 
| 
       42 
     | 
    
         
            -
                });
         
     | 
| 
       43 
     | 
    
         
            -
             
     | 
| 
       44 
     | 
    
         
            -
                return results.filter(Boolean).join(', ');
         
     | 
| 
       45 
     | 
    
         
            -
              } catch (err) {
         
     | 
| 
       46 
     | 
    
         
            -
                return `Сталася помилка.<!-- err: ${err.toString()} -->`;
         
     | 
| 
       47 
     | 
    
         
            -
              }
         
     | 
| 
       48 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            import { getPG, getSelect } from "@opengis/fastify-table/utils.js";
         
     | 
| 
      
 2 
     | 
    
         
            +
            const pg = getPG();
         
     | 
| 
      
 3 
     | 
    
         
            +
             
     | 
| 
      
 4 
     | 
    
         
            +
            export default async function select(ids, options) {
         
     | 
| 
      
 5 
     | 
    
         
            +
              if (!ids || (Array.isArray(ids) && ids.length === 0) || ids === '') {
         
     | 
| 
      
 6 
     | 
    
         
            +
                return '';
         
     | 
| 
      
 7 
     | 
    
         
            +
              }
         
     | 
| 
      
 8 
     | 
    
         
            +
             
     | 
| 
      
 9 
     | 
    
         
            +
              const data = options.hash?.data;
         
     | 
| 
      
 10 
     | 
    
         
            +
              if (!data && data !== false && data !== 0) return '';
         
     | 
| 
      
 11 
     | 
    
         
            +
              try {
         
     | 
| 
      
 12 
     | 
    
         
            +
                const idsArray = Array.isArray(ids) ? ids : [ids];
         
     | 
| 
      
 13 
     | 
    
         
            +
             
     | 
| 
      
 14 
     | 
    
         
            +
                const classifier = await getSelect(data);
         
     | 
| 
      
 15 
     | 
    
         
            +
             
     | 
| 
      
 16 
     | 
    
         
            +
             
     | 
| 
      
 17 
     | 
    
         
            +
             
     | 
| 
      
 18 
     | 
    
         
            +
                if (!classifier) return `Не знайдено класифікатор ${data}`;
         
     | 
| 
      
 19 
     | 
    
         
            +
             
     | 
| 
      
 20 
     | 
    
         
            +
                const arr = classifier.arr || []
         
     | 
| 
      
 21 
     | 
    
         
            +
                if (classifier.sql && typeof classifier.sql === 'string') {
         
     | 
| 
      
 22 
     | 
    
         
            +
             
     | 
| 
      
 23 
     | 
    
         
            +
             
     | 
| 
      
 24 
     | 
    
         
            +
             
     | 
| 
      
 25 
     | 
    
         
            +
                  const metaQuery = `SELECT * FROM (${classifier.sql})q LIMIT 0`;
         
     | 
| 
      
 26 
     | 
    
         
            +
                  const meta = await pg.query(metaQuery);
         
     | 
| 
      
 27 
     | 
    
         
            +
                  const idColumn = meta.fields[0].name;
         
     | 
| 
      
 28 
     | 
    
         
            +
                  const textColumn = meta.fields[1].name;
         
     | 
| 
      
 29 
     | 
    
         
            +
             
     | 
| 
      
 30 
     | 
    
         
            +
                  const q = `SELECT "${idColumn}" AS id, "${textColumn}" AS text FROM (${classifier.sql}) q WHERE "${idColumn}"::text = ANY($1::text[])`;
         
     | 
| 
      
 31 
     | 
    
         
            +
                  const values = [idsArray.map(id => String(id))];
         
     | 
| 
      
 32 
     | 
    
         
            +
             
     | 
| 
      
 33 
     | 
    
         
            +
                  const { rows } = await pg.query(q, values);
         
     | 
| 
      
 34 
     | 
    
         
            +
                  Object.assign(arr, rows);
         
     | 
| 
      
 35 
     | 
    
         
            +
             
     | 
| 
      
 36 
     | 
    
         
            +
                }
         
     | 
| 
      
 37 
     | 
    
         
            +
                if (!arr.length) return idsArray;
         
     | 
| 
      
 38 
     | 
    
         
            +
             
     | 
| 
      
 39 
     | 
    
         
            +
                const results = idsArray.map(id => {
         
     | 
| 
      
 40 
     | 
    
         
            +
                  const result = arr.find(el => String(el.id) == String(id));
         
     | 
| 
      
 41 
     | 
    
         
            +
                  return result ? result.text : '';
         
     | 
| 
      
 42 
     | 
    
         
            +
                });
         
     | 
| 
      
 43 
     | 
    
         
            +
             
     | 
| 
      
 44 
     | 
    
         
            +
                return results.filter(Boolean).join(', ');
         
     | 
| 
      
 45 
     | 
    
         
            +
              } catch (err) {
         
     | 
| 
      
 46 
     | 
    
         
            +
                return `Сталася помилка.<!-- err: ${err.toString()} -->`;
         
     | 
| 
      
 47 
     | 
    
         
            +
              }
         
     | 
| 
      
 48 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,19 +1,19 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import { setToken } from "@opengis/fastify-table/utils.js";
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            export default function token(params) {
         
     | 
| 
       4 
     | 
    
         
            -
             
     | 
| 
       5 
     | 
    
         
            -
                const { data, hash } = params;
         
     | 
| 
       6 
     | 
    
         
            -
             
     | 
| 
       7 
     | 
    
         
            -
                if (!data?.root?.user?.uid && !hash.uid) return '-';
         
     | 
| 
       8 
     | 
    
         
            -
                if (!hash || typeof hash !== 'object') return '-';
         
     | 
| 
       9 
     | 
    
         
            -
             
     | 
| 
       10 
     | 
    
         
            -
                // const id = hash?.edit ? hash?.id : data?.root?.id;
         
     | 
| 
       11 
     | 
    
         
            -
                // console.log(hash)
         
     | 
| 
       12 
     | 
    
         
            -
                const [token] = setToken({
         
     | 
| 
       13 
     | 
    
         
            -
                    ids: [JSON.stringify(hash)],
         
     | 
| 
       14 
     | 
    
         
            -
             
     | 
| 
       15 
     | 
    
         
            -
                    uid: data?.root?.user?.uid || hash.uid,
         
     | 
| 
       16 
     | 
    
         
            -
                    array: 1,
         
     | 
| 
       17 
     | 
    
         
            -
                });
         
     | 
| 
       18 
     | 
    
         
            -
                return token;
         
     | 
| 
      
 1 
     | 
    
         
            +
            import { setToken } from "@opengis/fastify-table/utils.js";
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            export default function token(params) {
         
     | 
| 
      
 4 
     | 
    
         
            +
             
     | 
| 
      
 5 
     | 
    
         
            +
                const { data, hash } = params;
         
     | 
| 
      
 6 
     | 
    
         
            +
             
     | 
| 
      
 7 
     | 
    
         
            +
                if (!data?.root?.user?.uid && !hash.uid) return '-';
         
     | 
| 
      
 8 
     | 
    
         
            +
                if (!hash || typeof hash !== 'object') return '-';
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
| 
      
 10 
     | 
    
         
            +
                // const id = hash?.edit ? hash?.id : data?.root?.id;
         
     | 
| 
      
 11 
     | 
    
         
            +
                // console.log(hash)
         
     | 
| 
      
 12 
     | 
    
         
            +
                const [token] = setToken({
         
     | 
| 
      
 13 
     | 
    
         
            +
                    ids: [JSON.stringify(hash)],
         
     | 
| 
      
 14 
     | 
    
         
            +
             
     | 
| 
      
 15 
     | 
    
         
            +
                    uid: data?.root?.user?.uid || hash.uid,
         
     | 
| 
      
 16 
     | 
    
         
            +
                    array: 1,
         
     | 
| 
      
 17 
     | 
    
         
            +
                });
         
     | 
| 
      
 18 
     | 
    
         
            +
                return token;
         
     | 
| 
       19 
19 
     | 
    
         
             
            }
         
     | 
| 
         @@ -1,21 +1,21 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import buttonAdd from "../utils/buttonAdd.js";
         
     | 
| 
       2 
     | 
    
         
            -
            import buttonDel from "../utils/buttonDel.js";
         
     | 
| 
       3 
     | 
    
         
            -
            import buttonEdit from "../utils/buttonEdit.js";
         
     | 
| 
       4 
     | 
    
         
            -
            import button from "../utils/button.js";
         
     | 
| 
       5 
     | 
    
         
            -
            export default function buttonHelper(data, opt) {
         
     | 
| 
       6 
     | 
    
         
            -
              const { hash } = opt;
         
     | 
| 
       7 
     | 
    
         
            -
             
     | 
| 
       8 
     | 
    
         
            -
              // console.log(params)
         
     | 
| 
       9 
     | 
    
         
            -
             
     | 
| 
       10 
     | 
    
         
            -
              if (!hash.token) return "token empty";
         
     | 
| 
       11 
     | 
    
         
            -
              if (hash.add) {
         
     | 
| 
       12 
     | 
    
         
            -
                return buttonAdd(hash.token, hash.title);
         
     | 
| 
       13 
     | 
    
         
            -
              }
         
     | 
| 
       14 
     | 
    
         
            -
              if (hash.del) {
         
     | 
| 
       15 
     | 
    
         
            -
                return buttonDel(hash.token, hash.title);
         
     | 
| 
       16 
     | 
    
         
            -
              }
         
     | 
| 
       17 
     | 
    
         
            -
              if (hash.edit) {
         
     | 
| 
       18 
     | 
    
         
            -
                return buttonEdit(hash.token, hash.title);
         
     | 
| 
       19 
     | 
    
         
            -
              }
         
     | 
| 
       20 
     | 
    
         
            -
              return button(hash.token, hash.title);
         
     | 
| 
       21 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            import buttonAdd from "../utils/buttonAdd.js";
         
     | 
| 
      
 2 
     | 
    
         
            +
            import buttonDel from "../utils/buttonDel.js";
         
     | 
| 
      
 3 
     | 
    
         
            +
            import buttonEdit from "../utils/buttonEdit.js";
         
     | 
| 
      
 4 
     | 
    
         
            +
            import button from "../utils/button.js";
         
     | 
| 
      
 5 
     | 
    
         
            +
            export default function buttonHelper(data, opt) {
         
     | 
| 
      
 6 
     | 
    
         
            +
              const { hash } = opt;
         
     | 
| 
      
 7 
     | 
    
         
            +
             
     | 
| 
      
 8 
     | 
    
         
            +
              // console.log(params)
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
| 
      
 10 
     | 
    
         
            +
              if (!hash.token) return "token empty";
         
     | 
| 
      
 11 
     | 
    
         
            +
              if (hash.add) {
         
     | 
| 
      
 12 
     | 
    
         
            +
                return buttonAdd(hash.token, hash.title);
         
     | 
| 
      
 13 
     | 
    
         
            +
              }
         
     | 
| 
      
 14 
     | 
    
         
            +
              if (hash.del) {
         
     | 
| 
      
 15 
     | 
    
         
            +
                return buttonDel(hash.token, hash.title);
         
     | 
| 
      
 16 
     | 
    
         
            +
              }
         
     | 
| 
      
 17 
     | 
    
         
            +
              if (hash.edit) {
         
     | 
| 
      
 18 
     | 
    
         
            +
                return buttonEdit(hash.token, hash.title);
         
     | 
| 
      
 19 
     | 
    
         
            +
              }
         
     | 
| 
      
 20 
     | 
    
         
            +
              return button(hash.token, hash.title);
         
     | 
| 
      
 21 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,6 +1,6 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
             
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            export default function button(token, title) {
         
     | 
| 
       4 
     | 
    
         
            -
                return `<button onclick="window.v3plugin.$form({ token: '${token}' })"
         
     | 
| 
       5 
     | 
    
         
            -
                class="inline-flex items-center px-2 py-1 text-sm font-medium text-white duration-300 bg-blue-600 border border-transparent rounded-lg gap-x-2 hover:bg-blue-700 hover:text-white">${title || 'Додати'}</button>`;
         
     | 
| 
      
 1 
     | 
    
         
            +
             
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            export default function button(token, title) {
         
     | 
| 
      
 4 
     | 
    
         
            +
                return `<button onclick="window.v3plugin.$form({ token: '${token}' })"
         
     | 
| 
      
 5 
     | 
    
         
            +
                class="inline-flex items-center px-2 py-1 text-sm font-medium text-white duration-300 bg-blue-600 border border-transparent rounded-lg gap-x-2 hover:bg-blue-700 hover:text-white">${title || 'Додати'}</button>`;
         
     | 
| 
       6 
6 
     | 
    
         
             
            }
         
     | 
| 
         @@ -1,58 +1,58 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
             
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            import { getPG, handlebars } from '@opengis/fastify-table/utils.js';
         
     | 
| 
       4 
     | 
    
         
            -
             
     | 
| 
       5 
     | 
    
         
            -
            const maxLimit = 100;
         
     | 
| 
       6 
     | 
    
         
            -
             
     | 
| 
       7 
     | 
    
         
            -
            /**
         
     | 
| 
       8 
     | 
    
         
            -
             * Відображення даних з таблиці або запиту до БД на сторінці. За запитом отримуємо масив json із рядків бази даних.
         
     | 
| 
       9 
     | 
    
         
            -
             * Є можливість застосування синтаксису sql у змінній query для формування запиту до БД.
         
     | 
| 
       10 
     | 
    
         
            -
             *
         
     | 
| 
       11 
     | 
    
         
            -
             * @summary Відображення контенту на сторінці. Є можливість застосування синтаксису sql у змінній query.
         
     | 
| 
       12 
     | 
    
         
            -
             * @priority 5
         
     | 
| 
       13 
     | 
    
         
            -
             * @type helper
         
     | 
| 
       14 
     | 
    
         
            -
             * @alias contentList
         
     | 
| 
       15 
     | 
    
         
            -
             * @example
         
     | 
| 
       16 
     | 
    
         
            -
             * {{#contentList table="help.doc_function" query="type='api'" sql1=1 limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
       17 
     | 
    
         
            -
             * @example
         
     | 
| 
       18 
     | 
    
         
            -
             * {{#contentList table="help.doc_function" sql1=1 query="name like '%form%'" limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
       19 
     | 
    
         
            -
             * @example
         
     | 
| 
       20 
     | 
    
         
            -
             * {{#contentList table="help.article" sql=1 query="module='CORE'" limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
       21 
     | 
    
         
            -
             * @param {String} table Таблиця в базі або конфіг таблиця
         
     | 
| 
       22 
     | 
    
         
            -
             * @param {String} query Запит до бази
         
     | 
| 
       23 
     | 
    
         
            -
             * @param {Number} limit Кількість рядків на сторінці
         
     | 
| 
       24 
     | 
    
         
            -
             * @param {String} sql Вивід sql запиту
         
     | 
| 
       25 
     | 
    
         
            -
             * @returns {String} Returns HTML
         
     | 
| 
       26 
     | 
    
         
            -
             */
         
     | 
| 
       27 
     | 
    
         
            -
            export default async function contentList(options) {
         
     | 
| 
       28 
     | 
    
         
            -
                const { table, limit, query, order, sql, debug } = options.hash;
         
     | 
| 
       29 
     | 
    
         
            -
                if (!table) { return 'Table undefined'; }
         
     | 
| 
       30 
     | 
    
         
            -
             
     | 
| 
       31 
     | 
    
         
            -
                try {
         
     | 
| 
       32 
     | 
    
         
            -
                    const pg = getPG();
         
     | 
| 
       33 
     | 
    
         
            -
             
     | 
| 
       34 
     | 
    
         
            -
                    const hasBrackets = table.trim().startsWith('(') && table.trim().endsWith(')');
         
     | 
| 
       35 
     | 
    
         
            -
             
     | 
| 
       36 
     | 
    
         
            -
                    const where = `where ${query ? query : '1=1'}`;
         
     | 
| 
       37 
     | 
    
         
            -
                    const _limit = limit !== undefined && limit !== null ? Math.min(maxLimit, +limit) : 15;
         
     | 
| 
       38 
     | 
    
         
            -
                    const _order = order ? `order by ${order}` : '';
         
     | 
| 
       39 
     | 
    
         
            -
             
     | 
| 
       40 
     | 
    
         
            -
                    const SQL = `select *,${pg.pk[table] || '1'}::text from ${hasBrackets ? table + ' t' : table} ${where} ${_order} limit ${_limit}`;
         
     | 
| 
       41 
     | 
    
         
            -
                    const compiledSQL = SQL.includes('{{') ? await handlebars.compile(SQL)({ ...options.data.root, hash: options.hash, opt: options.hash }) : SQL;
         
     | 
| 
       42 
     | 
    
         
            -
             
     | 
| 
       43 
     | 
    
         
            -
                    if (sql) {
         
     | 
| 
       44 
     | 
    
         
            -
                        return compiledSQL.replaceAll('&', '&').replaceAll('<', '<').replaceAll('>', '>').replaceAll('"', '"')
         
     | 
| 
       45 
     | 
    
         
            -
                            .replaceAll("'", ''');
         
     | 
| 
       46 
     | 
    
         
            -
                    }
         
     | 
| 
       47 
     | 
    
         
            -
                    const { rows } = await pg.query(compiledSQL);
         
     | 
| 
       48 
     | 
    
         
            -
                    const data = { rows, total: rows.length, ...options.data?.root };
         
     | 
| 
       49 
     | 
    
         
            -
             
     | 
| 
       50 
     | 
    
         
            -
                    if (debug) {
         
     | 
| 
       51 
     | 
    
         
            -
                        return JSON.stringify(data, null, 2);
         
     | 
| 
       52 
     | 
    
         
            -
                    }
         
     | 
| 
       53 
     | 
    
         
            -
             
     | 
| 
       54 
     | 
    
         
            -
                    return options.fn(data);
         
     | 
| 
       55 
     | 
    
         
            -
                } catch (err) {
         
     | 
| 
       56 
     | 
    
         
            -
                    return `Сталася помилка, зверніться до відділу підтримки.<!-- err: ${err.toString()} -->`;
         
     | 
| 
       57 
     | 
    
         
            -
                }
         
     | 
| 
       58 
     | 
    
         
            -
            };
         
     | 
| 
      
 1 
     | 
    
         
            +
             
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            import { getPG, handlebars } from '@opengis/fastify-table/utils.js';
         
     | 
| 
      
 4 
     | 
    
         
            +
             
     | 
| 
      
 5 
     | 
    
         
            +
            const maxLimit = 100;
         
     | 
| 
      
 6 
     | 
    
         
            +
             
     | 
| 
      
 7 
     | 
    
         
            +
            /**
         
     | 
| 
      
 8 
     | 
    
         
            +
             * Відображення даних з таблиці або запиту до БД на сторінці. За запитом отримуємо масив json із рядків бази даних.
         
     | 
| 
      
 9 
     | 
    
         
            +
             * Є можливість застосування синтаксису sql у змінній query для формування запиту до БД.
         
     | 
| 
      
 10 
     | 
    
         
            +
             *
         
     | 
| 
      
 11 
     | 
    
         
            +
             * @summary Відображення контенту на сторінці. Є можливість застосування синтаксису sql у змінній query.
         
     | 
| 
      
 12 
     | 
    
         
            +
             * @priority 5
         
     | 
| 
      
 13 
     | 
    
         
            +
             * @type helper
         
     | 
| 
      
 14 
     | 
    
         
            +
             * @alias contentList
         
     | 
| 
      
 15 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 16 
     | 
    
         
            +
             * {{#contentList table="help.doc_function" query="type='api'" sql1=1 limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
      
 17 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 18 
     | 
    
         
            +
             * {{#contentList table="help.doc_function" sql1=1 query="name like '%form%'" limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
      
 19 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 20 
     | 
    
         
            +
             * {{#contentList table="help.article" sql=1 query="module='CORE'" limit=1}}{{#each rows}}{{{JSON 2 this}}}{{/each}}{{/contentList}}
         
     | 
| 
      
 21 
     | 
    
         
            +
             * @param {String} table Таблиця в базі або конфіг таблиця
         
     | 
| 
      
 22 
     | 
    
         
            +
             * @param {String} query Запит до бази
         
     | 
| 
      
 23 
     | 
    
         
            +
             * @param {Number} limit Кількість рядків на сторінці
         
     | 
| 
      
 24 
     | 
    
         
            +
             * @param {String} sql Вивід sql запиту
         
     | 
| 
      
 25 
     | 
    
         
            +
             * @returns {String} Returns HTML
         
     | 
| 
      
 26 
     | 
    
         
            +
             */
         
     | 
| 
      
 27 
     | 
    
         
            +
            export default async function contentList(options) {
         
     | 
| 
      
 28 
     | 
    
         
            +
                const { table, limit, query, order, sql, debug } = options.hash;
         
     | 
| 
      
 29 
     | 
    
         
            +
                if (!table) { return 'Table undefined'; }
         
     | 
| 
      
 30 
     | 
    
         
            +
             
     | 
| 
      
 31 
     | 
    
         
            +
                try {
         
     | 
| 
      
 32 
     | 
    
         
            +
                    const pg = getPG();
         
     | 
| 
      
 33 
     | 
    
         
            +
             
     | 
| 
      
 34 
     | 
    
         
            +
                    const hasBrackets = table.trim().startsWith('(') && table.trim().endsWith(')');
         
     | 
| 
      
 35 
     | 
    
         
            +
             
     | 
| 
      
 36 
     | 
    
         
            +
                    const where = `where ${query ? query : '1=1'}`;
         
     | 
| 
      
 37 
     | 
    
         
            +
                    const _limit = limit !== undefined && limit !== null ? Math.min(maxLimit, +limit) : 15;
         
     | 
| 
      
 38 
     | 
    
         
            +
                    const _order = order ? `order by ${order}` : '';
         
     | 
| 
      
 39 
     | 
    
         
            +
             
     | 
| 
      
 40 
     | 
    
         
            +
                    const SQL = `select *,${pg.pk[table] || '1'}::text from ${hasBrackets ? table + ' t' : table} ${where} ${_order} limit ${_limit}`;
         
     | 
| 
      
 41 
     | 
    
         
            +
                    const compiledSQL = SQL.includes('{{') ? await handlebars.compile(SQL)({ ...options.data.root, hash: options.hash, opt: options.hash }) : SQL;
         
     | 
| 
      
 42 
     | 
    
         
            +
             
     | 
| 
      
 43 
     | 
    
         
            +
                    if (sql) {
         
     | 
| 
      
 44 
     | 
    
         
            +
                        return compiledSQL.replaceAll('&', '&').replaceAll('<', '<').replaceAll('>', '>').replaceAll('"', '"')
         
     | 
| 
      
 45 
     | 
    
         
            +
                            .replaceAll("'", ''');
         
     | 
| 
      
 46 
     | 
    
         
            +
                    }
         
     | 
| 
      
 47 
     | 
    
         
            +
                    const { rows } = await pg.query(compiledSQL);
         
     | 
| 
      
 48 
     | 
    
         
            +
                    const data = { rows, total: rows.length, ...options.data?.root };
         
     | 
| 
      
 49 
     | 
    
         
            +
             
     | 
| 
      
 50 
     | 
    
         
            +
                    if (debug) {
         
     | 
| 
      
 51 
     | 
    
         
            +
                        return JSON.stringify(data, null, 2);
         
     | 
| 
      
 52 
     | 
    
         
            +
                    }
         
     | 
| 
      
 53 
     | 
    
         
            +
             
     | 
| 
      
 54 
     | 
    
         
            +
                    return options.fn(data);
         
     | 
| 
      
 55 
     | 
    
         
            +
                } catch (err) {
         
     | 
| 
      
 56 
     | 
    
         
            +
                    return `Сталася помилка, зверніться до відділу підтримки.<!-- err: ${err.toString()} -->`;
         
     | 
| 
      
 57 
     | 
    
         
            +
                }
         
     | 
| 
      
 58 
     | 
    
         
            +
            };
         
     | 
| 
         @@ -1,101 +1,101 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            /**
         
     | 
| 
       2 
     | 
    
         
            -
             * Перетинає два масиви
         
     | 
| 
       3 
     | 
    
         
            -
             *
         
     | 
| 
       4 
     | 
    
         
            -
             * @example
         
     | 
| 
       5 
     | 
    
         
            -
             * // returns [1, 4, 5, 6]
         
     | 
| 
       6 
     | 
    
         
            -
             * intersect([1,2,3,4,5,6],[1,4,5,6,7,8,9,11])
         
     | 
| 
       7 
     | 
    
         
            -
             * @param {Array} a
         
     | 
| 
       8 
     | 
    
         
            -
             * @param {Array} b
         
     | 
| 
       9 
     | 
    
         
            -
             * @returns {Array} Returns new intersect array
         
     | 
| 
       10 
     | 
    
         
            -
             */
         
     | 
| 
       11 
     | 
    
         
            -
            function intersect(a, b) {
         
     | 
| 
       12 
     | 
    
         
            -
              let aN = a; let bN = b;
         
     | 
| 
       13 
     | 
    
         
            -
              if (b.length > a.length) {
         
     | 
| 
       14 
     | 
    
         
            -
                [aN, bN] = [bN, aN];
         
     | 
| 
       15 
     | 
    
         
            -
              }
         
     | 
| 
       16 
     | 
    
         
            -
              return aN.filter((e) => bN.includes(e));
         
     | 
| 
       17 
     | 
    
         
            -
            }
         
     | 
| 
       18 
     | 
    
         
            -
             
     | 
| 
       19 
     | 
    
         
            -
            /**
         
     | 
| 
       20 
     | 
    
         
            -
             * Створення шаблона або його частини внаслідок перевірки значення із веб-запиту та заздалегідь прописаного значення.
         
     | 
| 
       21 
     | 
    
         
            -
             * Дозволяє змінювати наповнення сторінки через ряд перевірок. Є можливість внесення додаткової умови - що робити, коли умова не виконується.
         
     | 
| 
       22 
     | 
    
         
            -
             *
         
     | 
| 
       23 
     | 
    
         
            -
             * @summary Перевірка двох значень та виконання коду при виконанні умови, а також у всіх інших випадках.
         
     | 
| 
       24 
     | 
    
         
            -
             * @priority 5
         
     | 
| 
       25 
     | 
    
         
            -
             * @alias ifCond
         
     | 
| 
       26 
     | 
    
         
            -
             * @type helper
         
     | 
| 
       27 
     | 
    
         
            -
             * @tag condition
         
     | 
| 
       28 
     | 
    
         
            -
             * @example
         
     | 
| 
       29 
     | 
    
         
            -
             * {{#ifCond @root.req.domain 'in' 'help.softpro.ua,123'}} {{select user.uid data="get_full_uid"}} {{^}} Умова не виконана {{/ifCond}}
         
     | 
| 
       30 
     | 
    
         
            -
             * @example
         
     | 
| 
       31 
     | 
    
         
            -
             * {{#ifCond "1234567890" 'in' @root.user.group_list}} 1=1 {{^}} uid='{{uid}}' {{/ifCond}}
         
     | 
| 
       32 
     | 
    
         
            -
             * @example
         
     | 
| 
       33 
     | 
    
         
            -
             * {{#ifCond 'debug' 'in' @root.setting.core.setting}}Умова виконана{{^}}Не виконана умова{{/ifCond}}
         
     | 
| 
       34 
     | 
    
         
            -
             * @param {Array} args Параметри для значень і умов
         
     | 
| 
       35 
     | 
    
         
            -
             * @param {Array} args[0]] Перше значення
         
     | 
| 
       36 
     | 
    
         
            -
             * @param {Array} args[1]] Оператор
         
     | 
| 
       37 
     | 
    
         
            -
             * @param {Array} args[2]] Друге значення
         
     | 
| 
       38 
     | 
    
         
            -
             * @returns {String} Returns HTML
         
     | 
| 
       39 
     | 
    
         
            -
             */
         
     | 
| 
       40 
     | 
    
         
            -
            export default function ifCond(v1, operator, v2, options) {
         
     | 
| 
       41 
     | 
    
         
            -
              const __obj = this;
         
     | 
| 
       42 
     | 
    
         
            -
             
     | 
| 
       43 
     | 
    
         
            -
              switch (operator) {
         
     | 
| 
       44 
     | 
    
         
            -
                case '==':
         
     | 
| 
       45 
     | 
    
         
            -
                  return (v1 == v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       46 
     | 
    
         
            -
                case '!=':
         
     | 
| 
       47 
     | 
    
         
            -
                  return (v1 != v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       48 
     | 
    
         
            -
                case '===':
         
     | 
| 
       49 
     | 
    
         
            -
                  return (v1 === v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       50 
     | 
    
         
            -
                case '!==':
         
     | 
| 
       51 
     | 
    
         
            -
                  return (v1 !== v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       52 
     | 
    
         
            -
                case '&&':
         
     | 
| 
       53 
     | 
    
         
            -
                  return (v1 && v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       54 
     | 
    
         
            -
                case '||':
         
     | 
| 
       55 
     | 
    
         
            -
                  return (v1 || v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       56 
     | 
    
         
            -
                case '<':
         
     | 
| 
       57 
     | 
    
         
            -
                  return (v1 < v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       58 
     | 
    
         
            -
                case '<=':
         
     | 
| 
       59 
     | 
    
         
            -
                  return (v1 <= v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       60 
     | 
    
         
            -
                case '>':
         
     | 
| 
       61 
     | 
    
         
            -
                  return (v1 > v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       62 
     | 
    
         
            -
                case '>=':
         
     | 
| 
       63 
     | 
    
         
            -
                  return (v1 >= v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
       64 
     | 
    
         
            -
                case '&':
         
     | 
| 
       65 
     | 
    
         
            -
                  return intersect(v1, v2).length !== 0
         
     | 
| 
       66 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       67 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       68 
     | 
    
         
            -
                case '!~':
         
     | 
| 
       69 
     | 
    
         
            -
                  return (v1 || '').indexOf(v2) === -1
         
     | 
| 
       70 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       71 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       72 
     | 
    
         
            -
                case '~':
         
     | 
| 
       73 
     | 
    
         
            -
                  return (v1 || '').indexOf(v2) !== -1
         
     | 
| 
       74 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       75 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       76 
     | 
    
         
            -
                case 'period':
         
     | 
| 
       77 
     | 
    
         
            -
                  return (new Date(v1) < new Date() && new Date(v2) > new Date())
         
     | 
| 
       78 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       79 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       80 
     | 
    
         
            -
                case 'in': {
         
     | 
| 
       81 
     | 
    
         
            -
                  if (typeof v2 === 'string') v2 = v2.split(',').map(item => item.trim());
         
     | 
| 
       82 
     | 
    
         
            -
             
     | 
| 
       83 
     | 
    
         
            -
                  if (Array.isArray(v1)) {
         
     | 
| 
       84 
     | 
    
         
            -
                    return v1.some((value) => v2.includes(value.toString()))
         
     | 
| 
       85 
     | 
    
         
            -
                      ? options.fn(__obj)
         
     | 
| 
       86 
     | 
    
         
            -
                      : options.inverse(__obj);
         
     | 
| 
       87 
     | 
    
         
            -
                  }
         
     | 
| 
       88 
     | 
    
         
            -
                  return v2.includes(v1?.toString())
         
     | 
| 
       89 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       90 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       91 
     | 
    
         
            -
                }
         
     | 
| 
       92 
     | 
    
         
            -
                case 'not in': {
         
     | 
| 
       93 
     | 
    
         
            -
                  if (typeof v2 === 'string') v2 = v2.split(',').map(item => item.trim());
         
     | 
| 
       94 
     | 
    
         
            -
                  return !v2.includes(v1?.toString())
         
     | 
| 
       95 
     | 
    
         
            -
                    ? options.fn(__obj)
         
     | 
| 
       96 
     | 
    
         
            -
                    : options.inverse(__obj);
         
     | 
| 
       97 
     | 
    
         
            -
                }
         
     | 
| 
       98 
     | 
    
         
            -
                default:
         
     | 
| 
       99 
     | 
    
         
            -
                  return options.inverse(__obj);
         
     | 
| 
       100 
     | 
    
         
            -
              }
         
     | 
| 
       101 
     | 
    
         
            -
            }
         
     | 
| 
      
 1 
     | 
    
         
            +
            /**
         
     | 
| 
      
 2 
     | 
    
         
            +
             * Перетинає два масиви
         
     | 
| 
      
 3 
     | 
    
         
            +
             *
         
     | 
| 
      
 4 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 5 
     | 
    
         
            +
             * // returns [1, 4, 5, 6]
         
     | 
| 
      
 6 
     | 
    
         
            +
             * intersect([1,2,3,4,5,6],[1,4,5,6,7,8,9,11])
         
     | 
| 
      
 7 
     | 
    
         
            +
             * @param {Array} a
         
     | 
| 
      
 8 
     | 
    
         
            +
             * @param {Array} b
         
     | 
| 
      
 9 
     | 
    
         
            +
             * @returns {Array} Returns new intersect array
         
     | 
| 
      
 10 
     | 
    
         
            +
             */
         
     | 
| 
      
 11 
     | 
    
         
            +
            function intersect(a, b) {
         
     | 
| 
      
 12 
     | 
    
         
            +
              let aN = a; let bN = b;
         
     | 
| 
      
 13 
     | 
    
         
            +
              if (b.length > a.length) {
         
     | 
| 
      
 14 
     | 
    
         
            +
                [aN, bN] = [bN, aN];
         
     | 
| 
      
 15 
     | 
    
         
            +
              }
         
     | 
| 
      
 16 
     | 
    
         
            +
              return aN.filter((e) => bN.includes(e));
         
     | 
| 
      
 17 
     | 
    
         
            +
            }
         
     | 
| 
      
 18 
     | 
    
         
            +
             
     | 
| 
      
 19 
     | 
    
         
            +
            /**
         
     | 
| 
      
 20 
     | 
    
         
            +
             * Створення шаблона або його частини внаслідок перевірки значення із веб-запиту та заздалегідь прописаного значення.
         
     | 
| 
      
 21 
     | 
    
         
            +
             * Дозволяє змінювати наповнення сторінки через ряд перевірок. Є можливість внесення додаткової умови - що робити, коли умова не виконується.
         
     | 
| 
      
 22 
     | 
    
         
            +
             *
         
     | 
| 
      
 23 
     | 
    
         
            +
             * @summary Перевірка двох значень та виконання коду при виконанні умови, а також у всіх інших випадках.
         
     | 
| 
      
 24 
     | 
    
         
            +
             * @priority 5
         
     | 
| 
      
 25 
     | 
    
         
            +
             * @alias ifCond
         
     | 
| 
      
 26 
     | 
    
         
            +
             * @type helper
         
     | 
| 
      
 27 
     | 
    
         
            +
             * @tag condition
         
     | 
| 
      
 28 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 29 
     | 
    
         
            +
             * {{#ifCond @root.req.domain 'in' 'help.softpro.ua,123'}} {{select user.uid data="get_full_uid"}} {{^}} Умова не виконана {{/ifCond}}
         
     | 
| 
      
 30 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 31 
     | 
    
         
            +
             * {{#ifCond "1234567890" 'in' @root.user.group_list}} 1=1 {{^}} uid='{{uid}}' {{/ifCond}}
         
     | 
| 
      
 32 
     | 
    
         
            +
             * @example
         
     | 
| 
      
 33 
     | 
    
         
            +
             * {{#ifCond 'debug' 'in' @root.setting.core.setting}}Умова виконана{{^}}Не виконана умова{{/ifCond}}
         
     | 
| 
      
 34 
     | 
    
         
            +
             * @param {Array} args Параметри для значень і умов
         
     | 
| 
      
 35 
     | 
    
         
            +
             * @param {Array} args[0]] Перше значення
         
     | 
| 
      
 36 
     | 
    
         
            +
             * @param {Array} args[1]] Оператор
         
     | 
| 
      
 37 
     | 
    
         
            +
             * @param {Array} args[2]] Друге значення
         
     | 
| 
      
 38 
     | 
    
         
            +
             * @returns {String} Returns HTML
         
     | 
| 
      
 39 
     | 
    
         
            +
             */
         
     | 
| 
      
 40 
     | 
    
         
            +
            export default function ifCond(v1, operator, v2, options) {
         
     | 
| 
      
 41 
     | 
    
         
            +
              const __obj = this;
         
     | 
| 
      
 42 
     | 
    
         
            +
             
     | 
| 
      
 43 
     | 
    
         
            +
              switch (operator) {
         
     | 
| 
      
 44 
     | 
    
         
            +
                case '==':
         
     | 
| 
      
 45 
     | 
    
         
            +
                  return (v1 == v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 46 
     | 
    
         
            +
                case '!=':
         
     | 
| 
      
 47 
     | 
    
         
            +
                  return (v1 != v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 48 
     | 
    
         
            +
                case '===':
         
     | 
| 
      
 49 
     | 
    
         
            +
                  return (v1 === v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 50 
     | 
    
         
            +
                case '!==':
         
     | 
| 
      
 51 
     | 
    
         
            +
                  return (v1 !== v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 52 
     | 
    
         
            +
                case '&&':
         
     | 
| 
      
 53 
     | 
    
         
            +
                  return (v1 && v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 54 
     | 
    
         
            +
                case '||':
         
     | 
| 
      
 55 
     | 
    
         
            +
                  return (v1 || v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 56 
     | 
    
         
            +
                case '<':
         
     | 
| 
      
 57 
     | 
    
         
            +
                  return (v1 < v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 58 
     | 
    
         
            +
                case '<=':
         
     | 
| 
      
 59 
     | 
    
         
            +
                  return (v1 <= v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 60 
     | 
    
         
            +
                case '>':
         
     | 
| 
      
 61 
     | 
    
         
            +
                  return (v1 > v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 62 
     | 
    
         
            +
                case '>=':
         
     | 
| 
      
 63 
     | 
    
         
            +
                  return (v1 >= v2) ? options.fn(__obj) : options.inverse(__obj);
         
     | 
| 
      
 64 
     | 
    
         
            +
                case '&':
         
     | 
| 
      
 65 
     | 
    
         
            +
                  return intersect(v1, v2).length !== 0
         
     | 
| 
      
 66 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 67 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 68 
     | 
    
         
            +
                case '!~':
         
     | 
| 
      
 69 
     | 
    
         
            +
                  return (v1 || '').indexOf(v2) === -1
         
     | 
| 
      
 70 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 71 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 72 
     | 
    
         
            +
                case '~':
         
     | 
| 
      
 73 
     | 
    
         
            +
                  return (v1 || '').indexOf(v2) !== -1
         
     | 
| 
      
 74 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 75 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 76 
     | 
    
         
            +
                case 'period':
         
     | 
| 
      
 77 
     | 
    
         
            +
                  return (new Date(v1) < new Date() && new Date(v2) > new Date())
         
     | 
| 
      
 78 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 79 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 80 
     | 
    
         
            +
                case 'in': {
         
     | 
| 
      
 81 
     | 
    
         
            +
                  if (typeof v2 === 'string') v2 = v2.split(',').map(item => item.trim());
         
     | 
| 
      
 82 
     | 
    
         
            +
             
     | 
| 
      
 83 
     | 
    
         
            +
                  if (Array.isArray(v1)) {
         
     | 
| 
      
 84 
     | 
    
         
            +
                    return v1.some((value) => v2.includes(value.toString()))
         
     | 
| 
      
 85 
     | 
    
         
            +
                      ? options.fn(__obj)
         
     | 
| 
      
 86 
     | 
    
         
            +
                      : options.inverse(__obj);
         
     | 
| 
      
 87 
     | 
    
         
            +
                  }
         
     | 
| 
      
 88 
     | 
    
         
            +
                  return v2.includes(v1?.toString())
         
     | 
| 
      
 89 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 90 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 91 
     | 
    
         
            +
                }
         
     | 
| 
      
 92 
     | 
    
         
            +
                case 'not in': {
         
     | 
| 
      
 93 
     | 
    
         
            +
                  if (typeof v2 === 'string') v2 = v2.split(',').map(item => item.trim());
         
     | 
| 
      
 94 
     | 
    
         
            +
                  return !v2.includes(v1?.toString())
         
     | 
| 
      
 95 
     | 
    
         
            +
                    ? options.fn(__obj)
         
     | 
| 
      
 96 
     | 
    
         
            +
                    : options.inverse(__obj);
         
     | 
| 
      
 97 
     | 
    
         
            +
                }
         
     | 
| 
      
 98 
     | 
    
         
            +
                default:
         
     | 
| 
      
 99 
     | 
    
         
            +
                  return options.inverse(__obj);
         
     | 
| 
      
 100 
     | 
    
         
            +
              }
         
     | 
| 
      
 101 
     | 
    
         
            +
            }
         
     | 
| 
         @@ -1,6 +1,6 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
             
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            export default function button(token, title) {
         
     | 
| 
       4 
     | 
    
         
            -
                return `<button onclick="window.v3plugin.$form({ token: '${token}' })"
         
     | 
| 
       5 
     | 
    
         
            -
                class="inline-flex items-center px-2 py-1 text-sm font-medium text-white duration-300 bg-blue-600 border border-transparent rounded-lg gap-x-2 hover:bg-blue-700 hover:text-white">${title || 'Редагувати'}</button>`;
         
     | 
| 
      
 1 
     | 
    
         
            +
             
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            export default function button(token, title) {
         
     | 
| 
      
 4 
     | 
    
         
            +
                return `<button onclick="window.v3plugin.$form({ token: '${token}' })"
         
     | 
| 
      
 5 
     | 
    
         
            +
                class="inline-flex items-center px-2 py-1 text-sm font-medium text-white duration-300 bg-blue-600 border border-transparent rounded-lg gap-x-2 hover:bg-blue-700 hover:text-white">${title || 'Редагувати'}</button>`;
         
     | 
| 
       6 
6 
     | 
    
         
             
            }
         
     | 
| 
         @@ -1,17 +1,17 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            import md from 'markdown-it';
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            const md1 = md({ html: true });
         
     | 
| 
       4 
     | 
    
         
            -
             
     | 
| 
       5 
     | 
    
         
            -
            /**
         
     | 
| 
       6 
     | 
    
         
            -
             * Перетворення з файла readme.md до формату HTML.
         
     | 
| 
       7 
     | 
    
         
            -
             * Потрабно вставити в хелпер шлях до файла або текст readme.md і за допомогою бібліотеки markdown-it перетвориться в HTML.
         
     | 
| 
       8 
     | 
    
         
            -
             
     | 
| 
       9 
     | 
    
         
            -
             * @returns {String} Returns HTML
         
     | 
| 
       10 
     | 
    
         
            -
            */
         
     | 
| 
       11 
     | 
    
         
            -
            export default function mdToHTML(data, options) {
         
     | 
| 
       12 
     | 
    
         
            -
              // auto detect HTML or MD
         
     | 
| 
       13 
     | 
    
         
            -
              // const result = md().render(data);
         
     | 
| 
       14 
     | 
    
         
            -
              if (!data) return 'empty data';
         
     | 
| 
       15 
     | 
    
         
            -
              const result = md1.render(data);
         
     | 
| 
       16 
     | 
    
         
            -
              return result;
         
     | 
| 
       17 
     | 
    
         
            -
            };
         
     | 
| 
      
 1 
     | 
    
         
            +
            import md from 'markdown-it';
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            const md1 = md({ html: true });
         
     | 
| 
      
 4 
     | 
    
         
            +
             
     | 
| 
      
 5 
     | 
    
         
            +
            /**
         
     | 
| 
      
 6 
     | 
    
         
            +
             * Перетворення з файла readme.md до формату HTML.
         
     | 
| 
      
 7 
     | 
    
         
            +
             * Потрабно вставити в хелпер шлях до файла або текст readme.md і за допомогою бібліотеки markdown-it перетвориться в HTML.
         
     | 
| 
      
 8 
     | 
    
         
            +
             
     | 
| 
      
 9 
     | 
    
         
            +
             * @returns {String} Returns HTML
         
     | 
| 
      
 10 
     | 
    
         
            +
            */
         
     | 
| 
      
 11 
     | 
    
         
            +
            export default function mdToHTML(data, options) {
         
     | 
| 
      
 12 
     | 
    
         
            +
              // auto detect HTML or MD
         
     | 
| 
      
 13 
     | 
    
         
            +
              // const result = md().render(data);
         
     | 
| 
      
 14 
     | 
    
         
            +
              if (!data) return 'empty data';
         
     | 
| 
      
 15 
     | 
    
         
            +
              const result = md1.render(data);
         
     | 
| 
      
 16 
     | 
    
         
            +
              return result;
         
     | 
| 
      
 17 
     | 
    
         
            +
            };
         
     | 
    
        package/server/plugins/cron.js
    CHANGED
    
    | 
         @@ -1,11 +1,11 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            //import { addCron } from '@opengis/fastify-table/utils.js';
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            async function deleteOldNotifications({ pg }) {
         
     | 
| 
       4 
     | 
    
         
            -
                const { rowCount } = pg?.pk?.['crm.notifications'] ? await pg.query('delete from crm.notifications where CURRENT_DATE - \'14 days\'::interval > cdate') : {};
         
     | 
| 
       5 
     | 
    
         
            -
                return { rowCount };
         
     | 
| 
       6 
     | 
    
         
            -
            }
         
     | 
| 
       7 
     | 
    
         
            -
             
     | 
| 
       8 
     | 
    
         
            -
            export default async function cron(fastify) {
         
     | 
| 
       9 
     | 
    
         
            -
                // addCron(deleteOldNotifications, 60 * 60 * 24, fastify);
         
     | 
| 
       10 
     | 
    
         
            -
                //  addCron(deleteOldNotifications, 60 * 1, fastify); // debug
         
     | 
| 
      
 1 
     | 
    
         
            +
            //import { addCron } from '@opengis/fastify-table/utils.js';
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            async function deleteOldNotifications({ pg }) {
         
     | 
| 
      
 4 
     | 
    
         
            +
                const { rowCount } = pg?.pk?.['crm.notifications'] ? await pg.query('delete from crm.notifications where CURRENT_DATE - \'14 days\'::interval > cdate') : {};
         
     | 
| 
      
 5 
     | 
    
         
            +
                return { rowCount };
         
     | 
| 
      
 6 
     | 
    
         
            +
            }
         
     | 
| 
      
 7 
     | 
    
         
            +
             
     | 
| 
      
 8 
     | 
    
         
            +
            export default async function cron(fastify) {
         
     | 
| 
      
 9 
     | 
    
         
            +
                // addCron(deleteOldNotifications, 60 * 60 * 24, fastify);
         
     | 
| 
      
 10 
     | 
    
         
            +
                //  addCron(deleteOldNotifications, 60 * 1, fastify); // debug
         
     | 
| 
       11 
11 
     | 
    
         
             
            }
         
     | 
    
        package/server/plugins/docs.js
    CHANGED
    
    | 
         @@ -1,28 +1,28 @@ 
     | 
|
| 
       1 
     | 
    
         
            -
            'use strict'
         
     | 
| 
       2 
     | 
    
         
            -
             
     | 
| 
       3 
     | 
    
         
            -
            import path, { dirname } from 'path';
         
     | 
| 
       4 
     | 
    
         
            -
            import { fileURLToPath } from 'url';
         
     | 
| 
       5 
     | 
    
         
            -
            import fs from 'fs';
         
     | 
| 
       6 
     | 
    
         
            -
             
     | 
| 
       7 
     | 
    
         
            -
            const dir = dirname(fileURLToPath(import.meta.url));
         
     | 
| 
       8 
     | 
    
         
            -
            const root = `${dir}/../../`;
         
     | 
| 
       9 
     | 
    
         
            -
             
     | 
| 
       10 
     | 
    
         
            -
             
     | 
| 
       11 
     | 
    
         
            -
            async function plugin(fastify, opts) {
         
     | 
| 
       12 
     | 
    
         
            -
              fastify.get('/docs*', async (req, reply) => {
         
     | 
| 
       13 
     | 
    
         
            -
                if (!fs.existsSync(path.join(root, 'docs/.vitepress/dist/'))) {
         
     | 
| 
       14 
     | 
    
         
            -
                  return reply.status(404).send('docs not exists');
         
     | 
| 
       15 
     | 
    
         
            -
                }
         
     | 
| 
       16 
     | 
    
         
            -
                const { params } = req;
         
     | 
| 
       17 
     | 
    
         
            -
                const url = params['*']
         
     | 
| 
       18 
     | 
    
         
            -
                const filePath = url && url[url.length - 1] !== '/' ? path.join(root, 'docs/.vitepress/dist/', url) : path.join(root, 'docs/.vitepress/dist/', url, 'index.html')
         
     | 
| 
       19 
     | 
    
         
            -
                const ext = path.extname(filePath);
         
     | 
| 
       20 
     | 
    
         
            -
                const mime = {
         
     | 
| 
       21 
     | 
    
         
            -
                  '.js': 'text/javascript', '.css': 'text/css', '.woff2': 'application/font-woff', '.png': 'image/png', '.svg': 'image/svg+xml', '.jpg': 'image/jpg'
         
     | 
| 
       22 
     | 
    
         
            -
                }[ext];
         
     | 
| 
       23 
     | 
    
         
            -
                const stream = fs.createReadStream(filePath);
         
     | 
| 
       24 
     | 
    
         
            -
                return mime ? reply.type(mime).send(stream) : stream;
         
     | 
| 
       25 
     | 
    
         
            -
             
     | 
| 
       26 
     | 
    
         
            -
              })
         
     | 
| 
       27 
     | 
    
         
            -
            }
         
     | 
| 
       28 
     | 
    
         
            -
            export default plugin;
         
     | 
| 
      
 1 
     | 
    
         
            +
            'use strict'
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            import path, { dirname } from 'path';
         
     | 
| 
      
 4 
     | 
    
         
            +
            import { fileURLToPath } from 'url';
         
     | 
| 
      
 5 
     | 
    
         
            +
            import fs from 'fs';
         
     | 
| 
      
 6 
     | 
    
         
            +
             
     | 
| 
      
 7 
     | 
    
         
            +
            const dir = dirname(fileURLToPath(import.meta.url));
         
     | 
| 
      
 8 
     | 
    
         
            +
            const root = `${dir}/../../`;
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
| 
      
 10 
     | 
    
         
            +
             
     | 
| 
      
 11 
     | 
    
         
            +
            async function plugin(fastify, opts) {
         
     | 
| 
      
 12 
     | 
    
         
            +
              fastify.get('/docs*', async (req, reply) => {
         
     | 
| 
      
 13 
     | 
    
         
            +
                if (!fs.existsSync(path.join(root, 'docs/.vitepress/dist/'))) {
         
     | 
| 
      
 14 
     | 
    
         
            +
                  return reply.status(404).send('docs not exists');
         
     | 
| 
      
 15 
     | 
    
         
            +
                }
         
     | 
| 
      
 16 
     | 
    
         
            +
                const { params } = req;
         
     | 
| 
      
 17 
     | 
    
         
            +
                const url = params['*']
         
     | 
| 
      
 18 
     | 
    
         
            +
                const filePath = url && url[url.length - 1] !== '/' ? path.join(root, 'docs/.vitepress/dist/', url) : path.join(root, 'docs/.vitepress/dist/', url, 'index.html')
         
     | 
| 
      
 19 
     | 
    
         
            +
                const ext = path.extname(filePath);
         
     | 
| 
      
 20 
     | 
    
         
            +
                const mime = {
         
     | 
| 
      
 21 
     | 
    
         
            +
                  '.js': 'text/javascript', '.css': 'text/css', '.woff2': 'application/font-woff', '.png': 'image/png', '.svg': 'image/svg+xml', '.jpg': 'image/jpg'
         
     | 
| 
      
 22 
     | 
    
         
            +
                }[ext];
         
     | 
| 
      
 23 
     | 
    
         
            +
                const stream = fs.createReadStream(filePath);
         
     | 
| 
      
 24 
     | 
    
         
            +
                return mime ? reply.type(mime).send(stream) : stream;
         
     | 
| 
      
 25 
     | 
    
         
            +
             
     | 
| 
      
 26 
     | 
    
         
            +
              })
         
     | 
| 
      
 27 
     | 
    
         
            +
            }
         
     | 
| 
      
 28 
     | 
    
         
            +
            export default plugin;
         
     |