@opengis/gis 0.2.119 → 0.2.120

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 (197) hide show
  1. package/README.md +148 -148
  2. package/dist/{CardIcon-ODZgA2Pb.js → CardIcon-Dx_myDwM.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-UMttc2-F.js → EntityTablePage-DY9v1aWL.js} +32 -31
  6. package/dist/{ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-DLAIOHJh.js → ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-DgQHLSak.js} +1 -1
  7. package/dist/{HeaderActions.vue_vue_type_script_setup_true_lang-CTKkmXWD.js → HeaderActions.vue_vue_type_script_setup_true_lang-BQ32ioLC.js} +30 -30
  8. package/dist/{MapSettings-CbW0TfZl.js → MapSettings-CaGPr5uG.js} +33 -33
  9. package/dist/{RastersTablePage-CXqscX0i.js → RastersTablePage-CJyO2DoI.js} +41 -39
  10. package/dist/{cartocss-306ZE1Ha.js → cartocss-Bhat1m61.js} +296 -294
  11. package/dist/{import-utils-CHIDSKHz.js → import-utils-BxeYqcn7.js} +33 -33
  12. package/dist/index-Ba6SE4TN.js +10203 -0
  13. package/dist/index.css +1 -1
  14. package/dist/index.js +1 -1
  15. package/dist/index.umd.cjs +31 -31
  16. package/dist/{raster-Cn6l0xfZ.js → raster-Cbs1xR5Y.js} +120 -121
  17. package/dist/{register-4wblFcw2.js → register-D0AsH83v.js} +8 -8
  18. package/dist/{service-2uWyAfnR.js → service-C4nt_DBO.js} +10 -10
  19. package/dist/{vs-datatable-D_Xzh4qf.js → vs-datatable-QG5b0CCL.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
  196. package/dist/FileEdit-GikPLv1T.js +0 -37
  197. package/dist/index-CbdZv1jy.js +0 -10123
@@ -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 && 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(config.mapnik?.folder || '', 'vector', data.source_path) };
99
- }
100
- if (data.type === 'css') {
101
- return { name: path.posix.join(config.mapnik?.folder || '', 'xml', id) };
102
- }
103
- return { path: path.posix.join(config.mapnik?.folder || '', 'raster', 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 && 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(config.mapnik?.folder || '', 'vector', data.source_path) };
99
+ }
100
+ if (data.type === 'css') {
101
+ return { name: path.posix.join(config.mapnik?.folder || '', 'xml', id) };
102
+ }
103
+ return { path: path.posix.join(config.mapnik?.folder || '', 'raster', 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
+ }