@opengis/fastify-table 1.2.42 → 1.2.44

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.
Files changed (54) hide show
  1. package/README.md +86 -86
  2. package/package.json +1 -1
  3. package/server/migrations/cls.sql +39 -39
  4. package/server/plugins/cron/funcs/addCron.js +130 -130
  5. package/server/plugins/cron/index.js +6 -6
  6. package/server/plugins/crud/funcs/dataDelete.js +7 -1
  7. package/server/plugins/crud/funcs/dataInsert.js +7 -0
  8. package/server/plugins/crud/funcs/dataUpdate.js +4 -1
  9. package/server/plugins/crud/funcs/getOpt.js +13 -13
  10. package/server/plugins/crud/funcs/setOpt.js +21 -21
  11. package/server/plugins/crud/funcs/setToken.js +44 -44
  12. package/server/plugins/crud/funcs/utils/getFolder.js +10 -10
  13. package/server/plugins/crud/index.js +23 -23
  14. package/server/plugins/extra/extraData.js +51 -28
  15. package/server/plugins/extra/extraDataGet.js +18 -4
  16. package/server/plugins/hook/index.js +8 -8
  17. package/server/plugins/logger/errorStatus.js +19 -19
  18. package/server/plugins/logger/index.js +21 -21
  19. package/server/plugins/migration/index.js +7 -7
  20. package/server/plugins/pg/pgClients.js +21 -21
  21. package/server/plugins/policy/sqlInjection.js +33 -33
  22. package/server/plugins/redis/client.js +8 -8
  23. package/server/plugins/redis/funcs/redisClients.js +3 -3
  24. package/server/plugins/redis/index.js +17 -17
  25. package/server/plugins/table/funcs/getFilterSQL/index.js +126 -96
  26. package/server/plugins/table/funcs/getFilterSQL/util/formatValue.js +37 -16
  27. package/server/plugins/table/funcs/getFilterSQL/util/getCustomQuery.js +13 -13
  28. package/server/plugins/table/funcs/getFilterSQL/util/getOptimizedQuery.js +12 -12
  29. package/server/plugins/table/funcs/getFilterSQL/util/getTableSql.js +34 -34
  30. package/server/plugins/table/funcs/getTemplates.js +19 -19
  31. package/server/plugins/table/funcs/gisIRColumn.js +82 -82
  32. package/server/plugins/table/funcs/loadTemplate.js +1 -1
  33. package/server/plugins/table/funcs/loadTemplatePath.js +1 -1
  34. package/server/plugins/table/funcs/metaFormat/index.js +3 -2
  35. package/server/plugins/table/funcs/userTemplateDir.js +1 -1
  36. package/server/plugins/table/index.js +13 -13
  37. package/server/plugins/util/index.js +7 -7
  38. package/server/routes/cron/index.js +14 -14
  39. package/server/routes/crud/controllers/table.js +88 -88
  40. package/server/routes/logger/controllers/logger.file.js +92 -92
  41. package/server/routes/logger/controllers/utils/checkUserAccess.js +19 -19
  42. package/server/routes/logger/controllers/utils/getRootDir.js +26 -26
  43. package/server/routes/logger/index.js +17 -17
  44. package/server/routes/properties/controllers/properties.add.js +55 -55
  45. package/server/routes/properties/controllers/properties.get.js +17 -17
  46. package/server/routes/properties/index.js +16 -16
  47. package/server/routes/table/controllers/data.js +24 -3
  48. package/server/routes/table/controllers/filter.js +3 -1
  49. package/server/routes/table/controllers/form.js +42 -42
  50. package/server/routes/table/controllers/search.js +74 -74
  51. package/server/routes/table/index.js +29 -29
  52. package/server/routes/table/schema.js +64 -64
  53. package/server/routes/util/controllers/status.monitor.js +8 -8
  54. package/server/routes/util/index.js +11 -11
@@ -1,96 +1,126 @@
1
- import getTemplate from '../getTemplate.js';
2
- import pgClients from '../../../pg/pgClients.js';
3
-
4
- // import { getTemplate, pgClients } from '../../../../../utils.js';
5
-
6
- // filter util
7
- import getTableSql from './util/getTableSql.js';
8
- import getFilterQuery from './util/getFilterQuery.js';
9
- import getOptimizedQuery from './util/getOptimizedQuery.js';
10
-
11
- async function getFilterSQL({
12
- table, filter, pg = pgClients.client, search, filterList, query, custom, state,
13
- }) {
14
- if (!table) return { error: 'param table is required', status: 400 };
15
-
16
- const body = await getTemplate('table', table);
17
-
18
- const sqlList = body?.sql?.length
19
- ? body?.sql?.filter((el) => !el.disabled && el?.sql?.replace)
20
- .map((el, i) => {
21
- Object.assign(el, { name: el.name || `t${i + 1}` });
22
- return ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `;
23
- }).join(' ')
24
- : '';
25
- const fieldQuery = `select * from ${body?.table || table} ${sqlList ? ` t ${sqlList}` : ''} limit 0`;
26
- const { fields = [] } = await pg.query(fieldQuery);
27
-
28
- const { fields: fieldsModel } = body?.table && pg.pk[body?.table] ? await pg.query(`select * from ${body.table} limit 0`) : {};
29
-
30
- const autoSearchColumn = fields?.filter((el) => pg.pgType?.[el.dataTypeID] === 'text')?.map((el) => el.name).join(',');
31
- const searchColumn = body?.search_column || body?.meta?.search || autoSearchColumn;
32
- const fieldsList = (fieldsModel || fields)?.map((el) => el.name);
33
- try {
34
- const tableSQL = await getTableSql({
35
- pg, body, table, fields,
36
- });
37
- const sval = `ilike '%${decodeURIComponent(search?.replace(/%/g, '%25')).replace(/'/g, "''").replace(/%/g, '\\%')}%'`;
38
- const searchQuery = search && searchColumn
39
- ? ` (${searchColumn.split(',')?.map((name) => {
40
- const { pk } = tableSQL.find((el) => el.name === name) || {};
41
- return pk && !fieldsList.includes(name) ? `${pk} in (select ${pk} from (${fieldQuery.replace(/limit 0/g, '')} where ${name} ${sval} )q where 1=1)` : `${name} ${sval}`;
42
- }).join(' or ')} )` : '';
43
-
44
- const filterList1 = await Promise.all((filterList || (body?.filter_list || []).concat(body?.filterInline || []).concat(body?.filterCustom || []).concat(body?.filterState || []).concat(body?.filterList || [])
45
- .concat(body?.filters || [])) /* .concat(extraFilters || []).concat(customFilters || []) */
46
-
47
- ?.map(async (el) => {
48
- if (!el?.data) return el;
49
- const cls = await getTemplate(['cls', 'select'], el.data);
50
- if (Array.isArray(cls) && cls?.length) {
51
- Object.assign(el, { options: cls });
52
- }
53
- else if (typeof (cls?.sql || cls) === 'string') {
54
- Object.assign(el, { sql: cls?.sql || cls });
55
- }
56
- return el;
57
- }));
58
-
59
- const filters = getFilterQuery({
60
- pg,
61
- filter,
62
- table,
63
- tableSQL,
64
- fields,
65
- filterList: filterList1,
66
- });
67
-
68
- // filter
69
- const customQuery = body?.filterCustom?.length && custom ? body.filterCustom?.find((el) => el.name === custom)?.sql : null;
70
- const stateQuery = body?.filterState?.length && state ? body.filterState?.find((el) => el.name === state)?.sql : null;
71
-
72
- const filterQuery = filters?.filter((el) => el.query)?.map((el) => `${el.query} `).join(' and ');
73
- const q = [body?.query, query, searchQuery, filterQuery, stateQuery, customQuery].filter((el) => el).join(' and ');
74
-
75
- // table
76
- const modelQuery = body?.model || body?.table || table;
77
- const optimizedSQL = `select * from ${getOptimizedQuery({ body, table, q })} `;
78
- const tableCount = getOptimizedQuery({ body, table, q }, true);
79
- // console.log(optimizedSQL);
80
- return {
81
- filterList,
82
-
83
- q,
84
- optimizedSQL,
85
- tableCount,
86
- table: modelQuery,
87
- // filter parts
88
- searchQuery,
89
- };
90
- }
91
- catch (err) {
92
- throw new Error(err.toString());
93
- }
94
- }
95
-
96
- export default getFilterSQL;
1
+ import getTemplate from '../getTemplate.js';
2
+ import pgClients from '../../../pg/pgClients.js';
3
+ import config from '../../../../../config.js';
4
+
5
+ // filter util
6
+ import getTableSql from './util/getTableSql.js';
7
+ import getFilterQuery from './util/getFilterQuery.js';
8
+ import getOptimizedQuery from './util/getOptimizedQuery.js';
9
+
10
+ const defaultTable = 'crm.extra_data';
11
+
12
+ function getExtraQuery(mainColumns, extraColumns, schema, table, pk, mode = 'property') {
13
+ const keysExtra = Object.keys(schema || {}).filter(key => !mainColumns.map(el => el?.name).includes(key) && extraColumns.map(el => el?.name).includes(key));
14
+ if (mode === 'column') {
15
+ return { q: `left join lateral (select ${keysExtra.map(key => `"${key}"`).join(',')} from ${table} where ${pk}=t.${pk} limit 1) extra on 1=1`, keysExtra };
16
+ }
17
+ return {
18
+ q: `left join lateral (select ${keysExtra.map(key => `"${key}"`).join(',')} from ${table} where ${pk}=t.${pk} limit 1) extra on 1=1`,
19
+ keysExtra,
20
+ };
21
+ }
22
+
23
+ export default async function getFilterSQL({
24
+ table, filter, pg = pgClients.client, search, filterList, query, custom, state,
25
+ }) {
26
+ if (!table) return { error: 'param table is required', status: 400 };
27
+
28
+ const body = await getTemplate('table', table);
29
+ const loadTemplate = body?.form ? await getTemplate('form', body?.form) : null;
30
+ const { extra } = loadTemplate || {};
31
+
32
+ const extraDataTable = extra
33
+ ? config.extraData?.[body.table]
34
+ || config.extraData?.[mainTable.split('.').shift()]
35
+ || config.extraData?.['default']
36
+ || config.extraData
37
+ || defaultTable
38
+ : undefined;
39
+
40
+ const { fields: fieldsExtra = [] } = pg.pk?.[extraDataTable] ? await pg.query(`select * from ${extraDataTable} limit 0`) : {};
41
+ const mode = fieldsExtra.find((col) => col?.name === pg.pk?.[body?.table || table]) ? 'column' : 'property';
42
+
43
+ const { fields: fieldsModel = [] } = body?.table && pg.pk[body?.table] ? await pg.query(`select * from ${body.table} limit 0`) : {};
44
+ const { q: extraSqlList, keysExtra } = pg.pk?.[extraDataTable] && pg.pk?.[body?.table]
45
+ ? getExtraQuery(fieldsModel, fieldsExtra, loadTemplate?.schema, extraDataTable, pg.pk[body?.table || table], mode)
46
+ : {};
47
+
48
+ // console.log('extra getFilterSQL', extraDataTable, pg.pk?.[extraDataTable]);
49
+
50
+ const sqlList = body?.sql?.length
51
+ ? body?.sql?.filter((el) => !el.disabled && el?.sql?.replace)
52
+ .map((el, i) => {
53
+ Object.assign(el, { name: el.name || `t${i + 1}` });
54
+ return ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `;
55
+ }).join(' ')
56
+ : '';
57
+ const fieldQuery = `select * from ${body?.table || table} t ${sqlList || ''} ${extraSqlList || ''} limit 0`;
58
+ const { fields = [] } = await pg.query(fieldQuery);
59
+ const autoSearchColumn = fields?.filter((el) => pg.pgType?.[el.dataTypeID] === 'text')?.map((el) => el.name).join(',');
60
+ const searchColumn = body?.search_column || body?.meta?.search || autoSearchColumn;
61
+ const fieldsList = (fieldsModel || fields)?.map((el) => el.name);
62
+ try {
63
+ const tableSQL = await getTableSql({
64
+ pg, body, table, fields,
65
+ });
66
+ const sval = `ilike '%${decodeURIComponent(search?.replace(/%/g, '%25')).replace(/'/g, "''").replace(/%/g, '\\%')}%'`;
67
+ const searchQuery = search && searchColumn
68
+ ? ` (${searchColumn.split(',')?.map((name) => {
69
+ const { pk } = tableSQL.find((el) => el.name === name) || {};
70
+ return pk && !fieldsList.includes(name) ? `${pk} in (select ${pk} from (${fieldQuery.replace(/limit 0/g, '')} where ${name} ${sval} )q where 1=1)` : `${name} ${sval}`;
71
+ }).join(' or ')} )` : '';
72
+
73
+ const filterList1 = await Promise.all((filterList || (body?.filter_list || []).concat(body?.filterInline || []).concat(body?.filterCustom || []).concat(body?.filterState || []).concat(body?.filterList || [])
74
+ .concat(body?.filters || [])) /* .concat(extraFilters || []).concat(customFilters || []) */
75
+
76
+ ?.map(async (el) => {
77
+ if (el.name && keysExtra?.includes?.(el.name)) {
78
+ Object.assign(el, { extra: { table: extraDataTable, mode } });
79
+ }
80
+ if (!el?.data) return el;
81
+ const cls = await getTemplate(['cls', 'select'], el.data);
82
+ if (Array.isArray(cls) && cls?.length) {
83
+ Object.assign(el, { options: cls });
84
+ }
85
+ else if (typeof (cls?.sql || cls) === 'string') {
86
+ Object.assign(el, { sql: cls?.sql || cls });
87
+ }
88
+ return el;
89
+ }));
90
+
91
+ const filters = getFilterQuery({
92
+ pg,
93
+ filter,
94
+ table: body?.table || table,
95
+ tableSQL,
96
+ fields,
97
+ filterList: filterList1,
98
+ });
99
+
100
+ // filter
101
+ const customQuery = body?.filterCustom?.length && custom ? body.filterCustom?.find((el) => el.name === custom)?.sql : null;
102
+ const stateQuery = body?.filterState?.length && state ? body.filterState?.find((el) => el.name === state)?.sql : null;
103
+
104
+ const filterQuery = filters?.filter((el) => el.query)?.map((el) => `${el.query} `).join(' and ');
105
+ const q = [body?.query, query, searchQuery, filterQuery, stateQuery, customQuery].filter((el) => el).join(' and ');
106
+
107
+ // table
108
+ const modelQuery = body?.model || body?.table || table;
109
+ const optimizedSQL = `select * from ${getOptimizedQuery({ body, extraSqlList, table, q })} `;
110
+ const tableCount = getOptimizedQuery({ body, extraSqlList, table, q }, true);
111
+ // console.log(optimizedSQL);
112
+ return {
113
+ filterList,
114
+
115
+ q,
116
+ optimizedSQL,
117
+ tableCount,
118
+ table: modelQuery,
119
+ // filter parts
120
+ searchQuery,
121
+ };
122
+ }
123
+ catch (err) {
124
+ throw new Error(err.toString());
125
+ }
126
+ }
@@ -29,7 +29,9 @@ function formatValue({
29
29
  const pk = pg?.pk && table ? pg.pk[table] : undefined;
30
30
 
31
31
  if (!dataTypeID && !extra) return {};
32
- const fieldType = extra ? pg.pgType?.[{ Date: 1114 }[filter?.type] || 25] : pg.pgType?.[dataTypeID];
32
+ // const fieldType = extra ? pg.pgType?.[{ Date: 1114 }[filter?.type] || 25] : pg.pgType?.[dataTypeID];
33
+ const fieldType = pg.pgType?.[dataTypeID] || pg.pgType?.[{ Date: 1114 }[filter?.type || ''] || 25];
34
+
33
35
  if (!name || !value || !fieldType) return {};
34
36
  const filterType = filter.type?.toLowerCase();
35
37
 
@@ -61,15 +63,21 @@ function formatValue({
61
63
  if (!isValidDate(startDate) || !isValidDate(endDate)) {
62
64
  return { op: 'between', query: 'false', extra };
63
65
  }
64
- const query = extra && pk
65
- ? `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_date::date >= '${min}'::date and value_date::date <= '${max}'::date)`
66
- : `${name}::date >= '${min}'::date and ${name}::date <= '${max}'::date`;
67
- return { op: 'between', query, extra };
66
+ if (extra?.table && pk) {
67
+ const query = extra?.mode === 'property' && pk
68
+ ? `${pk} in (select object_id from ${extra.table} where property_key='${name}' and value_date::date >= '${min}'::date and value_date::date <= '${max}'::date)`
69
+ : `${pk} in (select ${pk} from ${extra.table} where ${name}::date >= '${min}'::date and ${name}::date <= '${max}'::date)`;
70
+ return { op: 'between', query, extra };
71
+ }
72
+ return { op: 'between', query: `${name}::date >= '${min}'::date and ${name}::date <= '${max}'::date`, extra };
68
73
  }
69
74
 
70
75
  // my rows
71
76
  if (value === 'me' && uid && fieldType === 'text') {
72
- return { op: '=', query: extra ? `uid = '${uid}'` : `${name}::text = '${uid}'`, extra };
77
+ const query = extra?.table && pk
78
+ ? `${pk} in (select ${pk} from ${extra.table} where uid = '${uid}')`
79
+ : `${name}::text = '${uid}'`;
80
+ return { op: '=', query, extra };
73
81
  }
74
82
 
75
83
  const formatType = {
@@ -99,12 +107,18 @@ function formatValue({
99
107
  // multiple items of 1 param
100
108
  if (value?.indexOf(',') !== -1) {
101
109
  const values = value.split(',').filter((el) => el !== 'null');
102
- if (extra && pk) {
110
+ if (extra?.mode === 'property' && pk) {
111
+ const query = value?.indexOf('null') !== -1
112
+ ? `${pk} in (select object_id from ${extra?.table} where property_key='${name}' and ( value_text is null or value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) ) )`
113
+ : `${pk} in (select object_id from ${extra?.table} where property_key='${name}' and value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) )`;
114
+ return { op: 'in', query, extra };
115
+ } else if (extra?.mode === 'property' && pk) {
103
116
  const query = value?.indexOf('null') !== -1
104
- ? `${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(',')}) ) )`
105
- : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text in (${values?.map((el) => `'"${el}"'`).join(',')}) )`;
117
+ ? `${pk} in (select ${pk} from ${extra?.table} where ( "${name}" is null or "${name}" in (${values?.map((el) => `'"${el}"'`).join(',')}) ) )`
118
+ : `${pk} in (select ${pk} from ${extra?.table} where "${name}" in (${values?.map((el) => `'"${el}"'`).join(',')}) )`;
106
119
  return { op: 'in', query, extra };
107
120
  }
121
+
108
122
  const query = value?.indexOf('null') !== -1
109
123
  ? `( ${name} is null or ${name}::text in (${values?.map((el) => `'${el}'`).join(',')}) )`
110
124
  : `${name}::text in (${value.split(',')?.map((el) => `'${el}'`).join(',')})`;
@@ -142,19 +156,26 @@ function formatValue({
142
156
  if (['~', '='].includes(operator)) {
143
157
  const operator1 = (filterType === 'text' && (filter?.id || filter?.name) && operator === '=' ? '~' : operator);
144
158
  const matchNull = { null: 'is null', notnull: 'is not null' }[value];
145
- const match = matchNull || ((operator1 === '=' || filterType === 'autocomplete') ? `='${value}'` : `ilike '%${value}%'`);
146
- if (extra && pk) {
159
+ const matchBoolean = fieldType === 'boolean' ? { true: 'is true', false: 'is false' }[value] : null;
160
+ const match = matchNull || matchBoolean || ((operator1 === '=' || filterType === 'autocomplete') ? `='${value}'` : `ilike '%${value}%'`);
161
+
162
+ if (extra?.mode === 'property' && pk) {
163
+ const query = data && sql
164
+ ? `${pk} in (select object_id from ${extra?.table} where property_key='${name}' and value_text in ( ( with q(id,name) as (${sql}) select id from q where ${filterType === 'autocomplete' ? 'id' : 'name'} ${match})))`
165
+ : `${pk} in (select object_id from ${extra?.table} where property_key='${name}' and value_text ${match})`;
166
+ return { op: 'ilike', query, extra };
167
+ } else if (extra?.mode === 'column') {
147
168
  const query = data && sql
148
- ? `${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 ${filterType === 'autocomplete' ? 'id' : 'name'} ${match})))`
149
- : `${pk} in (select object_id from crm.extra_data where property_key='${name}' and value_text ${match})`;
169
+ ? `${pk} in (select ${pk} from ${extra?.table} where "${name}" in ( ( with q(id,name) as (${sql}) select id from q where ${filterType === 'autocomplete' ? 'id' : 'name'} ${match})))`
170
+ : `${pk} in (select ${pk} from ${extra?.table} where "${name}" ${match})`;
150
171
  return { op: 'ilike', query, extra };
151
172
  }
152
173
 
153
- const query = filter?.data && filter?.sql
154
- ? `${filter?.name || filter?.id} in ( ( with q(id,name) as (${filter?.sql}) select id from q where ${filterType === 'autocomplete' ? 'id' : 'name'}::text ${match}) )` // filter with cls
174
+ const query = data && sql
175
+ ? `${name || filter?.id} in ( ( with q(id,name) as (${sql}) select id from q where ${filterType === 'autocomplete' ? 'id' : 'name'}::text ${match}) )` // filter with cls
155
176
  : `${name}::text ${match}`; // simple filter
156
177
  // console.log(query);
157
- return { op: 'ilike', query };
178
+ return { op: 'ilike', query, extra };
158
179
  }
159
180
 
160
181
  // json
@@ -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,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, extraSqlList, 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} t ${sqlList ? ` ${sqlList}` : ''} ${extraSqlList || ''} 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;
@@ -1,19 +1,19 @@
1
- import fs from 'fs';
2
- import path from 'path';
3
-
4
- import config from '../../../../config.js';
5
-
6
- const loadTemplate = {};
7
-
8
- export default async function getTemplateDir(type) {
9
- if (!type) return null;
10
-
11
- const cwd = process.cwd();
12
- const typeDir = path.join(cwd, (config.templateDir || 'server/templates'), type);
13
-
14
- if (!loadTemplate[type]) {
15
- const typeList = fs.existsSync(typeDir) ? fs.readdirSync(typeDir) : [];
16
- loadTemplate[type] = typeList;
17
- }
18
- return loadTemplate[type];
19
- }
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+
4
+ import config from '../../../../config.js';
5
+
6
+ const loadTemplate = {};
7
+
8
+ export default async function getTemplateDir(type) {
9
+ if (!type) return null;
10
+
11
+ const cwd = process.cwd();
12
+ const typeDir = path.join(cwd, (config.templateDir || 'server/templates'), type);
13
+
14
+ if (!loadTemplate[type]) {
15
+ const typeList = fs.existsSync(typeDir) ? fs.readdirSync(typeDir) : [];
16
+ loadTemplate[type] = typeList;
17
+ }
18
+ return loadTemplate[type];
19
+ }