@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 CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@opengis/fastify-table",
3
- "version": "1.2.41",
3
+ "version": "1.2.43",
4
4
  "type": "module",
5
5
  "description": "core-plugins",
6
6
  "keywords": [
@@ -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
- async function getFilterSQL({
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 ? ` t ${sqlList}` : ''} limit 0`;
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
- 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,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 ? ` t ${sqlList}` : ''} where 1=1 and ${q?.replace('q.', 't.') || '1=1'} ${order})q`;
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
- await metaFormat({ rows, table: tokenData?.table || hookData?.table || params.table });
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 { optimizedSQL = `select * from ${loadTable.table}` } = loadTable?.sql || filter || custom || state || search ? await getFilterSQL({
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,