@opengis/fastify-table 1.2.41 → 1.2.43
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/package.json +1 -1
- 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/extra/extraData.js +84 -0
- package/server/plugins/extra/extraDataGet.js +58 -0
- package/server/plugins/table/funcs/getData.js +8 -0
- package/server/plugins/table/funcs/getFilterSQL/index.js +42 -12
- package/server/plugins/table/funcs/getFilterSQL/util/formatValue.js +37 -16
- package/server/plugins/table/funcs/getFilterSQL/util/getOptimizedQuery.js +2 -2
- package/server/plugins/table/funcs/metaFormat/index.js +3 -2
- package/server/routes/table/controllers/data.js +12 -1
- package/server/routes/table/controllers/filter.js +3 -1
- package/utils.js +2 -0
package/package.json
CHANGED
|
@@ -2,6 +2,7 @@ import getPG from '../../pg/funcs/getPG.js';
|
|
|
2
2
|
import getMeta from '../../pg/funcs/getMeta.js';
|
|
3
3
|
import getRedis from '../../redis/funcs/getRedis.js';
|
|
4
4
|
|
|
5
|
+
import extraData from '../../extra/extraData.js';
|
|
5
6
|
import logChanges from './utils/logChanges.js';
|
|
6
7
|
|
|
7
8
|
const rclient = getRedis();
|
|
@@ -15,10 +16,15 @@ export default async function dataDelete({
|
|
|
15
16
|
if (!pg.tlist?.includes(table1)) return 'table not exist';
|
|
16
17
|
const delQuery = `delete from ${table} WHERE ${pk} = $1 returning *`;
|
|
17
18
|
|
|
19
|
+
const extraRes = await extraData({
|
|
20
|
+
table, form: tokenData?.form, id, uid,
|
|
21
|
+
}, pg);
|
|
22
|
+
|
|
18
23
|
const res = await pg.query(delQuery, [id]).then(el => el.rows?.[0] || {});
|
|
24
|
+
|
|
19
25
|
await logChanges({
|
|
20
26
|
pg, table, tokenData, referer, id, uid, type: 'DELETE',
|
|
21
27
|
});
|
|
22
28
|
rclient.incr(`pg:${table}:crud`);
|
|
23
|
-
return res;
|
|
29
|
+
return { ...res, ...extraRes || {} };
|
|
24
30
|
}
|
|
@@ -3,6 +3,7 @@ import getMeta from '../../pg/funcs/getMeta.js';
|
|
|
3
3
|
import getRedis from '../../redis/funcs/getRedis.js';
|
|
4
4
|
|
|
5
5
|
import logChanges from './utils/logChanges.js';
|
|
6
|
+
import extraData from '../../extra/extraData.js';
|
|
6
7
|
|
|
7
8
|
const rclient = getRedis();
|
|
8
9
|
|
|
@@ -37,6 +38,12 @@ export default async function dataInsert({
|
|
|
37
38
|
const res = await pg.query(insertQuery, [...filterData.map((el) => (typeof el[1] === 'object' && (!Array.isArray(el[1]) || typeof el[1]?.[0] === 'object') ? JSON.stringify(el[1]) : el[1]))]) || {};
|
|
38
39
|
|
|
39
40
|
const table1 = pg.pk[table] ? table : table.replace(/"/g, '');
|
|
41
|
+
|
|
42
|
+
const extraRes = await extraData({ table, form: tokenData?.form, id: res.rows?.[0]?.[pg.pk[table1]], data, uid }, pg);
|
|
43
|
+
if (extraRes && res?.rows?.[0]) {
|
|
44
|
+
Object.assign(res.rows[0], { ...extraRes });
|
|
45
|
+
}
|
|
46
|
+
|
|
40
47
|
await logChanges({
|
|
41
48
|
pg,
|
|
42
49
|
table,
|
|
@@ -2,6 +2,7 @@ import getPG from '../../pg/funcs/getPG.js';
|
|
|
2
2
|
import getMeta from '../../pg/funcs/getMeta.js';
|
|
3
3
|
import getRedis from '../../redis/funcs/getRedis.js';
|
|
4
4
|
|
|
5
|
+
import extraData from '../../extra/extraData.js';
|
|
5
6
|
import logChanges from './utils/logChanges.js';
|
|
6
7
|
|
|
7
8
|
const rclient = getRedis();
|
|
@@ -56,10 +57,12 @@ export default async function dataUpdate({
|
|
|
56
57
|
// console.log(updateQuery, filterValue);
|
|
57
58
|
const res = await pg.query(updateQuery, [id, ...filterValue]).then(el => el?.rows?.[0]) || {};
|
|
58
59
|
|
|
60
|
+
const extraRes = await extraData({ table, form: tokenData?.form, id, data, uid }, pg);
|
|
61
|
+
|
|
59
62
|
await logChanges({
|
|
60
63
|
pg, table, tokenData, referer, data, id, uid, type: 'UPDATE',
|
|
61
64
|
});
|
|
62
65
|
|
|
63
66
|
rclient.incr(`pg:${table}:crud`);
|
|
64
|
-
return res;
|
|
67
|
+
return { ...res, ...extraRes || {} };
|
|
65
68
|
}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { config, getMeta, pgClients, getTemplate, dataInsert } from '../../../utils.js';
|
|
2
|
+
|
|
3
|
+
const defaultTable = 'crm.extra_data';
|
|
4
|
+
|
|
5
|
+
function format(key, value, schema) {
|
|
6
|
+
if (!key || !schema?.[key]) return value;
|
|
7
|
+
if (schema?.[key]?.type && ['Number', 'Switcher'].includes(schema?.[key]?.type)) {
|
|
8
|
+
return JSON.parse(value || null);
|
|
9
|
+
}
|
|
10
|
+
return value;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export default async function extraData({
|
|
14
|
+
table, form, id, data, uid,
|
|
15
|
+
}, pg = pgClients.client) {
|
|
16
|
+
if (!id || !table) {
|
|
17
|
+
return null;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const loadTemplate = await getTemplate('form', form);
|
|
21
|
+
if (!loadTemplate?.extra) return null;
|
|
22
|
+
|
|
23
|
+
const extraDataTable = config.extraData?.[table]
|
|
24
|
+
|| config.extraData?.[table.split('.').shift()]
|
|
25
|
+
|| config.extraData?.['default']
|
|
26
|
+
|| config.extraData
|
|
27
|
+
|| defaultTable;
|
|
28
|
+
|
|
29
|
+
const { pk: mainPK, columns: mainColumns = [] } = await getMeta({ pg, table });
|
|
30
|
+
const { pk, columns: extraColumns = [] } = await getMeta({ pg, table: extraDataTable });
|
|
31
|
+
|
|
32
|
+
if (!mainPK) {
|
|
33
|
+
return { error: `table pk not found: ${table}`, status: 404 };
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (!pk) {
|
|
37
|
+
return { error: `extra table pk not found: ${extraDataTable}`, status: 404 };
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const mode = extraColumns.find((col) => col?.name === mainPK)
|
|
41
|
+
? 'column'
|
|
42
|
+
: 'property';
|
|
43
|
+
|
|
44
|
+
if (mode === 'property') {
|
|
45
|
+
Object.assign(data || {}, { object_id: id });
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
const deleteRes = mode === 'property'
|
|
49
|
+
? await pg.query(`delete from ${extraDataTable} where object_id=$1 and property_key = any($2::text[]) returning *`, [id, Object.keys(loadTemplate?.schema || {})])
|
|
50
|
+
: await pg.query(`delete from ${extraDataTable} where ${mainPK}=$1 returning *`, [id]);
|
|
51
|
+
|
|
52
|
+
if (!data) {
|
|
53
|
+
const res1 = mode === 'property'
|
|
54
|
+
? deleteRes?.rows?.reduce((acc, curr) => Object.assign(acc, { [curr.property_key]: format(curr.property_key, curr.value_text, loadTemplate?.schema) }), {})
|
|
55
|
+
: deleteRes?.rows?.[0];
|
|
56
|
+
return res1;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (mode === 'column') {
|
|
60
|
+
const filteredData = Object.keys(data)
|
|
61
|
+
.filter(key => Object.keys(loadTemplate?.schema || {}).includes(key))
|
|
62
|
+
.reduce((acc, curr) => Object.assign(acc, { [curr]: data[curr] }), {});
|
|
63
|
+
Object.assign(filteredData, { [mainPK]: id });
|
|
64
|
+
const res = await dataInsert({ pg, table: extraDataTable, data: filteredData, uid }).then(el => el.rows?.[0] || {});
|
|
65
|
+
Object.assign(res, { id: res?.[pg.pk?.[table || '']] });
|
|
66
|
+
return res;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const rows = Object.keys(data || {})
|
|
70
|
+
.filter(key => Object.keys(loadTemplate?.schema || {}).includes(key))
|
|
71
|
+
.filter(key => !mainColumns.map(el => el.name).concat('id', 'token').includes(key))
|
|
72
|
+
.map(key => ({
|
|
73
|
+
object_id: id,
|
|
74
|
+
property_key: key,
|
|
75
|
+
property_entity: table,
|
|
76
|
+
value_text: data[key],
|
|
77
|
+
}));
|
|
78
|
+
|
|
79
|
+
const res = await Promise.all(rows.map(async (row) => dataInsert({ pg, table: extraDataTable, data: row, uid }).then(el => el.rows?.[0] || {})));
|
|
80
|
+
return {
|
|
81
|
+
...res.reduce((acc, curr) => Object.assign(acc, { [curr.property_key]: format(curr.property_key, curr.value_text, loadTemplate?.schema) }), {}),
|
|
82
|
+
id: res?.[0]?.object_id,
|
|
83
|
+
};
|
|
84
|
+
}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { config, getTemplate, pgClients, getMeta } from "../../../utils.js";
|
|
2
|
+
|
|
3
|
+
const defaultTable = 'crm.extra_data';
|
|
4
|
+
|
|
5
|
+
function format(key, value, schema) {
|
|
6
|
+
if (!key || !schema?.[key]) return undefined; // skip non-existing
|
|
7
|
+
if (schema?.[key]?.type && ['Number', 'Switcher'].includes(schema?.[key]?.type)) {
|
|
8
|
+
return JSON.parse(value || null);
|
|
9
|
+
}
|
|
10
|
+
return value;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export default async function extraDataGet({
|
|
14
|
+
table: mainTable, form, id,
|
|
15
|
+
}, pg = pgClients.client) {
|
|
16
|
+
if (!id || !form || !mainTable) {
|
|
17
|
+
return null;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const loadTemplate = await getTemplate('form', form);
|
|
21
|
+
if (!loadTemplate?.extra) return null;
|
|
22
|
+
|
|
23
|
+
const table = config.extraData?.[mainTable]
|
|
24
|
+
|| config.extraData?.[mainTable.split('.').shift()]
|
|
25
|
+
|| config.extraData?.['default']
|
|
26
|
+
|| config.extraData
|
|
27
|
+
|| defaultTable;
|
|
28
|
+
|
|
29
|
+
const { pk: mainPK } = await getMeta({ pg, table: mainTable });
|
|
30
|
+
const { pk, columns: extraColumns = [] } = await getMeta({ pg, table });
|
|
31
|
+
|
|
32
|
+
if (!mainPK) {
|
|
33
|
+
return { error: `table pk not found: ${mainTable}`, status: 404 };
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (!pk) {
|
|
37
|
+
return { error: `extra table pk not found: ${table}`, status: 404 };
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const mode = extraColumns.find((col) => col?.name === mainPK)
|
|
41
|
+
? 'column'
|
|
42
|
+
: 'property';
|
|
43
|
+
|
|
44
|
+
const { rows = [] } = await pg.query(`select * from ${table} where ${mode === 'column' ? mainPK : 'object_id'}=$1`, [id]);
|
|
45
|
+
|
|
46
|
+
if (mode === 'column') {
|
|
47
|
+
const res = Object.keys(rows[0] || {})
|
|
48
|
+
.map(key => [key, format(key, rows?.[0]?.[key], loadTemplate?.schema)])
|
|
49
|
+
.filter(el => el[1])
|
|
50
|
+
.reduce((acc, curr) => Object.assign(acc, { [curr[0]]: curr[1] }), {});
|
|
51
|
+
Object.assign(res, { id: rows?.[0]?.[pg.pk?.[mainTable || '']] });
|
|
52
|
+
return res;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const res = rows.reduce((acc, curr) => Object.assign(acc, { [curr.property_key]: format(curr.property_key, curr.value_text, loadTemplate?.schema) }), {});
|
|
56
|
+
Object.assign(res, { id: rows?.[0].object_id });
|
|
57
|
+
return res;
|
|
58
|
+
};
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import routeData from '../../../routes/table/controllers/data.js';
|
|
2
|
+
|
|
3
|
+
export default async function getData({ id, table, pg, filter, limit, page, search, user }) {
|
|
4
|
+
const params = { table, id };
|
|
5
|
+
const query = { filter, limit, page, search };
|
|
6
|
+
const result = await routeData({ pg, params, query, user });
|
|
7
|
+
return result;
|
|
8
|
+
}
|
|
@@ -1,19 +1,51 @@
|
|
|
1
1
|
import getTemplate from '../getTemplate.js';
|
|
2
2
|
import pgClients from '../../../pg/pgClients.js';
|
|
3
|
-
|
|
4
|
-
// import { getTemplate, pgClients } from '../../../../../utils.js';
|
|
3
|
+
import config from '../../../../../config.js';
|
|
5
4
|
|
|
6
5
|
// filter util
|
|
7
6
|
import getTableSql from './util/getTableSql.js';
|
|
8
7
|
import getFilterQuery from './util/getFilterQuery.js';
|
|
9
8
|
import getOptimizedQuery from './util/getOptimizedQuery.js';
|
|
10
9
|
|
|
11
|
-
|
|
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({
|
|
12
24
|
table, filter, pg = pgClients.client, search, filterList, query, custom, state,
|
|
13
25
|
}) {
|
|
14
26
|
if (!table) return { error: 'param table is required', status: 400 };
|
|
15
27
|
|
|
16
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]);
|
|
17
49
|
|
|
18
50
|
const sqlList = body?.sql?.length
|
|
19
51
|
? body?.sql?.filter((el) => !el.disabled && el?.sql?.replace)
|
|
@@ -22,11 +54,8 @@ async function getFilterSQL({
|
|
|
22
54
|
return ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `;
|
|
23
55
|
}).join(' ')
|
|
24
56
|
: '';
|
|
25
|
-
const fieldQuery = `select * from ${body?.table || table} ${sqlList
|
|
57
|
+
const fieldQuery = `select * from ${body?.table || table} t ${sqlList || ''} ${extraSqlList || ''} limit 0`;
|
|
26
58
|
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
59
|
const autoSearchColumn = fields?.filter((el) => pg.pgType?.[el.dataTypeID] === 'text')?.map((el) => el.name).join(',');
|
|
31
60
|
const searchColumn = body?.search_column || body?.meta?.search || autoSearchColumn;
|
|
32
61
|
const fieldsList = (fieldsModel || fields)?.map((el) => el.name);
|
|
@@ -45,6 +74,9 @@ async function getFilterSQL({
|
|
|
45
74
|
.concat(body?.filters || [])) /* .concat(extraFilters || []).concat(customFilters || []) */
|
|
46
75
|
|
|
47
76
|
?.map(async (el) => {
|
|
77
|
+
if (el.name && keysExtra?.includes?.(el.name)) {
|
|
78
|
+
Object.assign(el, { extra: { table: extraDataTable, mode } });
|
|
79
|
+
}
|
|
48
80
|
if (!el?.data) return el;
|
|
49
81
|
const cls = await getTemplate(['cls', 'select'], el.data);
|
|
50
82
|
if (Array.isArray(cls) && cls?.length) {
|
|
@@ -59,7 +91,7 @@ async function getFilterSQL({
|
|
|
59
91
|
const filters = getFilterQuery({
|
|
60
92
|
pg,
|
|
61
93
|
filter,
|
|
62
|
-
table,
|
|
94
|
+
table: body?.table || table,
|
|
63
95
|
tableSQL,
|
|
64
96
|
fields,
|
|
65
97
|
filterList: filterList1,
|
|
@@ -74,8 +106,8 @@ async function getFilterSQL({
|
|
|
74
106
|
|
|
75
107
|
// table
|
|
76
108
|
const modelQuery = body?.model || body?.table || table;
|
|
77
|
-
const optimizedSQL = `select * from ${getOptimizedQuery({ body, table, q })} `;
|
|
78
|
-
const tableCount = getOptimizedQuery({ body, table, q }, true);
|
|
109
|
+
const optimizedSQL = `select * from ${getOptimizedQuery({ body, extraSqlList, table, q })} `;
|
|
110
|
+
const tableCount = getOptimizedQuery({ body, extraSqlList, table, q }, true);
|
|
79
111
|
// console.log(optimizedSQL);
|
|
80
112
|
return {
|
|
81
113
|
filterList,
|
|
@@ -92,5 +124,3 @@ async function getFilterSQL({
|
|
|
92
124
|
throw new Error(err.toString());
|
|
93
125
|
}
|
|
94
126
|
}
|
|
95
|
-
|
|
96
|
-
export default getFilterSQL;
|
|
@@ -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,4 +1,4 @@
|
|
|
1
|
-
function getOptimizedQuery({ body, table, q }, count) {
|
|
1
|
+
function getOptimizedQuery({ body, extraSqlList, table, q }, count) {
|
|
2
2
|
const order = body?.orderby || body?.order ? `order by ${body?.orderby || body?.order}` : '';
|
|
3
3
|
|
|
4
4
|
const tableName = body?.table || body?.model || table;
|
|
@@ -6,7 +6,7 @@ function getOptimizedQuery({ body, table, q }, count) {
|
|
|
6
6
|
const sqlList = body?.sql?.filter((el) => !el.disabled && el?.sql?.replace && (count ? el.count !== false : true))
|
|
7
7
|
.map((el) => ` left join lateral (${el.filter ? el.sql.replace(/limit 1/ig, '') : el.sql}) as ${el.name} on 1=1 `).join(' ');
|
|
8
8
|
|
|
9
|
-
return `(select * from ${tableName} ${sqlList ? `
|
|
9
|
+
return `(select * from ${tableName} t ${sqlList ? ` ${sqlList}` : ''} ${extraSqlList || ''} where 1=1 and ${q?.replace('q.', 't.') || '1=1'} ${order})q`;
|
|
10
10
|
}
|
|
11
11
|
|
|
12
12
|
export default getOptimizedQuery;
|
|
@@ -2,10 +2,11 @@ import { handlebars } from '@opengis/fastify-hb/utils.js';
|
|
|
2
2
|
|
|
3
3
|
import getTemplate from '../getTemplate.js';
|
|
4
4
|
import getSelectVal from './getSelectVal.js';
|
|
5
|
+
import pgClients from '../../../pg/pgClients.js';
|
|
5
6
|
|
|
6
7
|
export default async function metaFormat({
|
|
7
8
|
rows, table, cls, sufix = true,
|
|
8
|
-
}) {
|
|
9
|
+
}, pg = pgClients.client) {
|
|
9
10
|
const loadTable = table ? await getTemplate('table', table) : {};
|
|
10
11
|
const selectCols = Object.keys(cls || {}).map(key => ({ name: key, data: cls[key] })).concat(loadTable?.columns?.filter((e) => e.data) || []);
|
|
11
12
|
const metaCls = Object.keys(loadTable?.meta?.cls || {}).map((el) => ({ name: el, data: loadTable?.meta?.cls[el] }));
|
|
@@ -24,7 +25,7 @@ export default async function metaFormat({
|
|
|
24
25
|
const values = [...new Set(rows?.map((el) => el[attr.name]).flat())].filter((el) => (typeof el === 'boolean' ? true : el));
|
|
25
26
|
if (!values.length) return null;
|
|
26
27
|
|
|
27
|
-
const clsValues = await getSelectVal({ name: attr.data, values });
|
|
28
|
+
const clsValues = await getSelectVal({ pg, name: attr.data, values });
|
|
28
29
|
if (!clsValues) return null;
|
|
29
30
|
|
|
30
31
|
rows.forEach(el => {
|
|
@@ -2,6 +2,8 @@ import {
|
|
|
2
2
|
config, getTemplate, getFilterSQL, getMeta, metaFormat, getAccess, setToken, gisIRColumn, applyHook, handlebars, handlebarsSync, getSelect, setOpt, getOpt, pgClients,
|
|
3
3
|
} from '../../../../utils.js';
|
|
4
4
|
|
|
5
|
+
import extraDataGet from '../../../plugins/extra/extraDataGet.js';
|
|
6
|
+
|
|
5
7
|
import conditions from './utils/conditions.js';
|
|
6
8
|
|
|
7
9
|
const components = {
|
|
@@ -150,7 +152,16 @@ export default async function dataAPI(req, reply, called) {
|
|
|
150
152
|
const { total, filtered } = counts || {};
|
|
151
153
|
const agg = Object.keys(counts).filter(el => !['total', 'filtered'].includes(el)).reduce((acc, el) => ({ ...acc, [el]: counts[el] }), {});
|
|
152
154
|
|
|
153
|
-
|
|
155
|
+
if (hookData?.id || params.id) {
|
|
156
|
+
const extraData = await extraDataGet({
|
|
157
|
+
table: loadTable?.table,
|
|
158
|
+
id: hookData?.id || params.id,
|
|
159
|
+
form,
|
|
160
|
+
}, pg);
|
|
161
|
+
Object.assign(rows[0] || {}, { ...extraData || {} });
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
await metaFormat({ rows, table: tokenData?.table || hookData?.table || params.table }, pg);
|
|
154
165
|
|
|
155
166
|
const status = [];
|
|
156
167
|
if (loadTable?.meta?.status) {
|
|
@@ -17,7 +17,9 @@ export default async function filterAPI(req) {
|
|
|
17
17
|
filter, custom, state, search,
|
|
18
18
|
} = query;
|
|
19
19
|
|
|
20
|
-
const {
|
|
20
|
+
const { extra } = loadTable?.form ? await getTemplate('form', loadTable?.form) : {};
|
|
21
|
+
|
|
22
|
+
const { optimizedSQL = `select * from ${loadTable.table}` } = loadTable?.sql || filter || custom || state || search || extra ? await getFilterSQL({
|
|
21
23
|
pg,
|
|
22
24
|
table: params.table,
|
|
23
25
|
filter,
|
package/utils.js
CHANGED
|
@@ -32,6 +32,7 @@ import getFilterSQL from './server/plugins/table/funcs/getFilterSQL/index.js';
|
|
|
32
32
|
import getSelect from './server/plugins/table/funcs/getSelect.js';
|
|
33
33
|
import getSelectMeta from './server/plugins/table/funcs/getSelectMeta.js';
|
|
34
34
|
import gisIRColumn from './server/plugins/table/funcs/gisIRColumn.js';
|
|
35
|
+
import getData from './server/plugins/table/funcs/getData.js';
|
|
35
36
|
|
|
36
37
|
// crud
|
|
37
38
|
import dataInsert from './server/plugins/crud/funcs/dataInsert.js';
|
|
@@ -114,6 +115,7 @@ export {
|
|
|
114
115
|
metaFormat,
|
|
115
116
|
getMeta,
|
|
116
117
|
gisIRColumn,
|
|
118
|
+
getData,
|
|
117
119
|
|
|
118
120
|
// pg
|
|
119
121
|
initPG,
|