@opengis/gis 0.2.120 → 0.2.122

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 (195) hide show
  1. package/README.md +148 -148
  2. package/dist/{CardIcon-Dx_myDwM.js → CardIcon-Bu_mm6QX.js} +1 -1
  3. package/dist/{EntityCreatePage-Ch8NU_Yw.js → EntityCreatePage-Da8XgBKU.js} +14 -14
  4. package/dist/{EntityEditPage-CRRcW7FL.js → EntityEditPage-QktKEAxz.js} +27 -27
  5. package/dist/{EntityTablePage-DY9v1aWL.js → EntityTablePage-BQKMFYxV.js} +26 -26
  6. package/dist/{ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-DgQHLSak.js → ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-CF1EEob3.js} +1 -1
  7. package/dist/{HeaderActions.vue_vue_type_script_setup_true_lang-BQ32ioLC.js → HeaderActions.vue_vue_type_script_setup_true_lang-B2ErIdAR.js} +30 -30
  8. package/dist/{MapSettings-CaGPr5uG.js → MapSettings-eE5rNSLf.js} +33 -33
  9. package/dist/{RastersTablePage-CJyO2DoI.js → RastersTablePage-B_vF9yl7.js} +29 -29
  10. package/dist/{cartocss-Bhat1m61.js → cartocss-DcepkzNt.js} +12 -12
  11. package/dist/{import-utils-BxeYqcn7.js → import-utils-CDIgsIUf.js} +33 -33
  12. package/dist/{index-Ba6SE4TN.js → index-CActL7Mw.js} +64 -64
  13. package/dist/index.css +1 -1
  14. package/dist/index.js +1 -1
  15. package/dist/index.umd.cjs +12 -12
  16. package/dist/{raster-Cbs1xR5Y.js → raster-BsBJMgM3.js} +101 -101
  17. package/dist/{register-D0AsH83v.js → register-BSlpAeRe.js} +8 -8
  18. package/dist/{service-C4nt_DBO.js → service-CbPCldfY.js} +11 -11
  19. package/dist/{vs-datatable-QG5b0CCL.js → vs-datatable-RkRQEa10.js} +28 -28
  20. package/module/cls.json +6 -6
  21. package/module/gis/card/gis.metadata.table/index.yml +22 -22
  22. package/module/gis/card/gis.metadata.table/main_info.hbs +20 -20
  23. package/module/gis/card/gis.metadata.table/metadata_info.hbs +27 -27
  24. package/module/gis/card/gis.metadata.table/other.hbs +25 -25
  25. package/module/gis/card/gis.rasters.table/index.yml +11 -11
  26. package/module/gis/card/gis.rasters.table/main_info.hbs +27 -27
  27. package/module/gis/card/gis.registers.table/cls.hbs +36 -36
  28. package/module/gis/card/gis.registers.table/columns.hbs +89 -89
  29. package/module/gis/card/gis.registers.table/filters.hbs +80 -80
  30. package/module/gis/card/gis.registers.table/index.yml +23 -23
  31. package/module/gis/card/gis.registers.table/main_info.hbs +35 -35
  32. package/module/gis/card/gis.registers.table/source.hbs +45 -45
  33. package/module/gis/card/gis.services.table/attributes.hbs +91 -91
  34. package/module/gis/card/gis.services.table/filters.hbs +83 -83
  35. package/module/gis/card/gis.services.table/index.yml +25 -25
  36. package/module/gis/card/gis.services.table/main_info.hbs +27 -27
  37. package/module/gis/card/gis.services.table/source.hbs +25 -25
  38. package/module/gis/cls/bool.yes_no.json +12 -12
  39. package/module/gis/cls/encoding.json +14 -14
  40. package/module/gis/cls/geom_type.json +14 -14
  41. package/module/gis/cls/gis.column_type.json +34 -34
  42. package/module/gis/cls/gis.column_view_type.json +26 -26
  43. package/module/gis/cls/gis.filter_type.json +22 -22
  44. package/module/gis/cls/language.json +10 -10
  45. package/module/gis/cls/meta.service_type.json +42 -42
  46. package/module/gis/cls/ogc.service.json +21 -21
  47. package/module/gis/cls/service_type.json +42 -42
  48. package/module/gis/cls/source_type.json +10 -10
  49. package/module/gis/cls/standarts.json +6 -6
  50. package/module/gis/cls/topic_category.json +106 -106
  51. package/module/gis/cls/update_frequency.json +29 -29
  52. package/module/gis/cls/view.json +21 -21
  53. package/module/gis/form/feedback.answer.form.json +10 -10
  54. package/module/gis/form/gis.cartocss.form.json +54 -54
  55. package/module/gis/form/gis.group_list.form.json +17 -17
  56. package/module/gis/form/gis.maps.form.json +61 -61
  57. package/module/gis/form/gis.metadata.form.json +240 -240
  58. package/module/gis/form/gis.metadata_new.form.json +195 -195
  59. package/module/gis/form/gis.ogc_service.form.json +51 -51
  60. package/module/gis/form/gis.rasters.form.json +76 -76
  61. package/module/gis/form/gis.registers.form.json +124 -124
  62. package/module/gis/form/gis.registers_column.form.json +84 -84
  63. package/module/gis/form/gis.registers_filter.form.json +65 -65
  64. package/module/gis/form/gis.services.form.json +111 -111
  65. package/module/gis/form/gis.services_attributes.form.json +75 -75
  66. package/module/gis/form/gis.services_filter.form.json +65 -65
  67. package/module/gis/menu.json +43 -43
  68. package/module/gis/select/admin.cls.name.sql +2 -2
  69. package/module/gis/select/layer_list.sql +7 -7
  70. package/module/gis/select/layer_list_text.sql +7 -7
  71. package/module/gis/select/pg.column_name.geometry.sql +12 -12
  72. package/module/gis/select/pg.columns.parent.sql +6 -6
  73. package/module/gis/select/pg.table_name.sql +17 -17
  74. package/module/gis/select/service_id.sql +1 -1
  75. package/module/gis/table/gis.cartocss.table.json +74 -74
  76. package/module/gis/table/gis.group_list.table.json +58 -58
  77. package/module/gis/table/gis.maps.table.json +110 -110
  78. package/module/gis/table/gis.metadata.table.json +70 -70
  79. package/module/gis/table/gis.metadata_new.table.json +137 -137
  80. package/module/gis/table/gis.ogc_service.table.json +100 -100
  81. package/module/gis/table/gis.rasters.table.json +101 -101
  82. package/module/gis/table/gis.registers.table.json +143 -143
  83. package/module/gis/table/gis.services.table.json +120 -120
  84. package/module/gis/table/site.gis.registers.table.json +88 -88
  85. package/module/gis/table/site.gis.services.table.json +106 -106
  86. package/module/gis/templates/ISO19136_2017_gml_template.xml +330 -330
  87. package/module/gis/tokens.yml +5 -5
  88. package/module/permissions/form/permissions.users.form.json +151 -151
  89. package/module/permissions/table/gis.permissions.table.json +97 -97
  90. package/module/test/cls/bp_build_type.json +37 -37
  91. package/module/test/cls/ts.temp_status.json +18 -18
  92. package/module/test/cls/ts.temp_structure.ts_class.json +49 -49
  93. package/module/test/cls/ts.temp_type.json +9 -9
  94. package/module/test/layer/bp.json +59 -59
  95. package/module/test/layer/bp1.yml +33 -33
  96. package/module/test/layer/individual.yml +53 -53
  97. package/module/test/layer/ts.linking_passport.yml +55 -55
  98. package/module/test/layer/ts.temp_structure.yml +50 -50
  99. package/module/test/map/addr.yml +21 -21
  100. package/module/test/map/bp_myo.json +43 -43
  101. package/module/test/map/bpf.json +43 -43
  102. package/module/test/map/main.json +37 -37
  103. package/module/test/map/mbd.json +92 -92
  104. package/module/test/map/ts.json +53 -53
  105. package/module/test/select/address_id.json +2 -2
  106. package/module/test/select/address_id.sql +7 -7
  107. package/module/test/table/data_address.street.table.json +69 -69
  108. package/module/test/table/data_bp_myo.bp.table.json +122 -122
  109. package/package.json +75 -75
  110. package/plugin.js +55 -55
  111. package/server/migrations/array_intersect.sql +13 -13
  112. package/server/migrations/cartocss.sql +27 -27
  113. package/server/migrations/group_list.sql +74 -74
  114. package/server/migrations/maps.sql +30 -30
  115. package/server/migrations/metadata.sql +415 -415
  116. package/server/migrations/ogc.sql +106 -106
  117. package/server/migrations/rasters.sql +265 -265
  118. package/server/migrations/services.sql +247 -247
  119. package/server/migrations/services_users_rel.sql +22 -22
  120. package/server/migrations/widgets.sql +20 -20
  121. package/server/plugins/crons.js +21 -21
  122. package/server/plugins/mapnik/funcs/checkRasterFile.js +109 -109
  123. package/server/plugins/mapnik/funcs/mapnik.js +146 -146
  124. package/server/plugins/mapnik/funcs/rootFolder.mjs +8 -8
  125. package/server/plugins/mapnik/map.proto +234 -234
  126. package/server/plugins/vite.js +74 -74
  127. package/server/routes/gis/cartocss/add.cartocss.js +42 -42
  128. package/server/routes/gis/cartocss/get.cartocss.js +50 -50
  129. package/server/routes/gis/checkXML.js +118 -118
  130. package/server/routes/gis/dashboard.js +29 -29
  131. package/server/routes/gis/index.mjs +87 -87
  132. package/server/routes/gis/metadata/editMetadata.js +34 -34
  133. package/server/routes/gis/metadata/metadataXML.js +13 -13
  134. package/server/routes/gis/ogc/map.info.point.js +124 -124
  135. package/server/routes/gis/registers/add.registry.js +35 -35
  136. package/server/routes/gis/registers/del.registry.js +14 -14
  137. package/server/routes/gis/registers/funcs/classifiers.js +100 -100
  138. package/server/routes/gis/registers/funcs/columns.js +4 -4
  139. package/server/routes/gis/registers/funcs/content.type.js +9 -9
  140. package/server/routes/gis/registers/funcs/get.info.js +89 -89
  141. package/server/routes/gis/registers/funcs/handleRegistryRequest.js +146 -146
  142. package/server/routes/gis/registers/gis.export.js +153 -153
  143. package/server/routes/gis/registers/gis.registry.js +64 -64
  144. package/server/routes/gis/registers/gis.registry.list.js +59 -59
  145. package/server/routes/gis/registers/insert.columns.js +107 -107
  146. package/server/routes/gis/registers/insert.filters.js +110 -110
  147. package/server/routes/gis/registers/map.registry.js +79 -79
  148. package/server/routes/gis/services/add.service.js +64 -64
  149. package/server/routes/gis/services/del.service.js +12 -12
  150. package/server/routes/gis/services/get.layer.geom.js +27 -27
  151. package/server/routes/gis/services/get.services.col.js +33 -33
  152. package/server/routes/gis/services/get.services.js +84 -84
  153. package/server/routes/gis/services/legend.auto.js +77 -77
  154. package/server/routes/map/controllers/clearTiles.js +34 -34
  155. package/server/routes/map/controllers/geojson.js +187 -187
  156. package/server/routes/map/controllers/jsonData.js +205 -205
  157. package/server/routes/map/controllers/layerList.js +102 -102
  158. package/server/routes/map/controllers/map.js +123 -123
  159. package/server/routes/map/controllers/mapCatalog.js +72 -72
  160. package/server/routes/map/controllers/mapCatalogAttribute.js +55 -55
  161. package/server/routes/map/controllers/mapFeatures.js +128 -128
  162. package/server/routes/map/controllers/mapFormat.js +222 -222
  163. package/server/routes/map/controllers/mapTiles.js +152 -152
  164. package/server/routes/map/controllers/maps.js +15 -15
  165. package/server/routes/map/controllers/marker_icon.js +43 -43
  166. package/server/routes/map/controllers/vtile.js +172 -172
  167. package/server/routes/map/index.mjs +169 -169
  168. package/server/routes/map/maps/add.map.js +43 -43
  169. package/server/routes/map/maps/del.map.js +18 -18
  170. package/server/routes/map/maps/get.map.js +112 -112
  171. package/server/routes/map/vtile1.js +179 -179
  172. package/server/routes/map/widgets/add.widget.js +38 -38
  173. package/server/routes/map/widgets/del.widget.js +22 -22
  174. package/server/routes/map/widgets/get.widget.js +40 -40
  175. package/server/routes/mapnik/controllers/clearTiles.js +106 -106
  176. package/server/routes/mapnik/controllers/createXml.js +67 -67
  177. package/server/routes/mapnik/controllers/createXmlMulti.js +76 -76
  178. package/server/routes/mapnik/controllers/fileSearch.js +38 -38
  179. package/server/routes/mapnik/controllers/fileStat.js +28 -28
  180. package/server/routes/mapnik/controllers/mapnikLogger.js +29 -29
  181. package/server/routes/mapnik/controllers/mapnikStat.js +20 -20
  182. package/server/routes/mapnik/controllers/rasterInfo.js +54 -54
  183. package/server/routes/mapnik/controllers/readDir.js +22 -22
  184. package/server/routes/mapnik/controllers/rtile.js +150 -150
  185. package/server/routes/mapnik/controllers/uploadRaster.js +157 -157
  186. package/server/routes/mapnik/functions/cartoBounds.js +28 -28
  187. package/server/routes/mapnik/functions/uploadXML.js +110 -110
  188. package/server/routes/mapnik/index.js +27 -27
  189. package/server/routes/permissions/controllers/catalog.permissions.edit.js +22 -22
  190. package/server/routes/permissions/controllers/catalog.permissions.js +7 -7
  191. package/server/routes/permissions/controllers/gis.catalog.js +80 -80
  192. package/server/routes/permissions/controllers/utils/get.permissions.js +59 -59
  193. package/server/routes/permissions/index.mjs +18 -18
  194. package/server/routes/root.mjs +3 -3
  195. package/utils.js +13 -13
@@ -1,34 +1,34 @@
1
- import path from 'node:path';
2
- import { existsSync } from 'node:fs';
3
- import {
4
- rm, stat, readdir, mkdir,
5
- } from 'node:fs/promises';
6
-
7
- import rootFolder from '../../../plugins/mapnik/funcs/rootFolder.mjs';
8
-
9
- export default async function clearTiles({
10
- params, query, user,
11
- }, reply) {
12
- const filepath = path.join(rootFolder, 'vtile', params.id);
13
-
14
- if (query.debug && user?.user_type?.includes?.('admin')) {
15
- return filepath;
16
- }
17
-
18
- const exists = existsSync(filepath);
19
- const isDirectory = exists ? await stat(filepath).then(el => el.isDirectory()) : false;
20
-
21
- const count = isDirectory ? await readdir(filepath).then(el => el.length) : 0;
22
-
23
- if (isDirectory && count === 0) {
24
- return reply.status(404).send({ error: 'Тайли не знайдено', code: 404 });
25
- }
26
-
27
- if (isDirectory && exists) {
28
- await rm(filepath, { force: true, recursive: true });
29
- await mkdir(filepath, { recursive: true });
30
- return reply.status(200).send({ message: 'Тайли очищено', code: 200 });
31
- }
32
-
33
- return reply.status(400).send({ error: 'Некорректний параметр: id', code: 400 });
34
- }
1
+ import path from 'node:path';
2
+ import { existsSync } from 'node:fs';
3
+ import {
4
+ rm, stat, readdir, mkdir,
5
+ } from 'node:fs/promises';
6
+
7
+ import rootFolder from '../../../plugins/mapnik/funcs/rootFolder.mjs';
8
+
9
+ export default async function clearTiles({
10
+ params, query, user,
11
+ }, reply) {
12
+ const filepath = path.join(rootFolder, 'vtile', params.id);
13
+
14
+ if (query.debug && user?.user_type?.includes?.('admin')) {
15
+ return filepath;
16
+ }
17
+
18
+ const exists = existsSync(filepath);
19
+ const isDirectory = exists ? await stat(filepath).then(el => el.isDirectory()) : false;
20
+
21
+ const count = isDirectory ? await readdir(filepath).then(el => el.length) : 0;
22
+
23
+ if (isDirectory && count === 0) {
24
+ return reply.status(404).send({ error: 'Тайли не знайдено', code: 404 });
25
+ }
26
+
27
+ if (isDirectory && exists) {
28
+ await rm(filepath, { force: true, recursive: true });
29
+ await mkdir(filepath, { recursive: true });
30
+ return reply.status(200).send({ message: 'Тайли очищено', code: 200 });
31
+ }
32
+
33
+ return reply.status(400).send({ error: 'Некорректний параметр: id', code: 400 });
34
+ }
@@ -1,187 +1,187 @@
1
- import path from 'node:path';
2
- import { existsSync } from 'node:fs';
3
- import { createHash } from 'node:crypto';
4
- import {
5
- readFile, stat, mkdir, writeFile,
6
- } from 'node:fs/promises';
7
-
8
- import {
9
- pgClients, getMeta,
10
- } from '@opengis/fastify-table/utils.js';
11
-
12
- import rootFolder from '../../../plugins/mapnik/funcs/rootFolder.mjs';
13
-
14
- const systemColumns = ['uid', 'cdate', 'editor_id', 'editor_date', 'created_by', 'created_at', 'updated_by', 'updated_at'];
15
-
16
- /**
17
- * Повертає GeoJSON для сервісу на основі налаштувань з gis.services
18
- *
19
- * @method GET
20
- * @param {string} params.layer - service_id
21
- * @param {string} query.sql - повернути SQL запит замість даних
22
- * @param {boolean} query.nocache - не використовувати кеш
23
- */
24
- export default async function geojson({
25
- pg = pgClients.client, params = {}, query = {}, unittest,
26
- }, reply) {
27
- const { layer } = params;
28
- const { sql, nocache, srid, type, extent } = query;
29
-
30
- if (!layer) {
31
- return reply.status(400).send('not enough params: layer (service_id)');
32
- }
33
-
34
- const serviceId = layer.replace('.json', '');
35
-
36
- if (!pg.pk?.['gis.services']) {
37
- return reply.status(404).send('services table not found');
38
- }
39
-
40
- const service = await pg.query(
41
- `select
42
- source_path,
43
- query,
44
- geometry_column,
45
- attributes,
46
- srid,
47
- is_public,
48
- is_active,
49
- geom_type
50
- from gis.services
51
- where service_id = $1 and is_active = true and is_public = true`,
52
- [serviceId],
53
- ).then(el => el.rows?.[0]);
54
-
55
- if (!service) {
56
- return reply.status(404).send('service not found or not available');
57
- }
58
-
59
- const {
60
- source_path: source,
61
- query: serviceQuery,
62
- geometry_column: gc,
63
- attributes,
64
- srid: serviceSrid,
65
- geom_type: geomType,
66
- } = service;
67
-
68
- if (!source) {
69
- return reply.status(400).send('invalid service source path');
70
- }
71
-
72
- const table = source;
73
-
74
- if (!pg.pk?.[table]?.length) {
75
- return reply.status(404).send('layer table not found');
76
- }
77
-
78
- const meta = await getMeta({ pg, table });
79
- const { columns = [] } = meta || {};
80
-
81
- const geom = gc || meta?.geom || 'geom';
82
- const pk = pg.pk?.[table] || columns[0]?.name;
83
-
84
- let attributeColumns = [];
85
- if (attributes && Array.isArray(attributes)) {
86
- attributeColumns = attributes
87
- .filter(attr => attr.name && !systemColumns.includes(attr.name))
88
- .map(attr => attr.name);
89
- }
90
-
91
- const columnsFiltered = attributeColumns.length > 0
92
- ? columns.filter(col => attributeColumns.includes(col.name) && !systemColumns.includes(col.name))
93
- : [];
94
-
95
- const props = columnsFiltered.map(el =>
96
- `"${el.name}"${['int4'].includes(el.type) ? '::text' : ''}`
97
- ).join(', ');
98
-
99
- const geometryType = (geomType || 'point').toLowerCase();
100
- const limit = geometryType === 'point' ? 40000 : 2500;
101
- const cacheHash = createHash('md5')
102
- .update([serviceId, serviceQuery, srid, type, extent].filter(el => el).join('-'))
103
- .digest('hex');
104
-
105
- const today = new Date().toISOString().split('T')[0];
106
- const filepath = path.join(
107
- rootFolder,
108
- `/geojson/${serviceId}/${today}/${cacheHash}.json`,
109
- );
110
-
111
- if (existsSync(filepath) && !nocache && !sql && !unittest) {
112
- const { birthtimeMs = Date.now() } = await stat(filepath);
113
- const ageInMs = Date.now() - birthtimeMs;
114
- if (ageInMs < 86400000) {
115
- const content = await readFile(filepath, 'utf8');
116
- return reply
117
- .status(200)
118
- .header('Content-Type', 'application/json')
119
- .header('Cache-Control', 'public, max-age=86400')
120
- .send(content);
121
- }
122
- }
123
-
124
- const targetSrid = srid || serviceSrid || 4326;
125
- const useCentroid = type === 'centroid';
126
-
127
- let geomExpression = geom;
128
-
129
- if (useCentroid) {
130
- geomExpression = `st_centroid(${geomExpression})`;
131
- }
132
-
133
- if (targetSrid !== 4326) {
134
- geomExpression = `st_transform(${geomExpression}, 4326)`;
135
- } else if (serviceSrid && serviceSrid !== 4326) {
136
- geomExpression = `st_transform(${geomExpression}, 4326)`;
137
- }
138
-
139
- const quotedTable = table.split('.').map(el => `"${el}"`).join('.');
140
- const quotedGeom = `"${geom}"`;
141
-
142
- const geomFinal = `st_force2d(${geomExpression.replace(new RegExp(`(\\b)${geom}(\\b)`, 'g'), `$1${quotedGeom}$2`)})`;
143
- const q = `
144
- SELECT 'FeatureCollection' As type,
145
- json_agg(f) As features
146
- FROM (
147
- SELECT 'Feature' As type,
148
- row_number() over() as id,
149
- st_asgeojson(${geomFinal}, 6, 0)::json as geometry,
150
- (select row_to_json(tc) from (
151
- select "${pk}" as id ${props ? ', ' + props : ''}
152
- ) tc) as properties
153
- FROM ${quotedTable}
154
- WHERE ${serviceQuery || '1=1'}
155
- AND ${quotedGeom} is not null
156
- AND st_srid(${quotedGeom}) > 0
157
- ${extent ? `AND ${quotedGeom} && 'BOX(${extent.split(',').reduce((p, el, i) => p + el + (i % 2 ? ',' : ' '), '')})'::box2d` : ''}
158
- LIMIT ${limit}
159
- ) f
160
- `;
161
-
162
- if (sql === '1') return q;
163
- if (sql === '2') return filepath.replace(/\\/g, '/');
164
-
165
- try {
166
- const result = await pg.query(q);
167
- const geojsonData = result.rows[0] || { type: 'FeatureCollection', features: [] };
168
-
169
- geojsonData.service_id = serviceId;
170
- geojsonData.count = geojsonData.features?.length || 0;
171
-
172
- const jsonString = JSON.stringify(geojsonData);
173
-
174
- if (!nocache && !unittest) {
175
- await mkdir(path.dirname(filepath), { recursive: true });
176
- await writeFile(filepath, jsonString, 'utf8');
177
- }
178
-
179
- return reply
180
- .status(200)
181
- .header('Content-Type', 'application/json')
182
- .header('Cache-Control', nocache ? 'no-cache' : 'public, max-age=86400')
183
- .send(jsonString);
184
- } catch (err) {
185
- return reply.status(500).send({ error: err.message });
186
- }
187
- }
1
+ import path from 'node:path';
2
+ import { existsSync } from 'node:fs';
3
+ import { createHash } from 'node:crypto';
4
+ import {
5
+ readFile, stat, mkdir, writeFile,
6
+ } from 'node:fs/promises';
7
+
8
+ import {
9
+ pgClients, getMeta,
10
+ } from '@opengis/fastify-table/utils.js';
11
+
12
+ import rootFolder from '../../../plugins/mapnik/funcs/rootFolder.mjs';
13
+
14
+ const systemColumns = ['uid', 'cdate', 'editor_id', 'editor_date', 'created_by', 'created_at', 'updated_by', 'updated_at'];
15
+
16
+ /**
17
+ * Повертає GeoJSON для сервісу на основі налаштувань з gis.services
18
+ *
19
+ * @method GET
20
+ * @param {string} params.layer - service_id
21
+ * @param {string} query.sql - повернути SQL запит замість даних
22
+ * @param {boolean} query.nocache - не використовувати кеш
23
+ */
24
+ export default async function geojson({
25
+ pg = pgClients.client, params = {}, query = {}, unittest,
26
+ }, reply) {
27
+ const { layer } = params;
28
+ const { sql, nocache, srid, type, extent } = query;
29
+
30
+ if (!layer) {
31
+ return reply.status(400).send('not enough params: layer (service_id)');
32
+ }
33
+
34
+ const serviceId = layer.replace('.json', '');
35
+
36
+ if (!pg.pk?.['gis.services']) {
37
+ return reply.status(404).send('services table not found');
38
+ }
39
+
40
+ const service = await pg.query(
41
+ `select
42
+ source_path,
43
+ query,
44
+ geometry_column,
45
+ attributes,
46
+ srid,
47
+ is_public,
48
+ is_active,
49
+ geom_type
50
+ from gis.services
51
+ where service_id = $1 and is_active = true and is_public = true`,
52
+ [serviceId],
53
+ ).then(el => el.rows?.[0]);
54
+
55
+ if (!service) {
56
+ return reply.status(404).send('service not found or not available');
57
+ }
58
+
59
+ const {
60
+ source_path: source,
61
+ query: serviceQuery,
62
+ geometry_column: gc,
63
+ attributes,
64
+ srid: serviceSrid,
65
+ geom_type: geomType,
66
+ } = service;
67
+
68
+ if (!source) {
69
+ return reply.status(400).send('invalid service source path');
70
+ }
71
+
72
+ const table = source;
73
+
74
+ if (!pg.pk?.[table]?.length) {
75
+ return reply.status(404).send('layer table not found');
76
+ }
77
+
78
+ const meta = await getMeta({ pg, table });
79
+ const { columns = [] } = meta || {};
80
+
81
+ const geom = gc || meta?.geom || 'geom';
82
+ const pk = pg.pk?.[table] || columns[0]?.name;
83
+
84
+ let attributeColumns = [];
85
+ if (attributes && Array.isArray(attributes)) {
86
+ attributeColumns = attributes
87
+ .filter(attr => attr.name && !systemColumns.includes(attr.name))
88
+ .map(attr => attr.name);
89
+ }
90
+
91
+ const columnsFiltered = attributeColumns.length > 0
92
+ ? columns.filter(col => attributeColumns.includes(col.name) && !systemColumns.includes(col.name))
93
+ : [];
94
+
95
+ const props = columnsFiltered.map(el =>
96
+ `"${el.name}"${['int4'].includes(el.type) ? '::text' : ''}`
97
+ ).join(', ');
98
+
99
+ const geometryType = (geomType || 'point').toLowerCase();
100
+ const limit = geometryType === 'point' ? 40000 : 2500;
101
+ const cacheHash = createHash('md5')
102
+ .update([serviceId, serviceQuery, srid, type, extent].filter(el => el).join('-'))
103
+ .digest('hex');
104
+
105
+ const today = new Date().toISOString().split('T')[0];
106
+ const filepath = path.join(
107
+ rootFolder,
108
+ `/geojson/${serviceId}/${today}/${cacheHash}.json`,
109
+ );
110
+
111
+ if (existsSync(filepath) && !nocache && !sql && !unittest) {
112
+ const { birthtimeMs = Date.now() } = await stat(filepath);
113
+ const ageInMs = Date.now() - birthtimeMs;
114
+ if (ageInMs < 86400000) {
115
+ const content = await readFile(filepath, 'utf8');
116
+ return reply
117
+ .status(200)
118
+ .header('Content-Type', 'application/json')
119
+ .header('Cache-Control', 'public, max-age=86400')
120
+ .send(content);
121
+ }
122
+ }
123
+
124
+ const targetSrid = srid || serviceSrid || 4326;
125
+ const useCentroid = type === 'centroid';
126
+
127
+ let geomExpression = geom;
128
+
129
+ if (useCentroid) {
130
+ geomExpression = `st_centroid(${geomExpression})`;
131
+ }
132
+
133
+ if (targetSrid !== 4326) {
134
+ geomExpression = `st_transform(${geomExpression}, 4326)`;
135
+ } else if (serviceSrid && serviceSrid !== 4326) {
136
+ geomExpression = `st_transform(${geomExpression}, 4326)`;
137
+ }
138
+
139
+ const quotedTable = table.split('.').map(el => `"${el}"`).join('.');
140
+ const quotedGeom = `"${geom}"`;
141
+
142
+ const geomFinal = `st_force2d(${geomExpression.replace(new RegExp(`(\\b)${geom}(\\b)`, 'g'), `$1${quotedGeom}$2`)})`;
143
+ const q = `
144
+ SELECT 'FeatureCollection' As type,
145
+ json_agg(f) As features
146
+ FROM (
147
+ SELECT 'Feature' As type,
148
+ row_number() over() as id,
149
+ st_asgeojson(${geomFinal}, 6, 0)::json as geometry,
150
+ (select row_to_json(tc) from (
151
+ select "${pk}" as id ${props ? ', ' + props : ''}
152
+ ) tc) as properties
153
+ FROM ${quotedTable}
154
+ WHERE ${serviceQuery || '1=1'}
155
+ AND ${quotedGeom} is not null
156
+ AND st_srid(${quotedGeom}) > 0
157
+ ${extent ? `AND ${quotedGeom} && 'BOX(${extent.split(',').reduce((p, el, i) => p + el + (i % 2 ? ',' : ' '), '')})'::box2d` : ''}
158
+ LIMIT ${limit}
159
+ ) f
160
+ `;
161
+
162
+ if (sql === '1') return q;
163
+ if (sql === '2') return filepath.replace(/\\/g, '/');
164
+
165
+ try {
166
+ const result = await pg.query(q);
167
+ const geojsonData = result.rows[0] || { type: 'FeatureCollection', features: [] };
168
+
169
+ geojsonData.service_id = serviceId;
170
+ geojsonData.count = geojsonData.features?.length || 0;
171
+
172
+ const jsonString = JSON.stringify(geojsonData);
173
+
174
+ if (!nocache && !unittest) {
175
+ await mkdir(path.dirname(filepath), { recursive: true });
176
+ await writeFile(filepath, jsonString, 'utf8');
177
+ }
178
+
179
+ return reply
180
+ .status(200)
181
+ .header('Content-Type', 'application/json')
182
+ .header('Cache-Control', nocache ? 'no-cache' : 'public, max-age=86400')
183
+ .send(jsonString);
184
+ } catch (err) {
185
+ return reply.status(500).send({ error: err.message });
186
+ }
187
+ }