@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.
- package/README.md +148 -148
- package/dist/{CardIcon-ODZgA2Pb.js → CardIcon-Dx_myDwM.js} +1 -1
- package/dist/{EntityCreatePage-Da8XgBKU.js → EntityCreatePage-Ch8NU_Yw.js} +14 -14
- package/dist/{EntityEditPage-QktKEAxz.js → EntityEditPage-CRRcW7FL.js} +27 -27
- package/dist/{EntityTablePage-UMttc2-F.js → EntityTablePage-DY9v1aWL.js} +32 -31
- package/dist/{ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-DLAIOHJh.js → ExtentOutlineLayer.vue_vue_type_script_setup_true_lang-DgQHLSak.js} +1 -1
- package/dist/{HeaderActions.vue_vue_type_script_setup_true_lang-CTKkmXWD.js → HeaderActions.vue_vue_type_script_setup_true_lang-BQ32ioLC.js} +30 -30
- package/dist/{MapSettings-CbW0TfZl.js → MapSettings-CaGPr5uG.js} +33 -33
- package/dist/{RastersTablePage-CXqscX0i.js → RastersTablePage-CJyO2DoI.js} +41 -39
- package/dist/{cartocss-306ZE1Ha.js → cartocss-Bhat1m61.js} +296 -294
- package/dist/{import-utils-CHIDSKHz.js → import-utils-BxeYqcn7.js} +33 -33
- package/dist/index-Ba6SE4TN.js +10203 -0
- package/dist/index.css +1 -1
- package/dist/index.js +1 -1
- package/dist/index.umd.cjs +31 -31
- package/dist/{raster-Cn6l0xfZ.js → raster-Cbs1xR5Y.js} +120 -121
- package/dist/{register-4wblFcw2.js → register-D0AsH83v.js} +8 -8
- package/dist/{service-2uWyAfnR.js → service-C4nt_DBO.js} +10 -10
- package/dist/{vs-datatable-D_Xzh4qf.js → vs-datatable-QG5b0CCL.js} +28 -28
- package/module/cls.json +6 -6
- package/module/gis/card/gis.metadata.table/index.yml +22 -22
- package/module/gis/card/gis.metadata.table/main_info.hbs +20 -20
- package/module/gis/card/gis.metadata.table/metadata_info.hbs +27 -27
- package/module/gis/card/gis.metadata.table/other.hbs +25 -25
- package/module/gis/card/gis.rasters.table/index.yml +11 -11
- package/module/gis/card/gis.rasters.table/main_info.hbs +27 -27
- package/module/gis/card/gis.registers.table/cls.hbs +36 -36
- package/module/gis/card/gis.registers.table/columns.hbs +89 -89
- package/module/gis/card/gis.registers.table/filters.hbs +80 -80
- package/module/gis/card/gis.registers.table/index.yml +23 -23
- package/module/gis/card/gis.registers.table/main_info.hbs +35 -35
- package/module/gis/card/gis.registers.table/source.hbs +45 -45
- package/module/gis/card/gis.services.table/attributes.hbs +91 -91
- package/module/gis/card/gis.services.table/filters.hbs +83 -83
- package/module/gis/card/gis.services.table/index.yml +25 -25
- package/module/gis/card/gis.services.table/main_info.hbs +27 -27
- package/module/gis/card/gis.services.table/source.hbs +25 -25
- package/module/gis/cls/bool.yes_no.json +12 -12
- package/module/gis/cls/encoding.json +14 -14
- package/module/gis/cls/geom_type.json +14 -14
- package/module/gis/cls/gis.column_type.json +34 -34
- package/module/gis/cls/gis.column_view_type.json +26 -26
- package/module/gis/cls/gis.filter_type.json +22 -22
- package/module/gis/cls/language.json +10 -10
- package/module/gis/cls/meta.service_type.json +42 -42
- package/module/gis/cls/ogc.service.json +21 -21
- package/module/gis/cls/service_type.json +42 -42
- package/module/gis/cls/source_type.json +10 -10
- package/module/gis/cls/standarts.json +6 -6
- package/module/gis/cls/topic_category.json +106 -106
- package/module/gis/cls/update_frequency.json +29 -29
- package/module/gis/cls/view.json +21 -21
- package/module/gis/form/feedback.answer.form.json +10 -10
- package/module/gis/form/gis.cartocss.form.json +54 -54
- package/module/gis/form/gis.group_list.form.json +17 -17
- package/module/gis/form/gis.maps.form.json +61 -61
- package/module/gis/form/gis.metadata.form.json +240 -240
- package/module/gis/form/gis.metadata_new.form.json +195 -195
- package/module/gis/form/gis.ogc_service.form.json +51 -51
- package/module/gis/form/gis.rasters.form.json +76 -76
- package/module/gis/form/gis.registers.form.json +124 -124
- package/module/gis/form/gis.registers_column.form.json +84 -84
- package/module/gis/form/gis.registers_filter.form.json +65 -65
- package/module/gis/form/gis.services.form.json +111 -111
- package/module/gis/form/gis.services_attributes.form.json +75 -75
- package/module/gis/form/gis.services_filter.form.json +65 -65
- package/module/gis/menu.json +43 -43
- package/module/gis/select/admin.cls.name.sql +2 -2
- package/module/gis/select/layer_list.sql +7 -7
- package/module/gis/select/layer_list_text.sql +7 -7
- package/module/gis/select/pg.column_name.geometry.sql +12 -12
- package/module/gis/select/pg.columns.parent.sql +6 -6
- package/module/gis/select/pg.table_name.sql +17 -17
- package/module/gis/select/service_id.sql +1 -1
- package/module/gis/table/gis.cartocss.table.json +74 -74
- package/module/gis/table/gis.group_list.table.json +58 -58
- package/module/gis/table/gis.maps.table.json +110 -110
- package/module/gis/table/gis.metadata.table.json +70 -70
- package/module/gis/table/gis.metadata_new.table.json +137 -137
- package/module/gis/table/gis.ogc_service.table.json +100 -100
- package/module/gis/table/gis.rasters.table.json +101 -101
- package/module/gis/table/gis.registers.table.json +143 -143
- package/module/gis/table/gis.services.table.json +120 -120
- package/module/gis/table/site.gis.registers.table.json +88 -88
- package/module/gis/table/site.gis.services.table.json +106 -106
- package/module/gis/templates/ISO19136_2017_gml_template.xml +330 -330
- package/module/gis/tokens.yml +5 -5
- package/module/permissions/form/permissions.users.form.json +151 -151
- package/module/permissions/table/gis.permissions.table.json +97 -97
- package/module/test/cls/bp_build_type.json +37 -37
- package/module/test/cls/ts.temp_status.json +18 -18
- package/module/test/cls/ts.temp_structure.ts_class.json +49 -49
- package/module/test/cls/ts.temp_type.json +9 -9
- package/module/test/layer/bp.json +59 -59
- package/module/test/layer/bp1.yml +33 -33
- package/module/test/layer/individual.yml +53 -53
- package/module/test/layer/ts.linking_passport.yml +55 -55
- package/module/test/layer/ts.temp_structure.yml +50 -50
- package/module/test/map/addr.yml +21 -21
- package/module/test/map/bp_myo.json +43 -43
- package/module/test/map/bpf.json +43 -43
- package/module/test/map/main.json +37 -37
- package/module/test/map/mbd.json +92 -92
- package/module/test/map/ts.json +53 -53
- package/module/test/select/address_id.json +2 -2
- package/module/test/select/address_id.sql +7 -7
- package/module/test/table/data_address.street.table.json +69 -69
- package/module/test/table/data_bp_myo.bp.table.json +122 -122
- package/package.json +75 -75
- package/plugin.js +55 -55
- package/server/migrations/array_intersect.sql +13 -13
- package/server/migrations/cartocss.sql +27 -27
- package/server/migrations/group_list.sql +74 -74
- package/server/migrations/maps.sql +30 -30
- package/server/migrations/metadata.sql +415 -415
- package/server/migrations/ogc.sql +106 -106
- package/server/migrations/rasters.sql +265 -265
- package/server/migrations/services.sql +247 -247
- package/server/migrations/services_users_rel.sql +22 -22
- package/server/migrations/widgets.sql +20 -20
- package/server/plugins/crons.js +21 -21
- package/server/plugins/mapnik/funcs/checkRasterFile.js +109 -109
- package/server/plugins/mapnik/funcs/mapnik.js +146 -146
- package/server/plugins/mapnik/funcs/rootFolder.mjs +8 -8
- package/server/plugins/mapnik/map.proto +234 -234
- package/server/plugins/vite.js +74 -74
- package/server/routes/gis/cartocss/add.cartocss.js +42 -42
- package/server/routes/gis/cartocss/get.cartocss.js +50 -50
- package/server/routes/gis/checkXML.js +118 -118
- package/server/routes/gis/dashboard.js +29 -29
- package/server/routes/gis/index.mjs +87 -87
- package/server/routes/gis/metadata/editMetadata.js +34 -34
- package/server/routes/gis/metadata/metadataXML.js +13 -13
- package/server/routes/gis/ogc/map.info.point.js +124 -124
- package/server/routes/gis/registers/add.registry.js +35 -35
- package/server/routes/gis/registers/del.registry.js +14 -14
- package/server/routes/gis/registers/funcs/classifiers.js +100 -100
- package/server/routes/gis/registers/funcs/columns.js +4 -4
- package/server/routes/gis/registers/funcs/content.type.js +9 -9
- package/server/routes/gis/registers/funcs/get.info.js +89 -89
- package/server/routes/gis/registers/funcs/handleRegistryRequest.js +146 -146
- package/server/routes/gis/registers/gis.export.js +153 -153
- package/server/routes/gis/registers/gis.registry.js +64 -64
- package/server/routes/gis/registers/gis.registry.list.js +59 -59
- package/server/routes/gis/registers/insert.columns.js +107 -107
- package/server/routes/gis/registers/insert.filters.js +110 -110
- package/server/routes/gis/registers/map.registry.js +79 -79
- package/server/routes/gis/services/add.service.js +64 -64
- package/server/routes/gis/services/del.service.js +12 -12
- package/server/routes/gis/services/get.layer.geom.js +27 -27
- package/server/routes/gis/services/get.services.col.js +33 -33
- package/server/routes/gis/services/get.services.js +84 -84
- package/server/routes/gis/services/legend.auto.js +77 -77
- package/server/routes/map/controllers/clearTiles.js +34 -34
- package/server/routes/map/controllers/geojson.js +187 -187
- package/server/routes/map/controllers/jsonData.js +205 -205
- package/server/routes/map/controllers/layerList.js +102 -102
- package/server/routes/map/controllers/map.js +123 -123
- package/server/routes/map/controllers/mapCatalog.js +72 -72
- package/server/routes/map/controllers/mapCatalogAttribute.js +55 -55
- package/server/routes/map/controllers/mapFeatures.js +128 -128
- package/server/routes/map/controllers/mapFormat.js +222 -222
- package/server/routes/map/controllers/mapTiles.js +152 -152
- package/server/routes/map/controllers/maps.js +15 -15
- package/server/routes/map/controllers/marker_icon.js +43 -43
- package/server/routes/map/controllers/vtile.js +172 -172
- package/server/routes/map/index.mjs +169 -169
- package/server/routes/map/maps/add.map.js +43 -43
- package/server/routes/map/maps/del.map.js +18 -18
- package/server/routes/map/maps/get.map.js +112 -112
- package/server/routes/map/vtile1.js +179 -179
- package/server/routes/map/widgets/add.widget.js +38 -38
- package/server/routes/map/widgets/del.widget.js +22 -22
- package/server/routes/map/widgets/get.widget.js +40 -40
- package/server/routes/mapnik/controllers/clearTiles.js +106 -106
- package/server/routes/mapnik/controllers/createXml.js +67 -67
- package/server/routes/mapnik/controllers/createXmlMulti.js +76 -76
- package/server/routes/mapnik/controllers/fileSearch.js +38 -38
- package/server/routes/mapnik/controllers/fileStat.js +28 -28
- package/server/routes/mapnik/controllers/mapnikLogger.js +29 -29
- package/server/routes/mapnik/controllers/mapnikStat.js +20 -20
- package/server/routes/mapnik/controllers/rasterInfo.js +54 -54
- package/server/routes/mapnik/controllers/readDir.js +22 -22
- package/server/routes/mapnik/controllers/rtile.js +150 -150
- package/server/routes/mapnik/controllers/uploadRaster.js +157 -157
- package/server/routes/mapnik/functions/cartoBounds.js +28 -28
- package/server/routes/mapnik/functions/uploadXML.js +110 -110
- package/server/routes/mapnik/index.js +27 -27
- package/server/routes/permissions/controllers/catalog.permissions.edit.js +22 -22
- package/server/routes/permissions/controllers/catalog.permissions.js +7 -7
- package/server/routes/permissions/controllers/gis.catalog.js +80 -80
- package/server/routes/permissions/controllers/utils/get.permissions.js +59 -59
- package/server/routes/permissions/index.mjs +18 -18
- package/server/routes/root.mjs +3 -3
- package/utils.js +13 -13
- package/dist/FileEdit-GikPLv1T.js +0 -37
- 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
|
+
}
|