@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.
- package/README.md +86 -86
- package/package.json +1 -1
- package/server/migrations/cls.sql +39 -39
- package/server/plugins/cron/funcs/addCron.js +130 -130
- package/server/plugins/cron/index.js +6 -6
- package/server/plugins/crud/funcs/dataDelete.js +7 -1
- package/server/plugins/crud/funcs/dataInsert.js +7 -0
- package/server/plugins/crud/funcs/dataUpdate.js +4 -1
- package/server/plugins/crud/funcs/getOpt.js +13 -13
- package/server/plugins/crud/funcs/setOpt.js +21 -21
- package/server/plugins/crud/funcs/setToken.js +44 -44
- package/server/plugins/crud/funcs/utils/getFolder.js +10 -10
- package/server/plugins/crud/index.js +23 -23
- package/server/plugins/extra/extraData.js +51 -28
- package/server/plugins/extra/extraDataGet.js +18 -4
- package/server/plugins/hook/index.js +8 -8
- package/server/plugins/logger/errorStatus.js +19 -19
- package/server/plugins/logger/index.js +21 -21
- package/server/plugins/migration/index.js +7 -7
- package/server/plugins/pg/pgClients.js +21 -21
- package/server/plugins/policy/sqlInjection.js +33 -33
- package/server/plugins/redis/client.js +8 -8
- package/server/plugins/redis/funcs/redisClients.js +3 -3
- package/server/plugins/redis/index.js +17 -17
- package/server/plugins/table/funcs/getFilterSQL/index.js +126 -96
- package/server/plugins/table/funcs/getFilterSQL/util/formatValue.js +37 -16
- package/server/plugins/table/funcs/getFilterSQL/util/getCustomQuery.js +13 -13
- package/server/plugins/table/funcs/getFilterSQL/util/getOptimizedQuery.js +12 -12
- package/server/plugins/table/funcs/getFilterSQL/util/getTableSql.js +34 -34
- package/server/plugins/table/funcs/getTemplates.js +19 -19
- package/server/plugins/table/funcs/gisIRColumn.js +82 -82
- package/server/plugins/table/funcs/loadTemplate.js +1 -1
- package/server/plugins/table/funcs/loadTemplatePath.js +1 -1
- package/server/plugins/table/funcs/metaFormat/index.js +3 -2
- package/server/plugins/table/funcs/userTemplateDir.js +1 -1
- package/server/plugins/table/index.js +13 -13
- package/server/plugins/util/index.js +7 -7
- package/server/routes/cron/index.js +14 -14
- package/server/routes/crud/controllers/table.js +88 -88
- package/server/routes/logger/controllers/logger.file.js +92 -92
- package/server/routes/logger/controllers/utils/checkUserAccess.js +19 -19
- package/server/routes/logger/controllers/utils/getRootDir.js +26 -26
- package/server/routes/logger/index.js +17 -17
- package/server/routes/properties/controllers/properties.add.js +55 -55
- package/server/routes/properties/controllers/properties.get.js +17 -17
- package/server/routes/properties/index.js +16 -16
- package/server/routes/table/controllers/data.js +24 -3
- package/server/routes/table/controllers/filter.js +3 -1
- package/server/routes/table/controllers/form.js +42 -42
- package/server/routes/table/controllers/search.js +74 -74
- package/server/routes/table/index.js +29 -29
- package/server/routes/table/schema.js +64 -64
- package/server/routes/util/controllers/status.monitor.js +8 -8
- 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
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
import
|
|
8
|
-
import
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
})
|
|
14
|
-
if (
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
const
|
|
29
|
-
|
|
30
|
-
const
|
|
31
|
-
|
|
32
|
-
const
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
fields,
|
|
65
|
-
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
const
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
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
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
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
|
-
|
|
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
|
|
105
|
-
: `${pk} in (select
|
|
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
|
|
146
|
-
|
|
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
|
|
149
|
-
: `${pk} in (select
|
|
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 =
|
|
154
|
-
? `${
|
|
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 ? `
|
|
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
|
+
}
|