@opengis/gis 0.2.144 → 0.2.146

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 (201) hide show
  1. package/README.md +181 -181
  2. package/dist/{CardIcon-Dnk1_6Ub.js → CardIcon-BCll_4SB.js} +1 -1
  3. package/dist/{EntityCreatePage-Da8XgBKU.js → EntityCreatePage-Ch8NU_Yw.js} +14 -14
  4. package/dist/{EntityEditPage-QktKEAxz.js → EntityEditPage-CRRcW7FL.js} +27 -27
  5. package/dist/{EntityTablePage-Dy0QCqHT.js → EntityTablePage-DXJPsOx8.js} +26 -26
  6. package/dist/{ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-2H8_Dlmj.js → ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-BJ-zsrml.js} +1 -1
  7. package/dist/{HeaderActions.vue_vue_type_script_setup_true_lang-BcabKIhe.js → HeaderActions.vue_vue_type_script_setup_true_lang-EZGjVDor.js} +5 -5
  8. package/dist/{MapSettings-CAT-ReIq.js → MapSettings-CG7u4F4i.js} +33 -33
  9. package/dist/{RastersTablePage-D0eotbTX.js → RastersTablePage-CN5hfj6F.js} +29 -29
  10. package/dist/cartocss-BNVXz6oJ.js +1065 -0
  11. package/dist/{import-utils--8RwDcC8.js → import-utils-BwcMU3Gp.js} +33 -33
  12. package/dist/{index-axggzg5T.js → index-8aAArh78.js} +63 -63
  13. package/dist/index.css +1 -1
  14. package/dist/index.js +1 -1
  15. package/dist/index.umd.cjs +35 -35
  16. package/dist/raster-BN1gB-NS.js +804 -0
  17. package/dist/{register-K8C2eD4s.js → register-DZ24Z6dC.js} +8 -8
  18. package/dist/{service-VAjz4FXX.js → service-DXLa4AiW.js} +392 -380
  19. package/dist/{vs-datatable-CwQOKMCU.js → vs-datatable-DVC6-2mA.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 +64 -64
  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 +9 -9
  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/doc_status.json +31 -31
  92. package/module/test/cls/ts.temp_status.json +18 -18
  93. package/module/test/cls/ts.temp_structure.ts_class.json +49 -49
  94. package/module/test/cls/ts.temp_type.json +9 -9
  95. package/module/test/layer/bp.json +59 -59
  96. package/module/test/layer/bp1.yml +33 -33
  97. package/module/test/layer/individual.yml +53 -53
  98. package/module/test/layer/ts.linking_passport.yml +55 -55
  99. package/module/test/layer/ts.temp_structure.yml +50 -50
  100. package/module/test/map/addr.yml +21 -21
  101. package/module/test/map/bp_myo.json +43 -43
  102. package/module/test/map/bpf.json +43 -43
  103. package/module/test/map/main.json +37 -37
  104. package/module/test/map/mbd.json +92 -92
  105. package/module/test/map/ts.json +53 -53
  106. package/module/test/select/address_id.json +2 -2
  107. package/module/test/select/address_id.sql +7 -7
  108. package/module/test/select/core.user_uid.sql +1 -1
  109. package/module/test/table/data_address.street.table.json +69 -69
  110. package/module/test/table/data_bp_myo.bp.table.json +122 -122
  111. package/package.json +75 -75
  112. package/plugin.js +55 -55
  113. package/server/migrations/array_intersect.sql +13 -13
  114. package/server/migrations/cartocss.sql +27 -27
  115. package/server/migrations/group_list.sql +74 -74
  116. package/server/migrations/maps.sql +30 -30
  117. package/server/migrations/metadata.sql +415 -415
  118. package/server/migrations/ogc.sql +106 -106
  119. package/server/migrations/rasters.sql +265 -265
  120. package/server/migrations/services.sql +247 -247
  121. package/server/migrations/services_users_rel.sql +22 -22
  122. package/server/migrations/widgets.sql +20 -20
  123. package/server/plugins/crons.js +21 -21
  124. package/server/plugins/mapnik/funcs/checkRasterFile.js +109 -109
  125. package/server/plugins/mapnik/funcs/mapnik.js +151 -151
  126. package/server/plugins/mapnik/funcs/rootFolder.mjs +8 -8
  127. package/server/plugins/mapnik/map.proto +230 -230
  128. package/server/plugins/vite.js +74 -74
  129. package/server/routes/gis/cartocss/add.cartocss.js +42 -42
  130. package/server/routes/gis/cartocss/get.cartocss.js +59 -59
  131. package/server/routes/gis/checkXML.js +117 -117
  132. package/server/routes/gis/dashboard.js +29 -29
  133. package/server/routes/gis/index.mjs +87 -87
  134. package/server/routes/gis/metadata/editMetadata.js +34 -34
  135. package/server/routes/gis/metadata/metadataXML.js +13 -13
  136. package/server/routes/gis/ogc/map.info.point.js +124 -124
  137. package/server/routes/gis/registers/add.registry.js +35 -35
  138. package/server/routes/gis/registers/del.registry.js +14 -14
  139. package/server/routes/gis/registers/funcs/classifiers.js +100 -100
  140. package/server/routes/gis/registers/funcs/columns.js +3 -3
  141. package/server/routes/gis/registers/funcs/content.type.js +9 -9
  142. package/server/routes/gis/registers/funcs/get.info.js +89 -89
  143. package/server/routes/gis/registers/funcs/handleRegistryRequest.js +146 -146
  144. package/server/routes/gis/registers/gis.export.js +153 -153
  145. package/server/routes/gis/registers/gis.registry.js +64 -64
  146. package/server/routes/gis/registers/gis.registry.list.js +59 -59
  147. package/server/routes/gis/registers/insert.columns.js +107 -107
  148. package/server/routes/gis/registers/insert.filters.js +110 -110
  149. package/server/routes/gis/registers/map.registry.js +79 -79
  150. package/server/routes/gis/services/add.service.js +64 -64
  151. package/server/routes/gis/services/del.service.js +12 -12
  152. package/server/routes/gis/services/get.layer.geom.js +27 -27
  153. package/server/routes/gis/services/get.services.col.js +33 -33
  154. package/server/routes/gis/services/get.services.js +84 -84
  155. package/server/routes/gis/services/legend.auto.js +77 -77
  156. package/server/routes/map/controllers/clearTiles.js +34 -34
  157. package/server/routes/map/controllers/geojson.js +187 -187
  158. package/server/routes/map/controllers/jsonData.js +205 -205
  159. package/server/routes/map/controllers/layerList.js +102 -102
  160. package/server/routes/map/controllers/map.js +123 -123
  161. package/server/routes/map/controllers/mapCatalog.js +72 -72
  162. package/server/routes/map/controllers/mapCatalogAttribute.js +55 -55
  163. package/server/routes/map/controllers/mapFeatures.js +128 -128
  164. package/server/routes/map/controllers/mapFormat.js +234 -234
  165. package/server/routes/map/controllers/mapTiles.js +152 -152
  166. package/server/routes/map/controllers/maps.js +15 -15
  167. package/server/routes/map/controllers/marker_icon.js +43 -43
  168. package/server/routes/map/controllers/vtile.js +172 -172
  169. package/server/routes/map/index.mjs +169 -169
  170. package/server/routes/map/maps/add.map.js +43 -43
  171. package/server/routes/map/maps/del.map.js +18 -18
  172. package/server/routes/map/maps/get.map.js +118 -118
  173. package/server/routes/map/vtile1.js +179 -179
  174. package/server/routes/map/widgets/add.widget.js +38 -38
  175. package/server/routes/map/widgets/del.widget.js +22 -22
  176. package/server/routes/map/widgets/get.widget.js +40 -40
  177. package/server/routes/mapnik/controllers/clearTiles.js +106 -106
  178. package/server/routes/mapnik/controllers/createXml.js +67 -67
  179. package/server/routes/mapnik/controllers/createXmlMulti.js +76 -76
  180. package/server/routes/mapnik/controllers/fileSearch.js +38 -38
  181. package/server/routes/mapnik/controllers/fileStat.js +28 -28
  182. package/server/routes/mapnik/controllers/mapnikLogger.js +29 -29
  183. package/server/routes/mapnik/controllers/mapnikStat.js +19 -19
  184. package/server/routes/mapnik/controllers/pretile.js +144 -144
  185. package/server/routes/mapnik/controllers/rasterInfo.js +57 -57
  186. package/server/routes/mapnik/controllers/readDir.js +22 -22
  187. package/server/routes/mapnik/controllers/rtile.js +150 -150
  188. package/server/routes/mapnik/controllers/uploadRaster.js +157 -157
  189. package/server/routes/mapnik/functions/cartoBounds.js +32 -32
  190. package/server/routes/mapnik/functions/convertBoundToTile.js +27 -27
  191. package/server/routes/mapnik/functions/uploadXML.js +113 -113
  192. package/server/routes/mapnik/index.js +40 -40
  193. package/server/routes/permissions/controllers/catalog.permissions.edit.js +22 -22
  194. package/server/routes/permissions/controllers/catalog.permissions.js +7 -7
  195. package/server/routes/permissions/controllers/gis.catalog.js +80 -80
  196. package/server/routes/permissions/controllers/utils/get.permissions.js +59 -59
  197. package/server/routes/permissions/index.mjs +18 -18
  198. package/server/routes/root.mjs +3 -3
  199. package/utils.js +13 -13
  200. package/dist/cartocss-B68llP5B.js +0 -1053
  201. package/dist/raster-BaKixPiV.js +0 -792
@@ -1,150 +1,150 @@
1
- import path from 'node:path';
2
- import { createHash } from 'node:crypto';
3
-
4
- import SphericalMercator from '@mapbox/sphericalmercator';
5
-
6
- import {
7
- config, logger, pgClients,
8
- } from '@opengis/fastify-table/utils.js';
9
-
10
- import mapnik from '../../../plugins/mapnik/funcs/mapnik.js';
11
-
12
- const { RenderTile } = mapnik();
13
-
14
- const mercator = new SphericalMercator({ size: 256 });
15
-
16
- /**
17
- * Формування растрового tile cartoCss
18
- *
19
- * @method GET
20
- * @alias rtile
21
- * @param {String} bbox - bbox
22
- * @param {Number} height - висота по координатам
23
- * @param {Number} width - ширина по координатам
24
- * @param {String} data - стилізація
25
- * @param {String} lang - мова
26
- * @param {String} z - координата z
27
- * @param {String} x - координата y
28
- * @param {String} y - координата x
29
- */
30
-
31
- export default async function gisRtile({
32
- pg = pgClients.client, params, query, user,
33
- }, reply) {
34
- if (!RenderTile) {
35
- return reply.status(400).send({ error: 'mapnik server address needed', code: 400 });
36
- }
37
-
38
- const { id, z, y } = params;
39
-
40
- const x = params.x.split('.')[0] - 0;
41
-
42
- if (!id) {
43
- return reply.status(400).send({ error: 'not enough params: id', code: 400 });
44
- }
45
-
46
- if (!x || !y || !z) {
47
- return reply.status(400).send({ error: 'not enough params: xyz', code: 400 });
48
- }
49
-
50
- const decodedPath = Buffer.from(id, 'base64url').toString('utf-8');
51
-
52
- const data = pg.pk?.['gis.rasters']
53
- ? await pg.queryCache(`select raster_id as id, source_path, is_public, 'raster' as type from gis.rasters where raster_id=$1::text or source_path=$2::text`, { args: [id, decodedPath], table: 'gis.rasters' })
54
- .then(el => el.rows?.[0] || {})
55
- : {};
56
-
57
- const carto = pg.pk?.['gis.cartocss']
58
- ? await pg.queryCache('select cartocss_id, source_path, is_public, is_static from gis.cartocss where cartocss_id=$1::text', { args: [id], table: 'gis.cartocss' }).then(el => el.rows?.[0] || {})
59
- : {};
60
-
61
- if (carto.cartocss_id) {
62
- Object.assign(data, {
63
- id: carto.cartocss_id,
64
- source_path: carto.source_path,
65
- is_public: carto.is_public,
66
- is_static: carto.is_static,
67
- type: 'css',
68
- });
69
- }
70
-
71
- if (!user?.uid && !config.local && data.is_public === false) {
72
- return reply.status(403).send({ error: 'access restricted', code: 403 });
73
- }
74
-
75
- if (!data.id && data.type !== 'css') {
76
- Object.assign(data, { source_path: decodedPath });
77
- }
78
-
79
- if (!data.source_path && data.type !== 'css') {
80
- return reply.status(404).send({ error: 'raster / cartocss not found', code: 404 });
81
- }
82
-
83
- const bbox = mercator.bbox(y, x, z, false, '900913');
84
-
85
- try {
86
- const ttl = (query.nocache ? '0' : null)
87
- || (data.type === 'css' && !data.is_static ? '1w' : null);
88
-
89
- const md5 = data.source_path
90
- ? createHash('md5').update(data.source_path).digest('hex')
91
- : undefined;
92
- const base64 = data.source_path
93
- ? Buffer.from(data.source_path).toString('base64url')
94
- : undefined;
95
-
96
- const getParams = () => {
97
- if (data.type === 'css' && data.source_path) {
98
- return { name: path.posix.join('vector', config.mapnik?.folder || '', data.source_path) };
99
- }
100
- if (data.type === 'css') {
101
- return { name: path.posix.join('xml', config.mapnik?.folder || '', id) };
102
- }
103
- return { path: path.posix.join('raster', config.mapnik?.folder || '', data.source_path) };
104
- };
105
-
106
- const { path: rpath, name } = getParams();
107
-
108
- const param = {
109
- path: rpath, // for rasters only
110
- name, // required for cartocss
111
- width: 256,
112
- bbox,
113
- ttl,
114
- debug: query.debug,
115
- };
116
-
117
- const result = await RenderTile(param);
118
-
119
- if (query.debug) {
120
- return {
121
- ...result,
122
- name,
123
- path: rpath,
124
- ttl,
125
- md5_path: md5,
126
- base64_path: base64,
127
- raster: !!data.id,
128
- carto: data.type === 'css',
129
- is_static: data.type === 'css' ? data.is_static : true,
130
- };
131
- }
132
-
133
- if (result.err) {
134
- logger.file('rtile/error', {
135
- error: result.err, x, y, z, id: data.id, type: data.type,
136
- });
137
- return reply.status(500).send({ error: config.local ? result.err : 'render error', code: 500 });
138
- }
139
-
140
- const buffer = Buffer.from(result.base64, 'base64');
141
-
142
- return reply.headers({ 'Content-Type': 'image/png', 'Cache-Control': query.nocache || query.nottl ? 'no-store, no-cache, must-revalidate' : 'public, max-age=2592000' }).send(buffer);
143
- }
144
- catch (err) {
145
- logger.file('rtile/error', {
146
- error: err.toString(), stack: err.stack, x, y, z, id: data.id, type: data.type,
147
- });
148
- return reply.status(500).send({ error: config.local ? err.toString() : 'rtile error', code: 500 });
149
- }
150
- }
1
+ import path from 'node:path';
2
+ import { createHash } from 'node:crypto';
3
+
4
+ import SphericalMercator from '@mapbox/sphericalmercator';
5
+
6
+ import {
7
+ config, logger, pgClients,
8
+ } from '@opengis/fastify-table/utils.js';
9
+
10
+ import mapnik from '../../../plugins/mapnik/funcs/mapnik.js';
11
+
12
+ const { RenderTile } = mapnik();
13
+
14
+ const mercator = new SphericalMercator({ size: 256 });
15
+
16
+ /**
17
+ * Формування растрового tile cartoCss
18
+ *
19
+ * @method GET
20
+ * @alias rtile
21
+ * @param {String} bbox - bbox
22
+ * @param {Number} height - висота по координатам
23
+ * @param {Number} width - ширина по координатам
24
+ * @param {String} data - стилізація
25
+ * @param {String} lang - мова
26
+ * @param {String} z - координата z
27
+ * @param {String} x - координата y
28
+ * @param {String} y - координата x
29
+ */
30
+
31
+ export default async function gisRtile({
32
+ pg = pgClients.client, params, query, user,
33
+ }, reply) {
34
+ if (!RenderTile) {
35
+ return reply.status(400).send({ error: 'mapnik server address needed', code: 400 });
36
+ }
37
+
38
+ const { id, z, y } = params;
39
+
40
+ const x = params.x.split('.')[0] - 0;
41
+
42
+ if (!id) {
43
+ return reply.status(400).send({ error: 'not enough params: id', code: 400 });
44
+ }
45
+
46
+ if (!x || !y || !z) {
47
+ return reply.status(400).send({ error: 'not enough params: xyz', code: 400 });
48
+ }
49
+
50
+ const decodedPath = Buffer.from(id, 'base64url').toString('utf-8');
51
+
52
+ const data = pg.pk?.['gis.rasters']
53
+ ? await pg.queryCache(`select raster_id as id, source_path, is_public, 'raster' as type from gis.rasters where raster_id=$1::text or source_path=$2::text`, { args: [id, decodedPath], table: 'gis.rasters' })
54
+ .then(el => el.rows?.[0] || {})
55
+ : {};
56
+
57
+ const carto = pg.pk?.['gis.cartocss']
58
+ ? await pg.queryCache('select cartocss_id, source_path, is_public, is_static from gis.cartocss where cartocss_id=$1::text', { args: [id], table: 'gis.cartocss' }).then(el => el.rows?.[0] || {})
59
+ : {};
60
+
61
+ if (carto.cartocss_id) {
62
+ Object.assign(data, {
63
+ id: carto.cartocss_id,
64
+ source_path: carto.source_path,
65
+ is_public: carto.is_public,
66
+ is_static: carto.is_static,
67
+ type: 'css',
68
+ });
69
+ }
70
+
71
+ if (!user?.uid && !config.local && data.is_public === false) {
72
+ return reply.status(403).send({ error: 'access restricted', code: 403 });
73
+ }
74
+
75
+ if (!data.id && data.type !== 'css') {
76
+ Object.assign(data, { source_path: decodedPath });
77
+ }
78
+
79
+ if (!data.source_path && data.type !== 'css') {
80
+ return reply.status(404).send({ error: 'raster / cartocss not found', code: 404 });
81
+ }
82
+
83
+ const bbox = mercator.bbox(y, x, z, false, '900913');
84
+
85
+ try {
86
+ const ttl = (query.nocache ? '0' : null)
87
+ || (data.type === 'css' && !data.is_static ? '1w' : null);
88
+
89
+ const md5 = data.source_path
90
+ ? createHash('md5').update(data.source_path).digest('hex')
91
+ : undefined;
92
+ const base64 = data.source_path
93
+ ? Buffer.from(data.source_path).toString('base64url')
94
+ : undefined;
95
+
96
+ const getParams = () => {
97
+ if (data.type === 'css' && data.source_path) {
98
+ return { name: path.posix.join('vector', config.mapnik?.folder || '', data.source_path) };
99
+ }
100
+ if (data.type === 'css') {
101
+ return { name: path.posix.join('xml', config.mapnik?.folder || '', id) };
102
+ }
103
+ return { path: path.posix.join('raster', config.mapnik?.folder || '', data.source_path) };
104
+ };
105
+
106
+ const { path: rpath, name } = getParams();
107
+
108
+ const param = {
109
+ path: rpath, // for rasters only
110
+ name, // required for cartocss
111
+ width: 256,
112
+ bbox,
113
+ ttl,
114
+ debug: query.debug,
115
+ };
116
+
117
+ const result = await RenderTile(param);
118
+
119
+ if (query.debug) {
120
+ return {
121
+ ...result,
122
+ name,
123
+ path: rpath,
124
+ ttl,
125
+ md5_path: md5,
126
+ base64_path: base64,
127
+ raster: !!data.id,
128
+ carto: data.type === 'css',
129
+ is_static: data.type === 'css' ? data.is_static : true,
130
+ };
131
+ }
132
+
133
+ if (result.err) {
134
+ logger.file('rtile/error', {
135
+ error: result.err, x, y, z, id: data.id, type: data.type,
136
+ });
137
+ return reply.status(500).send({ error: config.local ? result.err : 'render error', code: 500 });
138
+ }
139
+
140
+ const buffer = Buffer.from(result.base64, 'base64');
141
+
142
+ return reply.headers({ 'Content-Type': 'image/png', 'Cache-Control': query.nocache || query.nottl ? 'no-store, no-cache, must-revalidate' : 'public, max-age=2592000' }).send(buffer);
143
+ }
144
+ catch (err) {
145
+ logger.file('rtile/error', {
146
+ error: err.toString(), stack: err.stack, x, y, z, id: data.id, type: data.type,
147
+ });
148
+ return reply.status(500).send({ error: config.local ? err.toString() : 'rtile error', code: 500 });
149
+ }
150
+ }
@@ -1,157 +1,157 @@
1
- import path from 'node:path';
2
- import { ListObjectsV2Command } from '@aws-sdk/client-s3';
3
-
4
- /* eslint-disable no-await-in-loop */
5
- import {
6
- config, isFileExists, logger, pgClients, eventStream, downloadFile, s3Client,
7
- } from '@opengis/fastify-table/utils.js';
8
-
9
- import mapnik from '../../../plugins/mapnik/funcs/mapnik.js';
10
-
11
- const { UploadRaster, GetRasterStatus } = mapnik();
12
-
13
- const CHUNK_SIZE = 1024 * 1024; // 1 MB per chunk
14
-
15
- function sequence(files, data, fn) {
16
- return files.reduce(
17
- (promise, relpath) => promise.then(() => fn({
18
- ...data,
19
- relpath, // allow upload multiple to one directory for vrt
20
- })),
21
- Promise.resolve(),
22
- );
23
- }
24
-
25
- async function uploadRasterFile({
26
- id, prefix, relpath: relpathOriginal, callback = () => { },
27
- }) {
28
- const metadata = await isFileExists(relpathOriginal);
29
- const relpath = relpathOriginal.replace(config.folder, '').replace(prefix, '');
30
-
31
- let offset = 0;
32
-
33
- try {
34
- const uploadStatus = await GetRasterStatus({ path: relpath, md5: true });
35
-
36
- if (uploadStatus.finished) {
37
- callback(`Already uploaded: ${relpath}`);
38
- return { message: 'already uploaded', status: 200 };
39
- }
40
-
41
- if (!metadata || !metadata.ContentLength) {
42
- callback(`File not found at s3: ${relpath}`);
43
- return { error: `File not found at s3: ${relpath}`, code: 404 };
44
- }
45
-
46
- offset = uploadStatus.exists ? +uploadStatus.size : 0;
47
-
48
- if (offset === metadata.ContentLength) {
49
- callback(`Already uploaded: ${relpath}`);
50
- return { message: 'already uploaded', status: 200 };
51
- }
52
-
53
- while (offset < metadata.ContentLength) {
54
- const end = Math.min(offset + CHUNK_SIZE - 1, metadata.ContentLength - 1);
55
- const Range = `bytes=${offset}-${end}`;
56
-
57
- callback(`Uploading chunk ${(end / 1024 / 1024).toFixed(0)}/${(metadata.ContentLength / 1024 / 1024).toFixed(0)} MB...`);
58
-
59
- const chunk = await downloadFile(relpathOriginal, { Range, fallback: false });
60
- // const { Body: chunk } = await s3Client.send(new GetObjectCommand({ Bucket, Key, Range }));
61
- const bufferPart = chunk ? await chunk.transformToByteArray() : [];
62
-
63
- if (!bufferPart.length) {
64
- callback(`Server error: file not found / zero bytes: ${relpath}`);
65
- break;
66
- }
67
-
68
- const result = await UploadRaster({
69
- path: relpath,
70
- filesize: metadata.ContentLength,
71
- data: bufferPart,
72
- chunked: true,
73
- });
74
-
75
- if (result.finished) {
76
- callback(`Upload completed: ${relpath}`);
77
- break;
78
- }
79
-
80
- if (!result.uploaded) {
81
- callback(`Chunked upload error: ${relpath}`);
82
- break;
83
- }
84
-
85
- offset = +result.uploaded;
86
- }
87
-
88
- return null;
89
- }
90
- catch (err) {
91
- logger.file('mapnik/upload/error', { error: err.toString(), stack: err.stack });
92
- callback(config.local ? err.toString() : `Upload error: ${relpath}`);
93
- return null;
94
- }
95
- }
96
-
97
- export default async function uploadRaster({
98
- pg = pgClients.client, params,
99
- }, reply) {
100
- if (!UploadRaster || !GetRasterStatus) {
101
- return reply.status(400).send({ error: 'mapnik server address needed', code: 400 });
102
- }
103
-
104
- const { id } = params;
105
-
106
- if (!id) {
107
- return reply.status(400).send({ error: 'not enough params: id', code: 400 });
108
- }
109
-
110
- const data = pg.pk?.['gis.rasters']
111
- ? await pg.query('select raster_id as id, source_path from gis.rasters where raster_id=$1::text', [id])
112
- .then(el => el.rows?.[0])
113
- : null;
114
-
115
- if (!data?.id) {
116
- return reply.status(404).send({ error: 'raster not found', code: 404 });
117
- }
118
-
119
- if (!data.source_path) {
120
- return reply.status(400).send({ error: 'raster source_path not set', code: 400 });
121
- }
122
-
123
- const prefix = '/map/raster';
124
- const relpath = `${prefix}/${data.source_path}`;
125
-
126
- if (!path.extname(relpath)) {
127
- const metadata = await s3Client.send(new ListObjectsV2Command({
128
- Bucket: config.s3?.containerName || 'work',
129
- Prefix: path.join(config.folder, relpath).replace(/\\/g, '/'),
130
- }));
131
- if (!metadata.Contents?.length) {
132
- return reply.status(404).send({
133
- error: 'invalid: raster source_path: not found / empty directory',
134
- code: 404,
135
- });
136
- }
137
-
138
- // list rasters, skip preview subdir content, skip folders from s3 directory name of which only starts with same characters, but does not exactly match
139
- const files = metadata.Contents.map(file => file.Key).filter((file) => path.basename(path.dirname(file)) === path.basename(relpath) && !path.basename(path.dirname(file)).startsWith('preview') && path.extname(file).toLowerCase() === '.tif');
140
-
141
- if (!files.length) {
142
- return reply.status(400).send({
143
- error: 'invalid: raster source_path: no raster files at directory',
144
- code: 400,
145
- });
146
- }
147
- const callback = eventStream(reply);
148
- await sequence(files, { id, prefix, callback }, uploadRasterFile);
149
- return callback('finish', 1);
150
- }
151
-
152
- const callback = eventStream(reply);
153
- await uploadRasterFile({
154
- id, prefix, callback, relpath,
155
- });
156
- return callback('finish', 1);
157
- }
1
+ import path from 'node:path';
2
+ import { ListObjectsV2Command } from '@aws-sdk/client-s3';
3
+
4
+ /* eslint-disable no-await-in-loop */
5
+ import {
6
+ config, isFileExists, logger, pgClients, eventStream, downloadFile, s3Client,
7
+ } from '@opengis/fastify-table/utils.js';
8
+
9
+ import mapnik from '../../../plugins/mapnik/funcs/mapnik.js';
10
+
11
+ const { UploadRaster, GetRasterStatus } = mapnik();
12
+
13
+ const CHUNK_SIZE = 1024 * 1024; // 1 MB per chunk
14
+
15
+ function sequence(files, data, fn) {
16
+ return files.reduce(
17
+ (promise, relpath) => promise.then(() => fn({
18
+ ...data,
19
+ relpath, // allow upload multiple to one directory for vrt
20
+ })),
21
+ Promise.resolve(),
22
+ );
23
+ }
24
+
25
+ async function uploadRasterFile({
26
+ id, prefix, relpath: relpathOriginal, callback = () => { },
27
+ }) {
28
+ const metadata = await isFileExists(relpathOriginal);
29
+ const relpath = relpathOriginal.replace(config.folder, '').replace(prefix, '');
30
+
31
+ let offset = 0;
32
+
33
+ try {
34
+ const uploadStatus = await GetRasterStatus({ path: relpath, md5: true });
35
+
36
+ if (uploadStatus.finished) {
37
+ callback(`Already uploaded: ${relpath}`);
38
+ return { message: 'already uploaded', status: 200 };
39
+ }
40
+
41
+ if (!metadata || !metadata.ContentLength) {
42
+ callback(`File not found at s3: ${relpath}`);
43
+ return { error: `File not found at s3: ${relpath}`, code: 404 };
44
+ }
45
+
46
+ offset = uploadStatus.exists ? +uploadStatus.size : 0;
47
+
48
+ if (offset === metadata.ContentLength) {
49
+ callback(`Already uploaded: ${relpath}`);
50
+ return { message: 'already uploaded', status: 200 };
51
+ }
52
+
53
+ while (offset < metadata.ContentLength) {
54
+ const end = Math.min(offset + CHUNK_SIZE - 1, metadata.ContentLength - 1);
55
+ const Range = `bytes=${offset}-${end}`;
56
+
57
+ callback(`Uploading chunk ${(end / 1024 / 1024).toFixed(0)}/${(metadata.ContentLength / 1024 / 1024).toFixed(0)} MB...`);
58
+
59
+ const chunk = await downloadFile(relpathOriginal, { Range, fallback: false });
60
+ // const { Body: chunk } = await s3Client.send(new GetObjectCommand({ Bucket, Key, Range }));
61
+ const bufferPart = chunk ? await chunk.transformToByteArray() : [];
62
+
63
+ if (!bufferPart.length) {
64
+ callback(`Server error: file not found / zero bytes: ${relpath}`);
65
+ break;
66
+ }
67
+
68
+ const result = await UploadRaster({
69
+ path: relpath,
70
+ filesize: metadata.ContentLength,
71
+ data: bufferPart,
72
+ chunked: true,
73
+ });
74
+
75
+ if (result.finished) {
76
+ callback(`Upload completed: ${relpath}`);
77
+ break;
78
+ }
79
+
80
+ if (!result.uploaded) {
81
+ callback(`Chunked upload error: ${relpath}`);
82
+ break;
83
+ }
84
+
85
+ offset = +result.uploaded;
86
+ }
87
+
88
+ return null;
89
+ }
90
+ catch (err) {
91
+ logger.file('mapnik/upload/error', { error: err.toString(), stack: err.stack });
92
+ callback(config.local ? err.toString() : `Upload error: ${relpath}`);
93
+ return null;
94
+ }
95
+ }
96
+
97
+ export default async function uploadRaster({
98
+ pg = pgClients.client, params,
99
+ }, reply) {
100
+ if (!UploadRaster || !GetRasterStatus) {
101
+ return reply.status(400).send({ error: 'mapnik server address needed', code: 400 });
102
+ }
103
+
104
+ const { id } = params;
105
+
106
+ if (!id) {
107
+ return reply.status(400).send({ error: 'not enough params: id', code: 400 });
108
+ }
109
+
110
+ const data = pg.pk?.['gis.rasters']
111
+ ? await pg.query('select raster_id as id, source_path from gis.rasters where raster_id=$1::text', [id])
112
+ .then(el => el.rows?.[0])
113
+ : null;
114
+
115
+ if (!data?.id) {
116
+ return reply.status(404).send({ error: 'raster not found', code: 404 });
117
+ }
118
+
119
+ if (!data.source_path) {
120
+ return reply.status(400).send({ error: 'raster source_path not set', code: 400 });
121
+ }
122
+
123
+ const prefix = '/map/raster';
124
+ const relpath = `${prefix}/${data.source_path}`;
125
+
126
+ if (!path.extname(relpath)) {
127
+ const metadata = await s3Client.send(new ListObjectsV2Command({
128
+ Bucket: config.s3?.containerName || 'work',
129
+ Prefix: path.join(config.folder, relpath).replace(/\\/g, '/'),
130
+ }));
131
+ if (!metadata.Contents?.length) {
132
+ return reply.status(404).send({
133
+ error: 'invalid: raster source_path: not found / empty directory',
134
+ code: 404,
135
+ });
136
+ }
137
+
138
+ // list rasters, skip preview subdir content, skip folders from s3 directory name of which only starts with same characters, but does not exactly match
139
+ const files = metadata.Contents.map(file => file.Key).filter((file) => path.basename(path.dirname(file)) === path.basename(relpath) && !path.basename(path.dirname(file)).startsWith('preview') && path.extname(file).toLowerCase() === '.tif');
140
+
141
+ if (!files.length) {
142
+ return reply.status(400).send({
143
+ error: 'invalid: raster source_path: no raster files at directory',
144
+ code: 400,
145
+ });
146
+ }
147
+ const callback = eventStream(reply);
148
+ await sequence(files, { id, prefix, callback }, uploadRasterFile);
149
+ return callback('finish', 1);
150
+ }
151
+
152
+ const callback = eventStream(reply);
153
+ await uploadRasterFile({
154
+ id, prefix, callback, relpath,
155
+ });
156
+ return callback('finish', 1);
157
+ }