@opengis/admin 0.2.8 → 0.2.10

Sign up to get free protection for your applications and to get access to all the features.
Files changed (165) hide show
  1. package/README.md +29 -29
  2. package/config.js +4 -4
  3. package/dist/{IconChevronDown-BCpDxWU9.js → IconChevronDown-DZokai01.js} +1 -1
  4. package/dist/{add-page-CKy5L_78.js → add-page-CZerBOHh.js} +1 -1
  5. package/dist/{admin-interface-BX613rWQ.js → admin-interface-CHC7-JA_.js} +260 -259
  6. package/dist/{admin-view-DJBkad_B.js → admin-view-DJfa3luE.js} +3 -3
  7. package/dist/admin.js +1 -1
  8. package/dist/admin.umd.cjs +55 -55
  9. package/dist/assets/logo.svg +41 -41
  10. package/dist/{card-view-CdCsaogK.js → card-view-DGHbbk0Z.js} +1 -1
  11. package/dist/edit-page-DO6_OHVq.js +128 -0
  12. package/dist/{import-file-CknmeGNO.js → import-file-oUKCJadF.js} +6393 -6349
  13. package/dist/style.css +1 -1
  14. package/module/settings/card/admin.accounts.table/index.yml +7 -7
  15. package/module/settings/card/admin.accounts.table/rules.hbs +18 -18
  16. package/module/settings/card/admin.accounts.table/users.hbs +13 -13
  17. package/module/settings/card/admin.roles.table/access.hbs +3 -3
  18. package/module/settings/card/admin.roles.table/general_info.hbs +1 -1
  19. package/module/settings/card/admin.roles.table/index.yml +21 -21
  20. package/module/settings/card/admin.roles.table/users.hbs +6 -6
  21. package/module/settings/card/admin.routes.table/general_info.hbs +13 -13
  22. package/module/settings/card/admin.routes.table/groups.hbs +11 -11
  23. package/module/settings/card/admin.routes.table/index.yml +11 -11
  24. package/module/settings/card/admin.routes.table/users.hbs +16 -16
  25. package/module/settings/card/admin.users.table/context.hbs +14 -14
  26. package/module/settings/card/admin.users.table/general_info.hbs +12 -12
  27. package/module/settings/card/admin.users.table/index.yml +22 -22
  28. package/module/settings/card/admin.users.table/last_login.hbs +9 -9
  29. package/module/settings/card/admin.users.table/logs.hbs +10 -10
  30. package/module/settings/card/admin.users.table/routes.hbs +7 -7
  31. package/module/settings/card/admin.users.table/user_roles.hbs +12 -12
  32. package/module/settings/cls/core.actions.json +17 -17
  33. package/module/settings/cls/core.scope.json +13 -13
  34. package/module/settings/cls/properties.site_status.json +13 -13
  35. package/module/settings/cls/properties.widget_status.json +13 -13
  36. package/module/settings/cls/users.user_type.json +13 -13
  37. package/module/settings/form/admin.accounts.form.json +13 -13
  38. package/module/settings/form/admin.custom_column.form.json +71 -71
  39. package/module/settings/form/admin.properties.form.json +15 -15
  40. package/module/settings/form/admin.roles.form.json +21 -21
  41. package/module/settings/form/admin.routes.form.json +25 -25
  42. package/module/settings/form/admin.rules.form.json +30 -30
  43. package/module/settings/form/admin.user_properties.form.json +15 -15
  44. package/module/settings/form/admin.user_roles.form.json +13 -13
  45. package/module/settings/form/admin.user_roles_card.form.json +13 -13
  46. package/module/settings/form/admin.users.form.json +153 -153
  47. package/module/settings/form/context.account_grants.form.json +23 -23
  48. package/module/settings/form/context.account_users.form.json +12 -12
  49. package/module/settings/form/user.user_roles.form.json +13 -13
  50. package/module/settings/interface/admin.properties.json +4 -4
  51. package/module/settings/interface/admin.roles.json +4 -4
  52. package/module/settings/interface/admin.routes.json +4 -4
  53. package/module/settings/interface/admin.users.json +4 -4
  54. package/module/settings/menu.json +84 -84
  55. package/module/settings/select/core.routes.sql +1 -1
  56. package/module/settings/select/core.user_mentioned.sql +1 -1
  57. package/module/settings/select/core.user_uid.sql +1 -1
  58. package/module/settings/table/admin.accounts.table.json +42 -42
  59. package/module/settings/table/admin.custom_column.table.json +99 -99
  60. package/module/settings/table/admin.properties.table.json +39 -39
  61. package/module/settings/table/admin.roles.table.json +64 -64
  62. package/module/settings/table/admin.routes.table.json +73 -73
  63. package/module/settings/table/admin.rules.table.json +76 -76
  64. package/module/settings/table/admin.user_properties.table.json +34 -34
  65. package/module/settings/table/admin.user_roles.table.json +72 -72
  66. package/module/settings/table/admin.users.table.json +132 -132
  67. package/module/settings/table/context.account_grants.table.json +67 -67
  68. package/module/settings/table/context.account_users.table.json +37 -37
  69. package/package.json +83 -83
  70. package/plugin.js +29 -29
  71. package/server/helpers/core/badge.js +16 -16
  72. package/server/helpers/core/buttonHelper.js +21 -21
  73. package/server/helpers/core/select.js +48 -48
  74. package/server/helpers/core/token.js +18 -18
  75. package/server/helpers/index.js +29 -29
  76. package/server/helpers/list/buttonHelper.js +21 -21
  77. package/server/helpers/list/descriptionList.js +43 -43
  78. package/server/helpers/list/tableList.js +81 -81
  79. package/server/helpers/list/utils/button.js +5 -5
  80. package/server/helpers/temp/contentList.js +58 -58
  81. package/server/helpers/temp/ifCond.js +101 -101
  82. package/server/helpers/utils/button.js +5 -5
  83. package/server/helpers/utils/buttonAdd.js +5 -5
  84. package/server/helpers/utils/buttonDel.js +5 -5
  85. package/server/helpers/utils/buttonEdit.js +5 -5
  86. package/server/plugins/access/funcs/getAdminAccess.js +12 -12
  87. package/server/plugins/access/index.mjs +6 -6
  88. package/server/plugins/adminHook.js +81 -81
  89. package/server/plugins/cron.js +10 -10
  90. package/server/plugins/docs.js +28 -28
  91. package/server/plugins/hook.js +236 -236
  92. package/server/plugins/vite.js +71 -71
  93. package/server/routes/access/controllers/access.group.js +29 -29
  94. package/server/routes/access/controllers/access.group.post.js +49 -49
  95. package/server/routes/access/index.mjs +8 -8
  96. package/server/routes/access/schema.mjs +57 -57
  97. package/server/routes/calendar/controllers/calendar.data.js +87 -87
  98. package/server/routes/calendar/index.mjs +7 -7
  99. package/server/routes/calendar/schema.js +21 -21
  100. package/server/routes/data/controllers/cardData.js +105 -105
  101. package/server/routes/data/controllers/cardTabData.js +49 -49
  102. package/server/routes/data/controllers/funcs/getFilterSQL/index.js +92 -92
  103. package/server/routes/data/controllers/funcs/getFilterSQL/util/formatValue.js +170 -170
  104. package/server/routes/data/controllers/funcs/getFilterSQL/util/getCustomQuery.js +13 -13
  105. package/server/routes/data/controllers/funcs/getFilterSQL/util/getFilterQuery.js +64 -64
  106. package/server/routes/data/controllers/funcs/getFilterSQL/util/getOptimizedQuery.js +12 -12
  107. package/server/routes/data/controllers/funcs/getFilterSQL/util/getTableSql.js +34 -34
  108. package/server/routes/data/controllers/tableData.js +29 -29
  109. package/server/routes/data/controllers/tableDataId.js +27 -27
  110. package/server/routes/data/controllers/tableFilter.js +67 -67
  111. package/server/routes/data/controllers/tokenInfo.js +9 -9
  112. package/server/routes/data/controllers/utils/assignTokens.js +30 -30
  113. package/server/routes/data/controllers/utils/conditions.js +20 -20
  114. package/server/routes/data/controllers/utils/getColumns.js +8 -8
  115. package/server/routes/data/index.mjs +17 -17
  116. package/server/routes/data/schema.js +54 -54
  117. package/server/routes/menu/controllers/getMenu.js +58 -58
  118. package/server/routes/menu/index.mjs +5 -5
  119. package/server/routes/notifications/controllers/readNotifications.js +27 -27
  120. package/server/routes/notifications/controllers/testEmail.js +35 -35
  121. package/server/routes/notifications/controllers/userNotifications.js +53 -53
  122. package/server/routes/notifications/funcs/addNotification.js +21 -21
  123. package/server/routes/notifications/funcs/sendNotification.js +92 -92
  124. package/server/routes/notifications/hook/onWidgetSet.js +57 -57
  125. package/server/routes/notifications/index.mjs +27 -27
  126. package/server/routes/notifications/schema.js +16 -16
  127. package/server/routes/properties/controllers/admin.properties.get.js +29 -29
  128. package/server/routes/properties/controllers/user.properties.get.js +30 -30
  129. package/server/routes/properties/controllers/user.properties.post.js +30 -30
  130. package/server/routes/properties/funcs/getSettings.js +56 -56
  131. package/server/routes/properties/funcs/setSettings.js +44 -44
  132. package/server/routes/properties/funcs/utils/dataInsert.js +26 -26
  133. package/server/routes/properties/index.mjs +14 -14
  134. package/server/routes/properties/schema.js +10 -10
  135. package/server/routes/root.mjs +3 -3
  136. package/server/routes/templates/controllers/getTemplate.js +43 -43
  137. package/server/routes/templates/index.mjs +16 -16
  138. package/server/routes/templates/schema.js +8 -8
  139. package/server/routes/user/controllers/user.cls.id.js +14 -14
  140. package/server/routes/user/controllers/user.cls.js +71 -71
  141. package/server/routes/user/controllers/user.cls.post.js +52 -52
  142. package/server/routes/user/controllers/user.info.js +17 -17
  143. package/server/routes/user/schema.js +14 -14
  144. package/server/routes/widget/controllers/utils/historyFormat.js +75 -75
  145. package/server/routes/widget/controllers/utils/obj2db.js +13 -13
  146. package/server/routes/widget/controllers/widget.del.js +41 -41
  147. package/server/routes/widget/controllers/widget.get.js +96 -96
  148. package/server/routes/widget/controllers/widget.set.js +76 -76
  149. package/server/routes/widget/index.mjs +11 -11
  150. package/server/routes/widget/schema.js +12 -12
  151. package/server/templates/cls/itree.recrzone_category.json +73 -73
  152. package/server/templates/cls/test.json +9 -9
  153. package/server/templates/form/admin.user_cls.data.form.json +49 -49
  154. package/server/templates/form/admin.user_group_rel.form.json +21 -21
  155. package/server/templates/form/cp_building.form.json +32 -32
  156. package/server/templates/form/form-user-pass.json +10 -10
  157. package/server/templates/form/form-user_group.json +39 -39
  158. package/server/templates/form/form-users.json +156 -156
  159. package/server/templates/form/user_group_access.form.json +22 -22
  160. package/server/templates/select/account_id.json +2 -2
  161. package/server/templates/table/gis.dataset.table.json +43 -43
  162. package/server/templates/table/management.user_group.table.json +112 -112
  163. package/server/templates/table/management.users.table.json +126 -126
  164. package/utils.js +29 -29
  165. package/dist/edit-page-C76NxZRq.js +0 -120
@@ -1,170 +1,170 @@
1
- const dateTypeList = ['date', 'timestamp', 'timestamp without time zone'];
2
- const numberTypeList = ['float8', 'int4', 'int8', 'numeric', 'double precision', 'integer'];
3
-
4
- function dt(y, m, d) {
5
- return new Date(Date.UTC(y, m, d)).toISOString().slice(0, 10);
6
- }
7
- const dp = {
8
- d: new Date().getDate(),
9
- w: new Date().getDate() - (new Date().getDay() || 7) + 1,
10
- m: new Date().getMonth(),
11
- q: (new Date().getMonth() / 4).toFixed() * 3,
12
- y: new Date().getFullYear(),
13
- };
14
-
15
- function formatDateISOString(date) {
16
- if (!date?.includes('.')) return date;
17
- const [day, month, year] = date.split('.');
18
- return `${year}-${month}-${day}`;
19
- }
20
-
21
- function formatValue({
22
- pg, table, filter = {}, name, value, operator = '=', dataTypeID, uid = 1, optimize,
23
- }) {
24
- const { data, sql, extra } = filter;
25
- const pk = pg?.pk && table ? pg.pk[table] : undefined;
26
-
27
- if (!dataTypeID && !extra) return {};
28
- const fieldType = extra ? pg.pgType?.[{ 'Date': 1114 }[filter?.type] || 25] : pg.pgType?.[dataTypeID];
29
- if (!name || !value || !fieldType) return {};
30
- const filterType = filter.type?.toLowerCase();
31
-
32
- // current day, week, month, year etc.
33
- if (dateTypeList.includes(fieldType) && !value?.includes('_') && ['cd', 'cw', 'cm', 'cq', 'cy'].includes(value)) {
34
- const query = {
35
- cd: `${name}::date = '${dt(dp.y, dp.m, dp.d)}'::date`,
36
- cw: `${name}::date >= '${dt(dp.y, dp.m, dp.w)}'::date and ${name} <= '${dt(dp.y, dp.m, dp.w + 6)}'::date`,
37
- cm: `${name}::date >= '${dt(dp.y, dp.m, 1)}'::date and ${name} <= '${dt(dp.y, dp.m + 1, 0)}'::date`,
38
- cq: `${name}::date >= '${dt(dp.y, dp.q, 1)}'::date and ${name} <= '${dt(dp.y, dp.q + 3, 0)}'::date`,
39
- cy: `${name}::date >= '${dt(dp.y, 0, 1)}'::date and ${name}::date <= '${dt(dp.y, 11, 31)}'::date`,
40
- }[value];
41
- return { op: '=', query, extra };
42
- }
43
-
44
- // date range
45
- if (dateTypeList.includes(fieldType) && value?.includes('_')) {
46
- const [min, max] = value.split('_');
47
- const query = `${name} >= '${min}'::date and ${name} <= '${max}'::date`;
48
- return { op: 'between', query, extra };
49
- }
50
-
51
- // v3 filter date range, example - "01.01.2024-31.12.2024"
52
- if (dateTypeList.includes(fieldType) && value?.includes('.') && value?.indexOf('-') === 10 && value?.length === 21) {
53
- const [startDate, endDate] = value.split('-');
54
- const min = formatDateISOString(startDate);
55
- const max = formatDateISOString(endDate);
56
- const query = extra && pk
57
- ? `${pk} in (select object_id from crn.extra_data where property_key='${name}' and value_date::date >= '${min}'::date and value_date::date <= '${max}'::date)`
58
- : `${name}::date >= '${min}'::date and ${name}::date <= '${max}'::date`;
59
- return { op: 'between', query, extra };
60
- }
61
-
62
- // my rows
63
- if (value === 'me' && uid && fieldType === 'text') {
64
- return { op: '=', query: extra ? `uid = '${uid}'` : `${name}::text = '${uid}'`, extra };
65
- }
66
-
67
- const formatType = {
68
- float8: 'numeric',
69
- int4: 'numeric',
70
- int8: 'numeric',
71
- varchar: 'text',
72
- bool: 'boolean',
73
- geometry: 'geom',
74
- }[fieldType] || 'text';
75
-
76
- if (optimize && optimize.name !== optimize.pk) {
77
- const val = filterType === 'text' ? `ilike '%${value}%'` : `= any('{${value}}')`;
78
- return {
79
- op: '~',
80
- query: fieldType?.includes('[]')
81
- ? `${optimize.pk} && (select array_agg(${optimize.pk}) from ${optimize.table} where ${name} ${val} )`
82
- : `${optimize.pk} in (select ${optimize.pk} from ${optimize.table} where ${name} ${val} )`,
83
- extra,
84
- };
85
- }
86
-
87
- if (fieldType?.includes('[]')) {
88
- return { op: 'in', query: `'{${value}}'::text[] && ${name}::text[]`, extra };
89
- }
90
-
91
- // multiple items of 1 param
92
- if (value?.indexOf(',') !== -1) {
93
- const values = value.split(',').filter((el) => el !== 'null');
94
- if (extra && pk) {
95
- const query = value?.indexOf('null') !== -1
96
- ? `${pk} in (select object_id from crm.extra_data where property_key='${name}' and ( value_text is null or value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) ) )`
97
- : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) )`;
98
- return { op: 'in', query, extra };
99
- }
100
- const query = value?.indexOf('null') !== -1
101
- ? `( ${name} is null or ${name}::text in (${values?.map((el) => `'${el}'`).join(',')}) )`
102
- : `${name}::text in (${value.split(',')?.map((el) => `'${el}'`).join(',')})`;
103
- return { op: 'in', query, extra };
104
- }
105
-
106
- // v3 filter number range, example - "100_500"
107
- if (numberTypeList.includes(fieldType) && value?.indexOf('_') !== -1) {
108
- const [min, max] = value.split('_');
109
- const query = (max === 'max' ? `${name} > ${min}` : null) || (min === 'min' ? `${name} < ${max}` : null) || `${name} between ${min} and ${max}`;
110
- return { op: 'between', query, extra };
111
- }
112
-
113
- // number range
114
- if (numberTypeList.includes(fieldType) && value?.indexOf('-') !== -1) {
115
- const [min, max] = value.split('-');
116
- if (min === 'min' && max === 'max') return {};
117
- const query = (max === 'max' ? `${name} > ${min}` : null) || (min === 'min' ? `${name} < ${max}` : null) || `${name} between ${min} and ${max}`;
118
- return { op: 'between', query, extra };
119
- }
120
-
121
- if (['<', '>'].includes(operator)) {
122
- const query = `${name} ${operator} '${value}'::${formatType}`;
123
- return { op: operator, query, extra };
124
- }
125
-
126
- if (operator === '=' && filterType !== 'text' && !filter?.data) {
127
- const query = {
128
- null: `${name} is null`,
129
- notnull: `${name} is not null`,
130
- }[value] || `${name}::${formatType}='${value}'::${formatType}`;
131
- return { op: '=', query, extra };
132
- }
133
-
134
- if (['~', '='].includes(operator)) {
135
- const operator1 = (filterType === 'text' && (filter?.id || filter?.name) && operator === '=' ? '~' : operator);
136
- const match = operator1 === '=' ? `='${value}'` : `ilike '%${value}%'`;
137
- if (extra && pk) {
138
- const query = data && sql
139
- ? `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text in ( ( with q(id,name) as (${sql}) select id from q where name ${match})))`
140
- : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text ${match})`
141
- return { op: 'ilike', query, extra };
142
- }
143
-
144
- const query = filter?.data && filter?.sql
145
- ? `${filter?.name || filter?.id} in ( ( with q(id,name) as (${filter?.sql}) select id from q where name::text ${match}) )` // filter with cls
146
- : `${name}::text ${match}`; // simple filter
147
- // console.log(query);
148
- return { op: 'ilike', query };
149
- }
150
-
151
- // json
152
- if (name.includes('.')) {
153
- const [col, prop] = name.split('.');
154
- const query = ` ${col}->>'${prop}' in ('${value.join("','")}')`;
155
- return { op: 'in', query, extra };
156
- }
157
-
158
- // geometry
159
- if (['geometry'].includes(fieldType)) {
160
- const bbox = value[0].split('_');
161
-
162
- if (bbox?.length === 4) {
163
- const query = ` ${name} && 'box(${bbox[0]} ${bbox[1]},${bbox[2]} ${bbox[3]})'::box2d `;
164
- return { op: '&&', query, extra };
165
- }
166
- }
167
- return {};
168
- }
169
-
170
- export default formatValue;
1
+ const dateTypeList = ['date', 'timestamp', 'timestamp without time zone'];
2
+ const numberTypeList = ['float8', 'int4', 'int8', 'numeric', 'double precision', 'integer'];
3
+
4
+ function dt(y, m, d) {
5
+ return new Date(Date.UTC(y, m, d)).toISOString().slice(0, 10);
6
+ }
7
+ const dp = {
8
+ d: new Date().getDate(),
9
+ w: new Date().getDate() - (new Date().getDay() || 7) + 1,
10
+ m: new Date().getMonth(),
11
+ q: (new Date().getMonth() / 4).toFixed() * 3,
12
+ y: new Date().getFullYear(),
13
+ };
14
+
15
+ function formatDateISOString(date) {
16
+ if (!date?.includes('.')) return date;
17
+ const [day, month, year] = date.split('.');
18
+ return `${year}-${month}-${day}`;
19
+ }
20
+
21
+ function formatValue({
22
+ pg, table, filter = {}, name, value, operator = '=', dataTypeID, uid = 1, optimize,
23
+ }) {
24
+ const { data, sql, extra } = filter;
25
+ const pk = pg?.pk && table ? pg.pk[table] : undefined;
26
+
27
+ if (!dataTypeID && !extra) return {};
28
+ const fieldType = extra ? pg.pgType?.[{ 'Date': 1114 }[filter?.type] || 25] : pg.pgType?.[dataTypeID];
29
+ if (!name || !value || !fieldType) return {};
30
+ const filterType = filter.type?.toLowerCase();
31
+
32
+ // current day, week, month, year etc.
33
+ if (dateTypeList.includes(fieldType) && !value?.includes('_') && ['cd', 'cw', 'cm', 'cq', 'cy'].includes(value)) {
34
+ const query = {
35
+ cd: `${name}::date = '${dt(dp.y, dp.m, dp.d)}'::date`,
36
+ cw: `${name}::date >= '${dt(dp.y, dp.m, dp.w)}'::date and ${name} <= '${dt(dp.y, dp.m, dp.w + 6)}'::date`,
37
+ cm: `${name}::date >= '${dt(dp.y, dp.m, 1)}'::date and ${name} <= '${dt(dp.y, dp.m + 1, 0)}'::date`,
38
+ cq: `${name}::date >= '${dt(dp.y, dp.q, 1)}'::date and ${name} <= '${dt(dp.y, dp.q + 3, 0)}'::date`,
39
+ cy: `${name}::date >= '${dt(dp.y, 0, 1)}'::date and ${name}::date <= '${dt(dp.y, 11, 31)}'::date`,
40
+ }[value];
41
+ return { op: '=', query, extra };
42
+ }
43
+
44
+ // date range
45
+ if (dateTypeList.includes(fieldType) && value?.includes('_')) {
46
+ const [min, max] = value.split('_');
47
+ const query = `${name} >= '${min}'::date and ${name} <= '${max}'::date`;
48
+ return { op: 'between', query, extra };
49
+ }
50
+
51
+ // v3 filter date range, example - "01.01.2024-31.12.2024"
52
+ if (dateTypeList.includes(fieldType) && value?.includes('.') && value?.indexOf('-') === 10 && value?.length === 21) {
53
+ const [startDate, endDate] = value.split('-');
54
+ const min = formatDateISOString(startDate);
55
+ const max = formatDateISOString(endDate);
56
+ const query = extra && pk
57
+ ? `${pk} in (select object_id from crn.extra_data where property_key='${name}' and value_date::date >= '${min}'::date and value_date::date <= '${max}'::date)`
58
+ : `${name}::date >= '${min}'::date and ${name}::date <= '${max}'::date`;
59
+ return { op: 'between', query, extra };
60
+ }
61
+
62
+ // my rows
63
+ if (value === 'me' && uid && fieldType === 'text') {
64
+ return { op: '=', query: extra ? `uid = '${uid}'` : `${name}::text = '${uid}'`, extra };
65
+ }
66
+
67
+ const formatType = {
68
+ float8: 'numeric',
69
+ int4: 'numeric',
70
+ int8: 'numeric',
71
+ varchar: 'text',
72
+ bool: 'boolean',
73
+ geometry: 'geom',
74
+ }[fieldType] || 'text';
75
+
76
+ if (optimize && optimize.name !== optimize.pk) {
77
+ const val = filterType === 'text' ? `ilike '%${value}%'` : `= any('{${value}}')`;
78
+ return {
79
+ op: '~',
80
+ query: fieldType?.includes('[]')
81
+ ? `${optimize.pk} && (select array_agg(${optimize.pk}) from ${optimize.table} where ${name} ${val} )`
82
+ : `${optimize.pk} in (select ${optimize.pk} from ${optimize.table} where ${name} ${val} )`,
83
+ extra,
84
+ };
85
+ }
86
+
87
+ if (fieldType?.includes('[]')) {
88
+ return { op: 'in', query: `'{${value}}'::text[] && ${name}::text[]`, extra };
89
+ }
90
+
91
+ // multiple items of 1 param
92
+ if (value?.indexOf(',') !== -1) {
93
+ const values = value.split(',').filter((el) => el !== 'null');
94
+ if (extra && pk) {
95
+ const query = value?.indexOf('null') !== -1
96
+ ? `${pk} in (select object_id from crm.extra_data where property_key='${name}' and ( value_text is null or value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) ) )`
97
+ : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) )`;
98
+ return { op: 'in', query, extra };
99
+ }
100
+ const query = value?.indexOf('null') !== -1
101
+ ? `( ${name} is null or ${name}::text in (${values?.map((el) => `'${el}'`).join(',')}) )`
102
+ : `${name}::text in (${value.split(',')?.map((el) => `'${el}'`).join(',')})`;
103
+ return { op: 'in', query, extra };
104
+ }
105
+
106
+ // v3 filter number range, example - "100_500"
107
+ if (numberTypeList.includes(fieldType) && value?.indexOf('_') !== -1) {
108
+ const [min, max] = value.split('_');
109
+ const query = (max === 'max' ? `${name} > ${min}` : null) || (min === 'min' ? `${name} < ${max}` : null) || `${name} between ${min} and ${max}`;
110
+ return { op: 'between', query, extra };
111
+ }
112
+
113
+ // number range
114
+ if (numberTypeList.includes(fieldType) && value?.indexOf('-') !== -1) {
115
+ const [min, max] = value.split('-');
116
+ if (min === 'min' && max === 'max') return {};
117
+ const query = (max === 'max' ? `${name} > ${min}` : null) || (min === 'min' ? `${name} < ${max}` : null) || `${name} between ${min} and ${max}`;
118
+ return { op: 'between', query, extra };
119
+ }
120
+
121
+ if (['<', '>'].includes(operator)) {
122
+ const query = `${name} ${operator} '${value}'::${formatType}`;
123
+ return { op: operator, query, extra };
124
+ }
125
+
126
+ if (operator === '=' && filterType !== 'text' && !filter?.data) {
127
+ const query = {
128
+ null: `${name} is null`,
129
+ notnull: `${name} is not null`,
130
+ }[value] || `${name}::${formatType}='${value}'::${formatType}`;
131
+ return { op: '=', query, extra };
132
+ }
133
+
134
+ if (['~', '='].includes(operator)) {
135
+ const operator1 = (filterType === 'text' && (filter?.id || filter?.name) && operator === '=' ? '~' : operator);
136
+ const match = operator1 === '=' ? `='${value}'` : `ilike '%${value}%'`;
137
+ if (extra && pk) {
138
+ const query = data && sql
139
+ ? `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text in ( ( with q(id,name) as (${sql}) select id from q where name ${match})))`
140
+ : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text ${match})`
141
+ return { op: 'ilike', query, extra };
142
+ }
143
+
144
+ const query = filter?.data && filter?.sql
145
+ ? `${filter?.name || filter?.id} in ( ( with q(id,name) as (${filter?.sql}) select id from q where name::text ${match}) )` // filter with cls
146
+ : `${name}::text ${match}`; // simple filter
147
+ // console.log(query);
148
+ return { op: 'ilike', query };
149
+ }
150
+
151
+ // json
152
+ if (name.includes('.')) {
153
+ const [col, prop] = name.split('.');
154
+ const query = ` ${col}->>'${prop}' in ('${value.join("','")}')`;
155
+ return { op: 'in', query, extra };
156
+ }
157
+
158
+ // geometry
159
+ if (['geometry'].includes(fieldType)) {
160
+ const bbox = value[0].split('_');
161
+
162
+ if (bbox?.length === 4) {
163
+ const query = ` ${name} && 'box(${bbox[0]} ${bbox[1]},${bbox[2]} ${bbox[3]})'::box2d `;
164
+ return { op: '&&', query, extra };
165
+ }
166
+ }
167
+ return {};
168
+ }
169
+
170
+ export default formatValue;
@@ -1,13 +1,13 @@
1
- async function getCustomQuery({
2
- pg, table, customFilter,
3
- }) {
4
- if (!customFilter) return null;
5
- const customFilterList = customFilter?.split(',')?.map((el) => el?.split('_').pop());
6
- const { property_json: customFilterSQL } = await pg.one(`select json_agg(json_build_object('id',property_id,'name',property_key,'query',property_text)
7
- ) as property_json from admin.properties where property_key is not null and property_entity='customQuery' and object_id=$1`, [table]);
8
- const data = customFilterSQL?.length ? customFilterSQL.filter((el) => customFilterList.includes(el.id)) || [] : [];
9
- const customQuery = data?.map((el) => el.query).join(' and ');
10
- return `${customQuery}`;
11
- }
12
-
13
- export default getCustomQuery;
1
+ async function getCustomQuery({
2
+ pg, table, customFilter,
3
+ }) {
4
+ if (!customFilter) return null;
5
+ const customFilterList = customFilter?.split(',')?.map((el) => el?.split('_').pop());
6
+ const { property_json: customFilterSQL } = await pg.one(`select json_agg(json_build_object('id',property_id,'name',property_key,'query',property_text)
7
+ ) as property_json from admin.properties where property_key is not null and property_entity='customQuery' and object_id=$1`, [table]);
8
+ const data = customFilterSQL?.length ? customFilterSQL.filter((el) => customFilterList.includes(el.id)) || [] : [];
9
+ const customQuery = data?.map((el) => el.query).join(' and ');
10
+ return `${customQuery}`;
11
+ }
12
+
13
+ export default getCustomQuery;
@@ -1,64 +1,64 @@
1
- /* eslint-disable no-continue */
2
-
3
- /**
4
- * @param {Number} opt.json - (1|0) 1 - Результат - Object, 0 - String
5
- * @param {String} opt.query - запит до таблиці
6
- * @param {String} opt.hash - інформація з хешу по запиту
7
- */
8
-
9
- import formatValue from './formatValue.js';
10
-
11
- function getQuery({
12
- pg, filter: filterStr, table, tableSQL, fields, filterList,
13
- }) {
14
- if (!filterStr) return null; // filter list API
15
-
16
- const mainOperators = ['=', '~', '>', '<'];
17
-
18
- const filterQueryArray = decodeURI(filterStr?.replace(/(^,)|(,$)/g, '')).replace(/'/g, '').split(/[;|]/);
19
-
20
- const resultList = [];
21
-
22
- for (let i = 0; i < filterQueryArray.length; i += 1) {
23
- const item = filterQueryArray[i];
24
- const operator = mainOperators?.find((el) => item.indexOf(el) !== -1) || '=';
25
- const [name] = item.split(operator);
26
-
27
- // skip already added filter
28
- if (resultList.find((el) => el.name === name)) {
29
- continue;
30
- }
31
-
32
- // filter
33
- const filter = filterList?.find((el) => [el.id, el.name].includes(name)) || { type: 'text' };
34
-
35
- // find all value
36
- const value = filterQueryArray.filter((el) => el.startsWith(name)).map((el) => el.substring(name.length + 1)).join(',');
37
-
38
- const optimize = fields?.find((el) => el.name === name) ? null : tableSQL.find((el) => el.name === name);
39
-
40
- // find field and skip not exists
41
- const { dataTypeID } = fields?.find((el) => el.name === name) || fields?.find((el) => el.name === optimize?.pk) || {};
42
-
43
- // format query
44
- const { op, query, filterType, fieldType } = formatValue({
45
- pg,
46
- table,
47
- filter,
48
- optimize,
49
- name,
50
- value: decodeURIComponent(value),
51
- operator,
52
- dataTypeID,
53
- }) || {};
54
- if (!query) continue;
55
-
56
- resultList.push({
57
- name, value, query, operator: op, filterType, type: fieldType,
58
- });
59
- }
60
-
61
- return resultList;
62
- }
63
-
64
- export default getQuery;
1
+ /* eslint-disable no-continue */
2
+
3
+ /**
4
+ * @param {Number} opt.json - (1|0) 1 - Результат - Object, 0 - String
5
+ * @param {String} opt.query - запит до таблиці
6
+ * @param {String} opt.hash - інформація з хешу по запиту
7
+ */
8
+
9
+ import formatValue from './formatValue.js';
10
+
11
+ function getQuery({
12
+ pg, filter: filterStr, table, tableSQL, fields, filterList,
13
+ }) {
14
+ if (!filterStr) return null; // filter list API
15
+
16
+ const mainOperators = ['=', '~', '>', '<'];
17
+
18
+ const filterQueryArray = decodeURI(filterStr?.replace(/(^,)|(,$)/g, '')).replace(/'/g, '').split(/[;|]/);
19
+
20
+ const resultList = [];
21
+
22
+ for (let i = 0; i < filterQueryArray.length; i += 1) {
23
+ const item = filterQueryArray[i];
24
+ const operator = mainOperators?.find((el) => item.indexOf(el) !== -1) || '=';
25
+ const [name] = item.split(operator);
26
+
27
+ // skip already added filter
28
+ if (resultList.find((el) => el.name === name)) {
29
+ continue;
30
+ }
31
+
32
+ // filter
33
+ const filter = filterList?.find((el) => [el.id, el.name].includes(name)) || { type: 'text' };
34
+
35
+ // find all value
36
+ const value = filterQueryArray.filter((el) => el.startsWith(name)).map((el) => el.substring(name.length + 1)).join(',');
37
+
38
+ const optimize = fields?.find((el) => el.name === name) ? null : tableSQL.find((el) => el.name === name);
39
+
40
+ // find field and skip not exists
41
+ const { dataTypeID } = fields?.find((el) => el.name === name) || fields?.find((el) => el.name === optimize?.pk) || {};
42
+
43
+ // format query
44
+ const { op, query, filterType, fieldType } = formatValue({
45
+ pg,
46
+ table,
47
+ filter,
48
+ optimize,
49
+ name,
50
+ value: decodeURIComponent(value),
51
+ operator,
52
+ dataTypeID,
53
+ }) || {};
54
+ if (!query) continue;
55
+
56
+ resultList.push({
57
+ name, value, query, operator: op, filterType, type: fieldType,
58
+ });
59
+ }
60
+
61
+ return resultList;
62
+ }
63
+
64
+ export default getQuery;
@@ -1,12 +1,12 @@
1
- function getOptimizedQuery({ body, table, q }, count) {
2
- const order = body?.orderby || body?.order ? `order by ${body?.orderby || body?.order}` : '';
3
-
4
- const tableName = body?.table || body?.model || table;
5
-
6
- const sqlList = body?.sql?.filter((el) => !el.disabled && el?.sql?.replace && (count ? el.count !== false : true))
7
- .map((el) => ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `).join(' ');
8
-
9
- return `(select * from ${tableName} ${sqlList ? ` t ${sqlList}` : ''} where 1=1 and ${q?.replace('q.', 't.') || '1=1'} ${order})q`;
10
- }
11
-
12
- export default getOptimizedQuery;
1
+ function getOptimizedQuery({ body, table, q }, count) {
2
+ const order = body?.orderby || body?.order ? `order by ${body?.orderby || body?.order}` : '';
3
+
4
+ const tableName = body?.table || body?.model || table;
5
+
6
+ const sqlList = body?.sql?.filter((el) => !el.disabled && el?.sql?.replace && (count ? el.count !== false : true))
7
+ .map((el) => ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `).join(' ');
8
+
9
+ return `(select * from ${tableName} ${sqlList ? ` t ${sqlList}` : ''} where 1=1 and ${q?.replace('q.', 't.') || '1=1'} ${order})q`;
10
+ }
11
+
12
+ export default getOptimizedQuery;
@@ -1,34 +1,34 @@
1
- function getTable(table) {
2
- const result = table?.toLowerCase()?.replace(/[\n\r]+/g, ' ')?.split(' from ')?.filter((el) => /^[a-z0-9_]+\.[a-z0-9_]+/.test(el))
3
- ?.map((el) => el.split(/[ )]/)[0]);
4
- return result;
5
- }
6
-
7
- /**
8
- * @param {Number} opt.json - (1|0) 1 - Результат - Object, 0 - String
9
- * @param {String} opt.query - запит до таблиці
10
- * @param {String} opt.hash - інформація з хешу по запиту
11
- */
12
- const tableSql = {};
13
- async function getTableSql({
14
- pg, body, table, fields,
15
- }) {
16
- if (tableSql[table]) return tableSql[table];
17
-
18
- const fieldList = fields.map((el) => el.name);
19
-
20
- const tableList = body?.sql?.map((el) => getTable(el.sql)).reduce((acc, el) => acc.concat(el), []).filter((el) => fieldList.includes(pg.pk[el]));
21
-
22
- if (!tableList) { tableSql[table] = []; return []; }
23
-
24
- const data = await Promise.all(tableList?.map(async (tableEl) => {
25
- const { fields: fieldsEl } = await pg.query(`select * from ${tableEl} limit 0`);
26
- return fieldsEl.map((el) => ({ name: el.name, table: tableEl, pk: pg.pk[tableEl] }));
27
- }));
28
-
29
- tableSql[table] = data.reduce((acc, el) => acc.concat(el), []);
30
-
31
- return tableSql[table];
32
- }
33
-
34
- export default getTableSql;
1
+ function getTable(table) {
2
+ const result = table?.toLowerCase()?.replace(/[\n\r]+/g, ' ')?.split(' from ')?.filter((el) => /^[a-z0-9_]+\.[a-z0-9_]+/.test(el))
3
+ ?.map((el) => el.split(/[ )]/)[0]);
4
+ return result;
5
+ }
6
+
7
+ /**
8
+ * @param {Number} opt.json - (1|0) 1 - Результат - Object, 0 - String
9
+ * @param {String} opt.query - запит до таблиці
10
+ * @param {String} opt.hash - інформація з хешу по запиту
11
+ */
12
+ const tableSql = {};
13
+ async function getTableSql({
14
+ pg, body, table, fields,
15
+ }) {
16
+ if (tableSql[table]) return tableSql[table];
17
+
18
+ const fieldList = fields.map((el) => el.name);
19
+
20
+ const tableList = body?.sql?.map((el) => getTable(el.sql)).reduce((acc, el) => acc.concat(el), []).filter((el) => fieldList.includes(pg.pk[el]));
21
+
22
+ if (!tableList) { tableSql[table] = []; return []; }
23
+
24
+ const data = await Promise.all(tableList?.map(async (tableEl) => {
25
+ const { fields: fieldsEl } = await pg.query(`select * from ${tableEl} limit 0`);
26
+ return fieldsEl.map((el) => ({ name: el.name, table: tableEl, pk: pg.pk[tableEl] }));
27
+ }));
28
+
29
+ tableSql[table] = data.reduce((acc, el) => acc.concat(el), []);
30
+
31
+ return tableSql[table];
32
+ }
33
+
34
+ export default getTableSql;