From 32421ea66db35040f1026be734a7ec9f905ecab7 Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Thu, 13 Jul 2023 17:14:53 +0200 Subject: [PATCH 001/214] v4.0.0-alpha.13 --- CHANGELOG.md | 7 +++++++ lerna.json | 2 +- modules/3d-tiles/package.json | 12 ++++++------ modules/arrow/package.json | 6 +++--- modules/bson/package.json | 8 ++++---- modules/compression/package.json | 6 +++--- modules/core/package.json | 6 +++--- modules/crypto/package.json | 6 +++--- modules/csv/package.json | 6 +++--- modules/draco/package.json | 10 +++++----- modules/excel/package.json | 6 +++--- modules/flatgeobuf/package.json | 6 +++--- modules/geopackage/package.json | 8 ++++---- modules/geotiff/package.json | 2 +- modules/gis/package.json | 6 +++--- modules/gltf/package.json | 10 +++++----- modules/i3s/package.json | 16 ++++++++-------- modules/images/package.json | 4 ++-- modules/json/package.json | 8 ++++---- modules/kml/package.json | 8 ++++---- modules/las/package.json | 6 +++--- modules/loader-utils/package.json | 4 ++-- modules/math/package.json | 6 +++--- modules/mvt/package.json | 8 ++++---- modules/netcdf/package.json | 4 ++-- modules/obj/package.json | 6 +++--- modules/parquet/package.json | 10 +++++----- modules/pcd/package.json | 6 +++--- modules/ply/package.json | 6 +++--- modules/polyfills/package.json | 2 +- modules/potree/package.json | 4 ++-- modules/schema/package.json | 2 +- modules/shapefile/package.json | 8 ++++---- modules/terrain/package.json | 8 ++++---- modules/textures/package.json | 12 ++++++------ modules/tile-converter/package.json | 26 +++++++++++++------------- modules/tiles/package.json | 6 +++--- modules/video/package.json | 6 +++--- modules/wkt/package.json | 6 +++--- modules/wms/package.json | 10 +++++----- modules/worker-utils/package.json | 2 +- modules/xml/package.json | 6 +++--- modules/zarr/package.json | 2 +- modules/zip/package.json | 2 +- yarn.lock | 6 +++--- 45 files changed, 157 insertions(+), 150 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6fd33c8370..ad9b5bdf4c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,13 @@ > The early pre-release track was abandoned due to build incompatibility problems. release info (#2491)) +### v4.0.0-alpha.13 + +- fix(gltf): 3D tiles extension types & docs (#2542) +- fix(tile-converter): failing test (#2540) +- chore: bump fast-xml-parser (#2538) +- fix(3d-tiles): implicit tiling v1.1 (#2539) + ### v4.0.0-alpha.12 - fix(tile-converter): proceed failed tiles (#2536) diff --git a/lerna.json b/lerna.json index 7cdd477322..f54f8040db 100644 --- a/lerna.json +++ b/lerna.json @@ -1,6 +1,6 @@ { "lerna": "2.9.1", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "command": { "publish": {}, "bootstrap": {} diff --git a/modules/3d-tiles/package.json b/modules/3d-tiles/package.json index 3e5b562086..a5890468f8 100644 --- a/modules/3d-tiles/package.json +++ b/modules/3d-tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/3d-tiles", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "3D Tiles, an open standard for streaming massive heterogeneous 3D geospatial datasets.", "license": "MIT", "publishConfig": { @@ -34,11 +34,11 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.12", - "@loaders.gl/gltf": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/math": "4.0.0-alpha.12", - "@loaders.gl/tiles": "4.0.0-alpha.12", + "@loaders.gl/draco": "4.0.0-alpha.13", + "@loaders.gl/gltf": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/math": "4.0.0-alpha.13", + "@loaders.gl/tiles": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1", "@math.gl/geospatial": "^3.5.1", "@probe.gl/log": "^4.0.4", diff --git a/modules/arrow/package.json b/modules/arrow/package.json index f7fed130cf..e7cbd22288 100644 --- a/modules/arrow/package.json +++ b/modules/arrow/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/arrow", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Simple columnar table loader for the Apache Arrow format", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ "build-worker2": "esbuild src/workers/arrow-worker.ts --bundle --outfile=dist/arrow-worker.js --platform=browser --external:{stream}" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/bson/package.json b/modules/bson/package.json index 6927b38a23..133ab80b16 100644 --- a/modules/bson/package.json +++ b/modules/bson/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/bson", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -34,9 +34,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@types/bson": "4.2.0", "bson": "4.2.0" }, diff --git a/modules/compression/package.json b/modules/compression/package.json index 6a1257aef7..25d5b71326 100644 --- a/modules/compression/package.json +++ b/modules/compression/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/compression", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Decompression and compression plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "@types/brotli": "^1.3.0", "@types/pako": "^1.0.1", "fflate": "0.7.4", diff --git a/modules/core/package.json b/modules/core/package.json index ff14a1da3a..5b8323bf29 100644 --- a/modules/core/package.json +++ b/modules/core/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/core", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "The core API for working with loaders.gl loaders and writers", "license": "MIT", "publishConfig": { @@ -43,8 +43,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "@probe.gl/log": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/crypto/package.json b/modules/crypto/package.json index 1411a84359..e156ad1600 100644 --- a/modules/crypto/package.json +++ b/modules/crypto/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/crypto", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Cryptographic/hashing plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "@types/crypto-js": "^4.0.2" }, "devDependencies": { diff --git a/modules/csv/package.json b/modules/csv/package.json index 12f6c8f87f..5d77424937 100644 --- a/modules/csv/package.json +++ b/modules/csv/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/csv", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for CSV and DSV table formats", "license": "MIT", "publishConfig": { @@ -30,8 +30,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "devDependencies": { "d3-dsv": "^1.2.0" diff --git a/modules/draco/package.json b/modules/draco/package.json index 784c786f63..50402e47d7 100644 --- a/modules/draco/package.json +++ b/modules/draco/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/draco", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader and writer for Draco compressed meshes and point clouds", "license": "MIT", "publishConfig": { @@ -42,13 +42,13 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "draco3d": "1.5.5" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.12" + "@loaders.gl/polyfills": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/excel/package.json b/modules/excel/package.json index 8173ad4f91..feafbf4a15 100644 --- a/modules/excel/package.json +++ b/modules/excel/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/excel", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for Excel files", "license": "MIT", "publishConfig": { @@ -35,8 +35,8 @@ "build-worker": "esbuild src/workers/excel-worker.ts --bundle --outfile=dist/excel-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "xlsx": "^0.17.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/flatgeobuf/package.json b/modules/flatgeobuf/package.json index a2cd785747..fbc1fadabb 100644 --- a/modules/flatgeobuf/package.json +++ b/modules/flatgeobuf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/flatgeobuf", "description": "Loader for FlatGeobuf", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -32,8 +32,8 @@ "build-worker": "esbuild src/workers/flatgeobuf-worker.ts --bundle --outfile=dist/flatgeobuf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", "@math.gl/proj4": "^3.3.1", "flatgeobuf": "3.6.5" }, diff --git a/modules/geopackage/package.json b/modules/geopackage/package.json index 68d0f4adeb..e1fe18682d 100644 --- a/modules/geopackage/package.json +++ b/modules/geopackage/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/geopackage", "description": "GeoPackage data loaders", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -25,9 +25,9 @@ ], "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", - "@loaders.gl/wkt": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/wkt": "4.0.0-alpha.13", "@math.gl/proj4": "^3.5.1", "@types/sql.js": "^1.4.2", "sql.js": "1.5.0" diff --git a/modules/geotiff/package.json b/modules/geotiff/package.json index 16bd98d962..8f6d952dbe 100644 --- a/modules/geotiff/package.json +++ b/modules/geotiff/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/geotiff", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for tiff and geotiff", "license": "MIT", "publishConfig": { diff --git a/modules/gis/package.json b/modules/gis/package.json index 160f91ef9d..c43684895f 100644 --- a/modules/gis/package.json +++ b/modules/gis/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/gis", "description": "Helpers for GIS category data", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -24,8 +24,8 @@ "README.md" ], "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@mapbox/vector-tile": "^1.3.1", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" diff --git a/modules/gltf/package.json b/modules/gltf/package.json index 108b61924a..fb45a5b3f2 100644 --- a/modules/gltf/package.json +++ b/modules/gltf/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/gltf", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the glTF format", "license": "MIT", "publishConfig": { @@ -34,10 +34,10 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.12", - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/textures": "4.0.0-alpha.12", + "@loaders.gl/draco": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/textures": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/i3s/package.json b/modules/i3s/package.json index 3e4881a202..e5bbcfefd4 100644 --- a/modules/i3s/package.json +++ b/modules/i3s/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/i3s", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "i3s .", "license": "MIT", "publishConfig": { @@ -33,13 +33,13 @@ "build-worker-node": "esbuild src/workers/i3s-content-worker-node.ts --outfile=dist/i3s-content-worker-node.js --platform=node --target=node16 --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/compression": "4.0.0-alpha.12", - "@loaders.gl/draco": "4.0.0-alpha.12", - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", - "@loaders.gl/textures": "4.0.0-alpha.12", - "@loaders.gl/tiles": "4.0.0-alpha.12", + "@loaders.gl/compression": "4.0.0-alpha.13", + "@loaders.gl/draco": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/textures": "4.0.0-alpha.13", + "@loaders.gl/tiles": "4.0.0-alpha.13", "@luma.gl/constants": "^8.5.4", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", diff --git a/modules/images/package.json b/modules/images/package.json index 6ff0730d7b..f1f300eef4 100644 --- a/modules/images/package.json +++ b/modules/images/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/images", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders and writers for images (PNG, JPG, ...)", "license": "MIT", "publishConfig": { @@ -32,7 +32,7 @@ "build-bundle": "esbuild src/bundle.ts --outfile=dist/dist.min.js --bundle --minify --sourcemap" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5", "devDependencies": { diff --git a/modules/json/package.json b/modules/json/package.json index e2fa209bcd..4a62eeb407 100644 --- a/modules/json/package.json +++ b/modules/json/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/json", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ "build-worker": "esbuild src/workers/geojson-worker.ts --bundle --outfile=dist/geojson-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/kml/package.json b/modules/kml/package.json index 49c463a1df..107aa043cc 100644 --- a/modules/kml/package.json +++ b/modules/kml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/kml", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the KML format", "license": "MIT", "publishConfig": { @@ -32,9 +32,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@tmcw/togeojson": "^4.5.0", "@xmldom/xmldom": "^0.7.5" }, diff --git a/modules/las/package.json b/modules/las/package.json index fe1b085d8e..3f3735d222 100644 --- a/modules/las/package.json +++ b/modules/las/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/las", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the LAS and LAZ formats", "license": "MIT", "publishConfig": { @@ -41,8 +41,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/loader-utils/package.json b/modules/loader-utils/package.json index 5c14920f45..933e73fc7b 100644 --- a/modules/loader-utils/package.json +++ b/modules/loader-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/loader-utils", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for 3D graphics formats", "license": "MIT", "publishConfig": { @@ -38,7 +38,7 @@ "scripts": {}, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "@probe.gl/stats": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/math/package.json b/modules/math/package.json index 2fb1529676..f07c355868 100644 --- a/modules/math/package.json +++ b/modules/math/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/math", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Experimental math classes for loaders.gl", "license": "MIT", "publishConfig": { @@ -31,8 +31,8 @@ "pre-build": "echo \"Nothing to build in @loaders.gl/math\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/mvt/package.json b/modules/mvt/package.json index ca7d7b473a..b5cc700408 100644 --- a/modules/mvt/package.json +++ b/modules/mvt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/mvt", "description": "Loader for Mapbox Vector Tiles", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -32,9 +32,9 @@ "build-worker": "esbuild src/workers/mvt-worker.ts --bundle --outfile=dist/mvt-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" }, diff --git a/modules/netcdf/package.json b/modules/netcdf/package.json index cb14af50cf..bf71f13e74 100644 --- a/modules/netcdf/package.json +++ b/modules/netcdf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/netcdf", "description": "Loader for NetCDF", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -30,7 +30,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/obj/package.json b/modules/obj/package.json index ad9fba39c7..a17ce58469 100644 --- a/modules/obj/package.json +++ b/modules/obj/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/obj", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the OBJ format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/parquet/package.json b/modules/parquet/package.json index 165a36013f..0c473775b3 100644 --- a/modules/parquet/package.json +++ b/modules/parquet/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/parquet", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for Apache Parquet files", "license": "MIT", "publishConfig": { @@ -41,10 +41,10 @@ "./src/lib/wasm/load-wasm/load-wasm-node.ts": "./src/lib/wasm/load-wasm/load-wasm-browser.ts" }, "dependencies": { - "@loaders.gl/bson": "4.0.0-alpha.12", - "@loaders.gl/compression": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/bson": "4.0.0-alpha.13", + "@loaders.gl/compression": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "async-mutex": "^0.2.2", "brotli": "^1.3.2", "int53": "^0.2.4", diff --git a/modules/pcd/package.json b/modules/pcd/package.json index 3a540f4c36..4a13c82c9e 100644 --- a/modules/pcd/package.json +++ b/modules/pcd/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/pcd", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the PCD format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-worker": "esbuild src/workers/pcd-worker.ts --bundle --outfile=dist/pcd-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/ply/package.json b/modules/ply/package.json index 34d2440416..0c722292f5 100644 --- a/modules/ply/package.json +++ b/modules/ply/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/ply", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for the PLY format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/polyfills/package.json b/modules/polyfills/package.json index 0933772d29..adb3617d52 100644 --- a/modules/polyfills/package.json +++ b/modules/polyfills/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/polyfills", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Polyfills for TextEncoder/TextDecoder", "license": "MIT", "publishConfig": { diff --git a/modules/potree/package.json b/modules/potree/package.json index 611d041620..07ee05f1c9 100644 --- a/modules/potree/package.json +++ b/modules/potree/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/potree", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "potree loaders for large point clouds.", "license": "MIT", "publishConfig": { @@ -35,7 +35,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/math": "4.0.0-alpha.12", + "@loaders.gl/math": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/schema/package.json b/modules/schema/package.json index 949fed34ee..d881cf1b61 100644 --- a/modules/schema/package.json +++ b/modules/schema/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/schema", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Table format APIs for JSON, CSV, etc...", "license": "MIT", "publishConfig": { diff --git a/modules/shapefile/package.json b/modules/shapefile/package.json index 9c8f58278e..5bb2c6a081 100644 --- a/modules/shapefile/package.json +++ b/modules/shapefile/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/shapefile", "description": "Loader for the Shapefile Format", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -37,9 +37,9 @@ "build-worker-dbf": "esbuild src/workers/dbf-worker.ts --bundle --outfile=dist/dbf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/gis": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@math.gl/proj4": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/terrain/package.json b/modules/terrain/package.json index db4fcf2192..05f4afc1b8 100644 --- a/modules/terrain/package.json +++ b/modules/terrain/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/terrain", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loader for terrain raster formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "@mapbox/martini": "^0.2.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/textures/package.json b/modules/textures/package.json index 076f53e7c8..8c43b754a2 100644 --- a/modules/textures/package.json +++ b/modules/textures/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/textures", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for compressed and super compressed (basis) textures ", "license": "MIT", "publishConfig": { @@ -45,15 +45,15 @@ "build-crunch-worker": "esbuild src/workers/crunch-worker.ts --outfile=dist/crunch-worker.js --target=esnext --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "ktx-parse": "^0.0.4", "texture-compressor": "^1.0.2" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.12" + "@loaders.gl/polyfills": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index 64dfe7df48..32b9470e08 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tile-converter", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Converter", "license": "MIT", "publishConfig": { @@ -46,18 +46,18 @@ "build-3d-tiles-attributes-worker": "esbuild src/workers/3d-tiles-attributes-worker.ts --outfile=dist/3d-tiles-attributes-worker.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/3d-tiles": "4.0.0-alpha.12", - "@loaders.gl/crypto": "4.0.0-alpha.12", - "@loaders.gl/draco": "4.0.0-alpha.12", - "@loaders.gl/gltf": "4.0.0-alpha.12", - "@loaders.gl/i3s": "4.0.0-alpha.12", - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/polyfills": "4.0.0-alpha.12", - "@loaders.gl/textures": "4.0.0-alpha.12", - "@loaders.gl/tiles": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", - "@loaders.gl/zip": "4.0.0-alpha.12", + "@loaders.gl/3d-tiles": "4.0.0-alpha.13", + "@loaders.gl/crypto": "4.0.0-alpha.13", + "@loaders.gl/draco": "4.0.0-alpha.13", + "@loaders.gl/gltf": "4.0.0-alpha.13", + "@loaders.gl/i3s": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/polyfills": "4.0.0-alpha.13", + "@loaders.gl/textures": "4.0.0-alpha.13", + "@loaders.gl/tiles": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/zip": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geoid": "^3.5.1", diff --git a/modules/tiles/package.json b/modules/tiles/package.json index a7154b1121..8e863978c4 100644 --- a/modules/tiles/package.json +++ b/modules/tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tiles", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Common components for different tiles loaders.", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/math": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/math": "4.0.0-alpha.13", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geospatial": "^3.5.1", diff --git a/modules/video/package.json b/modules/video/package.json index 50cc44e6e5..c9da64377f 100644 --- a/modules/video/package.json +++ b/modules/video/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/video", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders and writers for video (MP4, WEBM, ...)", "license": "MIT", "publishConfig": { @@ -32,8 +32,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/worker-utils": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.13", "gifshot": "^0.4.5" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/wkt/package.json b/modules/wkt/package.json index b6587fe356..6d25dafa0b 100644 --- a/modules/wkt/package.json +++ b/modules/wkt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/wkt", "description": "Loader and Writer for the WKT (Well Known Text) Format", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "license": "MIT", "publishConfig": { "access": "public" @@ -35,8 +35,8 @@ "fuzzer": "^0.2.1" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12" + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/wms/package.json b/modules/wms/package.json index 961a781150..b7b3ee8fad 100644 --- a/modules/wms/package.json +++ b/modules/wms/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/wms", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for the WMS (Web Map Service) standard", "license": "MIT", "publishConfig": { @@ -39,10 +39,10 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.12", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", - "@loaders.gl/xml": "4.0.0-alpha.12", + "@loaders.gl/images": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/xml": "4.0.0-alpha.13", "@turf/rewind": "^5.1.5", "deep-strict-equal": "^0.2.0", "lerc": "^4.0.1" diff --git a/modules/worker-utils/package.json b/modules/worker-utils/package.json index ba31de2cd5..5e0f5362ff 100644 --- a/modules/worker-utils/package.json +++ b/modules/worker-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/worker-utils", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Utilities for running tasks on worker threads", "license": "MIT", "publishConfig": { diff --git a/modules/xml/package.json b/modules/xml/package.json index 3a0dc60de6..51ef59b291 100644 --- a/modules/xml/package.json +++ b/modules/xml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/xml", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for the XML (eXtensible Markup Language) format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.12", - "@loaders.gl/schema": "4.0.0-alpha.12", + "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/schema": "4.0.0-alpha.13", "fast-xml-parser": "^4.2.5" }, "devDependencies": { diff --git a/modules/zarr/package.json b/modules/zarr/package.json index 1df91ad1b0..57c1f8041e 100644 --- a/modules/zarr/package.json +++ b/modules/zarr/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zarr", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Framework-independent loaders for Zarr", "license": "MIT", "publishConfig": { diff --git a/modules/zip/package.json b/modules/zip/package.json index 4d62315222..86a799d18b 100644 --- a/modules/zip/package.json +++ b/modules/zip/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zip", - "version": "4.0.0-alpha.12", + "version": "4.0.0-alpha.13", "description": "Zip Archive Loader", "license": "MIT", "publishConfig": { diff --git a/yarn.lock b/yarn.lock index 5981e0be86..1c741c82b5 100644 --- a/yarn.lock +++ b/yarn.lock @@ -4065,9 +4065,9 @@ camelcase@^5.0.0, camelcase@^5.3.1: integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== caniuse-lite@^1.0.30001449: - version "1.0.30001513" - resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001513.tgz" - integrity sha512-pnjGJo7SOOjAGytZZ203Em95MRM8Cr6jhCXNF/FAXTpCTRTECnqQWLpiTRqrFtdYcth8hf4WECUpkezuYsMVww== + version "1.0.30001515" + resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001515.tgz" + integrity sha512-eEFDwUOZbE24sb+Ecsx3+OvNETqjWIdabMy52oOkIgcUtAsQifjUG9q4U9dgTHJM2mfk4uEPxc0+xuFdJ629QA== caseless@~0.12.0: version "0.12.0" From 452e5cfe7e750a46c9d28eba7199ec4a3c899f1e Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Fri, 14 Jul 2023 11:36:50 +0200 Subject: [PATCH 002/214] fix(tile-converter): i3s->3dtiles regression (#2550) --- modules/tile-converter/package.json | 6 +- .../src/3d-tiles-attributes-worker.ts | 43 -------------- .../3d-tiles-converter/3d-tiles-converter.ts | 38 ++++-------- .../helpers/b3dm-converter.ts | 7 ++- .../src/i3s-attributes-worker.ts | 59 ------------------- .../helpers/batch-ids-extensions.ts | 2 +- .../helpers/geometry-converter.ts | 9 +-- .../i3s-converter/helpers/gltf-attributes.ts | 4 +- .../src/i3s-converter/i3s-converter.ts | 5 -- .../tile-converter/src/i3s-converter/types.ts | 33 ++++++++++- .../src/workers/3d-tiles-attributes-worker.ts | 6 -- .../src/workers/i3s-attributes-worker.ts | 7 --- .../helpers/geometry-converter.spec.js | 6 -- 13 files changed, 54 insertions(+), 171 deletions(-) delete mode 100644 modules/tile-converter/src/3d-tiles-attributes-worker.ts delete mode 100644 modules/tile-converter/src/i3s-attributes-worker.ts delete mode 100644 modules/tile-converter/src/workers/3d-tiles-attributes-worker.ts delete mode 100644 modules/tile-converter/src/workers/i3s-attributes-worker.ts diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index 32b9470e08..220faaa114 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -38,12 +38,10 @@ "join-images": false }, "scripts": { - "pre-build": "npm run build-bundle && npm run build-converter-bundle && npm run build-i3s-attributes-worker && npm run build-slpk-extractor-bundle && npm run build-3d-tiles-attributes-worker", + "pre-build": "npm run build-bundle && npm run build-converter-bundle && npm run build-slpk-extractor-bundle", "build-bundle": "esbuild ./src/index.ts --bundle --outfile=dist/dist.min.js --platform=node --external:join-images", "build-converter-bundle": "esbuild src/converter-cli.ts --outfile=dist/converter.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", - "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", - "build-i3s-attributes-worker": "esbuild src/workers/i3s-attributes-worker.ts --outfile=dist/i3s-attributes-worker.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"", - "build-3d-tiles-attributes-worker": "esbuild src/workers/3d-tiles-attributes-worker.ts --outfile=dist/3d-tiles-attributes-worker.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"" + "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { "@loaders.gl/3d-tiles": "4.0.0-alpha.13", diff --git a/modules/tile-converter/src/3d-tiles-attributes-worker.ts b/modules/tile-converter/src/3d-tiles-attributes-worker.ts deleted file mode 100644 index 5361ec9c07..0000000000 --- a/modules/tile-converter/src/3d-tiles-attributes-worker.ts +++ /dev/null @@ -1,43 +0,0 @@ -import type {WorkerObject} from '@loaders.gl/worker-utils'; -import type {FeatureAttribute} from '@loaders.gl/i3s'; - -import {processOnWorker} from '@loaders.gl/worker-utils'; - -// __VERSION__ is injected by babel-plugin-version-inline -// @ts-ignore TS2304: Cannot find name '__VERSION__'. -const VERSION = typeof __VERSION__ !== 'undefined' ? __VERSION__ : 'latest'; - -export type Tile3DAttributesWorkerOptions = { - featureAttributes: FeatureAttribute | null; - source: string; -}; - -export type I3SAttributesData = { - tileContent: any; - textureFormat: string; -}; - -/** - * I3S Attributes Worker to handle B3DM object - */ -export const Tile3dAttributesWorker = { - id: '3d-tiles-attributes', - name: '3DTiles Attributes Worker', - module: 'tile-converter', - version: VERSION, - options: { - featureAttributes: null - } -}; - -/** - * Performs I3S attributes transformation - */ -export function transform3DTilesAttributesOnWorker( - i3sAttributesData: I3SAttributesData, - options: Tile3DAttributesWorkerOptions -): Promise { - return processOnWorker(Tile3dAttributesWorker, i3sAttributesData, options); -} - -export const _typecheckI3SAttributesWorker: WorkerObject = Tile3dAttributesWorker; diff --git a/modules/tile-converter/src/3d-tiles-converter/3d-tiles-converter.ts b/modules/tile-converter/src/3d-tiles-converter/3d-tiles-converter.ts index 8c4d28c99e..3b6cbf47b8 100644 --- a/modules/tile-converter/src/3d-tiles-converter/3d-tiles-converter.ts +++ b/modules/tile-converter/src/3d-tiles-converter/3d-tiles-converter.ts @@ -4,7 +4,7 @@ import type {Tiles3DTileJSON} from '@loaders.gl/3d-tiles'; import {join} from 'path'; import process from 'process'; import transform from 'json-map-transform'; -import {fetchFile, getLoaderOptions, load, isBrowser} from '@loaders.gl/core'; +import {load, isBrowser} from '@loaders.gl/core'; import {I3SLoader, I3SAttributeLoader, COORDINATE_SYSTEM} from '@loaders.gl/i3s'; import {Tileset3D, Tile3D} from '@loaders.gl/tiles'; import {Geoid} from '@math.gl/geoid'; @@ -16,13 +16,9 @@ import {writeFile, removeDir} from '../lib/utils/file-utils'; import {calculateFilesSize, timeConverter} from '../lib/utils/statistic-utills'; import {TILESET as tilesetTemplate} from './json-templates/tileset'; import {createObbFromMbs} from '../i3s-converter/helpers/coordinate-converter'; -import { - I3SAttributesData, - Tile3dAttributesWorker, - transform3DTilesAttributesOnWorker -} from '../3d-tiles-attributes-worker'; -import {getWorkerURL, WorkerFarm} from '@loaders.gl/worker-utils'; +import {WorkerFarm} from '@loaders.gl/worker-utils'; import {BROWSER_ERROR_MESSAGE} from '../constants'; +import B3dmConverter, {I3SAttributesData} from './helpers/b3dm-converter'; const I3S = 'I3S'; @@ -78,19 +74,17 @@ export default class Tiles3DConverter { this.geoidHeightModel = await load(egmFilePath, PGMLoader); console.log('Loading egm file completed!'); // eslint-disable-line - await this.loadWorkers(); - const sourceTilesetJson = await load(inputUrl, I3SLoader, {}); this.sourceTileset = new Tileset3D(sourceTilesetJson, { loadOptions: { _nodeWorkers: true, reuseWorkers: true, - i3s: {coordinateSystem: COORDINATE_SYSTEM.LNGLAT_OFFSETS, decodeTextures: false} - // TODO should no longer be needed with new workers - // 'i3s-content-nodejs': { - // workerUrl: './modules/i3s/dist/i3s-content-nodejs-worker.js' - // } + i3s: {coordinateSystem: COORDINATE_SYSTEM.LNGLAT_OFFSETS, decodeTextures: false}, + // We need to load local fs workers because nodejs can't load workers from the Internet + 'i3s-content': { + workerUrl: './modules/i3s/dist/i3s-content-worker-node.js' + } } }); @@ -171,10 +165,8 @@ export default class Tiles3DConverter { textureFormat: sourceChild?.header?.textureFormat }; - const b3dm = await transform3DTilesAttributesOnWorker(i3sAttributesData, { - source: this.workerSource.tile3dWorkerSource, - featureAttributes - }); + const b3dmConverter = new B3dmConverter(); + const b3dm = await b3dmConverter.convert(i3sAttributesData, featureAttributes); child.content = { uri: `${sourceChild.id}.b3dm`, @@ -342,14 +334,4 @@ export default class Tiles3DConverter { console.log(`File(s) size: `, filesSize, ' bytes'); // eslint-disable-line console.log(`------------------------------------------------`); // eslint-disable-line } - - private async loadWorkers(): Promise { - console.log(`Loading workers source...`); // eslint-disable-line no-undef, no-console - const tile3dAttributesWorkerUrl = getWorkerURL(Tile3dAttributesWorker, {...getLoaderOptions()}); - const sourceResponse = await fetchFile(tile3dAttributesWorkerUrl); - const source = await sourceResponse.text(); - - this.workerSource.tile3dWorkerSource = source; - console.log(`Loading workers source completed!`); // eslint-disable-line no-undef, no-console - } } diff --git a/modules/tile-converter/src/3d-tiles-converter/helpers/b3dm-converter.ts b/modules/tile-converter/src/3d-tiles-converter/helpers/b3dm-converter.ts index 65b4d90298..59998ed265 100644 --- a/modules/tile-converter/src/3d-tiles-converter/helpers/b3dm-converter.ts +++ b/modules/tile-converter/src/3d-tiles-converter/helpers/b3dm-converter.ts @@ -1,5 +1,3 @@ -import type {I3SAttributesData} from '../../3d-tiles-attributes-worker'; - import {encodeSync} from '@loaders.gl/core'; import {GLTFScenegraph, GLTFWriter} from '@loaders.gl/gltf'; import {Tile3DWriter} from '@loaders.gl/3d-tiles'; @@ -11,6 +9,11 @@ import {generateSyntheticIndices} from '../../lib/utils/geometry-utils'; const Z_UP_TO_Y_UP_MATRIX = new Matrix4([1, 0, 0, 0, 0, 0, -1, 0, 0, 1, 0, 0, 0, 0, 0, 1]); const scratchVector = new Vector3(); +export type I3SAttributesData = { + tileContent: any; + textureFormat: string; +}; + /** * Converts content of an I3S node to *.b3dm's file content */ diff --git a/modules/tile-converter/src/i3s-attributes-worker.ts b/modules/tile-converter/src/i3s-attributes-worker.ts deleted file mode 100644 index 0de06d79fc..0000000000 --- a/modules/tile-converter/src/i3s-attributes-worker.ts +++ /dev/null @@ -1,59 +0,0 @@ -import type {WorkerObject} from '@loaders.gl/worker-utils'; -import type {ConvertedAttributes} from './i3s-converter/types'; -import type {Matrix4, Vector3} from '@math.gl/core'; -import type {GLTFNodePostprocessed} from '@loaders.gl/gltf'; - -import {processOnWorker} from '@loaders.gl/worker-utils'; - -// __VERSION__ is injected by babel-plugin-version-inline -// @ts-ignore TS2304: Cannot find name '__VERSION__'. -const VERSION = typeof __VERSION__ !== 'undefined' ? __VERSION__ : 'latest'; - -export type I3SAttributesWorkerOptions = { - _nodeWorkers: boolean; - reuseWorkers: boolean; - useCartesianPositions: boolean; - source: string; -}; - -export type TextureImageProperties = { - data: Uint8Array; - compressed?: boolean; - height?: number; - width?: number; - components?: number; - mimeType?: string; -}; - -export type B3DMAttributesData = { - gltfMaterials?: {id: string}[]; - nodes: GLTFNodePostprocessed[]; - images: (null | TextureImageProperties)[]; - cartographicOrigin: Vector3; - cartesianModelMatrix: Matrix4; -}; - -/** - * I3S Attributes Worker to handle B3DM object - */ -export const I3SAttributesWorker = { - id: 'i3s-attributes', - name: 'I3S Attributes Worker', - module: 'tile-converter', - version: VERSION, - options: { - useCartesianPositions: false - } -}; - -/** - * Performs I3S attributes transformation - */ -export function transformI3SAttributesOnWorker( - attributesData: B3DMAttributesData, - options: I3SAttributesWorkerOptions -): Promise> { - return processOnWorker(I3SAttributesWorker, attributesData, options); -} - -export const _typecheckI3SAttributesWorker: WorkerObject = I3SAttributesWorker; diff --git a/modules/tile-converter/src/i3s-converter/helpers/batch-ids-extensions.ts b/modules/tile-converter/src/i3s-converter/helpers/batch-ids-extensions.ts index 4cd55aefab..b10a8bf237 100644 --- a/modules/tile-converter/src/i3s-converter/helpers/batch-ids-extensions.ts +++ b/modules/tile-converter/src/i3s-converter/helpers/batch-ids-extensions.ts @@ -5,7 +5,7 @@ import type { GLTF_EXT_feature_metadata_Primitive } from '@loaders.gl/gltf'; import {TypedArray} from '@math.gl/core'; -import {TextureImageProperties} from '../../i3s-attributes-worker'; +import {TextureImageProperties} from '../types'; const EXT_MESH_FEATURES = 'EXT_mesh_features'; const EXT_FEATURE_METADATA = 'EXT_feature_metadata'; diff --git a/modules/tile-converter/src/i3s-converter/helpers/geometry-converter.ts b/modules/tile-converter/src/i3s-converter/helpers/geometry-converter.ts index 313f231eba..37f3a69bf4 100644 --- a/modules/tile-converter/src/i3s-converter/helpers/geometry-converter.ts +++ b/modules/tile-converter/src/i3s-converter/helpers/geometry-converter.ts @@ -34,11 +34,6 @@ import { } from '@loaders.gl/i3s'; import {NumberArray, TypedArray} from '@loaders.gl/loader-utils'; import {Geoid} from '@math.gl/geoid'; -/** Usage of worker here brings more overhead than advantage */ -import { - B3DMAttributesData /*, transformI3SAttributesOnWorker*/, - TextureImageProperties -} from '../../i3s-attributes-worker'; import {prepareDataForAttributesConversion} from './gltf-attributes'; import {handleBatchIdsExtensions} from './batch-ids-extensions'; import {checkPropertiesLength, flattenPropertyTableByFeatureIds} from './feature-attributes'; @@ -49,7 +44,7 @@ import {GL} from '@loaders.gl/math'; So the following import is replaced with the local import import type {TypedArrayConstructor} from '@math.gl/types'; */ -import type {TypedArrayConstructor} from '../types'; +import type {GLTFAttributesData, TextureImageProperties, TypedArrayConstructor} from '../types'; import {generateSyntheticIndices} from '../../lib/utils/geometry-utils'; import {BoundingSphere, OrientedBoundingBox} from '@math.gl/culling'; @@ -320,7 +315,7 @@ async function _makeNodeResources({ * @returns map of converted geometry attributes */ export async function convertAttributes( - attributesData: B3DMAttributesData, + attributesData: GLTFAttributesData, materialAndTextureList: I3SMaterialWithTexture[], useCartesianPositions: boolean ): Promise> { diff --git a/modules/tile-converter/src/i3s-converter/helpers/gltf-attributes.ts b/modules/tile-converter/src/i3s-converter/helpers/gltf-attributes.ts index eb83890cb1..0c2af1d9c7 100644 --- a/modules/tile-converter/src/i3s-converter/helpers/gltf-attributes.ts +++ b/modules/tile-converter/src/i3s-converter/helpers/gltf-attributes.ts @@ -1,9 +1,9 @@ import type {Tiles3DTileContent} from '@loaders.gl/3d-tiles'; import type {GLTFAccessorPostprocessed, GLTFNodePostprocessed} from '@loaders.gl/gltf'; -import type {B3DMAttributesData} from '../../i3s-attributes-worker'; import {Matrix4, TypedArray, Vector3} from '@math.gl/core'; import {BoundingSphere, OrientedBoundingBox} from '@math.gl/culling'; import {Ellipsoid} from '@math.gl/geospatial'; +import {GLTFAttributesData} from '../types'; /** * Prepare attributes for conversion to avoid binary data breaking in worker thread. @@ -17,7 +17,7 @@ export function prepareDataForAttributesConversion( tileContent: Tiles3DTileContent, tileTransform: Matrix4, boundingVolume: OrientedBoundingBox | BoundingSphere -): B3DMAttributesData { +): GLTFAttributesData { let nodes = tileContent.gltf?.scene?.nodes || tileContent.gltf?.scenes?.[0]?.nodes || diff --git a/modules/tile-converter/src/i3s-converter/i3s-converter.ts b/modules/tile-converter/src/i3s-converter/i3s-converter.ts index a95a57dc92..5a5a0f10a1 100644 --- a/modules/tile-converter/src/i3s-converter/i3s-converter.ts +++ b/modules/tile-converter/src/i3s-converter/i3s-converter.ts @@ -59,7 +59,6 @@ import { import {getWorkerURL, WorkerFarm} from '@loaders.gl/worker-utils'; import {DracoWriterWorker} from '@loaders.gl/draco'; import WriteQueue from '../lib/utils/write-queue'; -import {I3SAttributesWorker} from '../i3s-attributes-worker'; import {BROWSER_ERROR_MESSAGE} from '../constants'; import { createdStorageAttribute, @@ -1211,10 +1210,6 @@ export default class I3SConverter { this.workerSource.ktx2 = source; } - const i3sAttributesWorkerUrl = getWorkerURL(I3SAttributesWorker, {...getLoaderOptions()}); - const sourceResponse = await fetchFile(i3sAttributesWorkerUrl); - const source = await sourceResponse.text(); - this.workerSource.I3SAttributes = source; console.log(`Loading workers source completed!`); // eslint-disable-line no-undef, no-console } } diff --git a/modules/tile-converter/src/i3s-converter/types.ts b/modules/tile-converter/src/i3s-converter/types.ts index aedd2ca84e..275ba64645 100644 --- a/modules/tile-converter/src/i3s-converter/types.ts +++ b/modules/tile-converter/src/i3s-converter/types.ts @@ -1,10 +1,11 @@ -import {GLTFImagePostprocessed} from '@loaders.gl/gltf'; +import {GLTFImagePostprocessed, GLTFNodePostprocessed} from '@loaders.gl/gltf'; import { BoundingVolumes, I3SMaterialDefinition, MaterialDefinitionInfo, TextureDefinitionInfo } from '@loaders.gl/i3s'; +import {Matrix4, Vector3} from '@math.gl/core'; /** Converted resources for specific node */ export type I3SConvertedResources = { @@ -183,3 +184,33 @@ export type PreprocessData = { /** Mesh topology types used in gltf primitives of the tileset */ meshTopologyTypes: Set; }; + +/** Texture image properties required for conversion */ +export type TextureImageProperties = { + /** Array with image data */ + data: Uint8Array; + /** Is the texture compressed */ + compressed?: boolean; + /** Height of the texture's image */ + height?: number; + /** Width of the texture's image */ + width?: number; + /** Number of components (3 for RGB, 4 for RGBA) */ + components?: number; + /** Mime type of the texture's image */ + mimeType?: string; +}; + +/** glTF attributes data, prepared for conversion */ +export type GLTFAttributesData = { + /** glTF PBR materials (only id is required) */ + gltfMaterials?: {id: string}[]; + /** glTF geometry nodes */ + nodes: GLTFNodePostprocessed[]; + /** glTF texture images (set to null for compressed textures) */ + images: (null | TextureImageProperties)[]; + /** Source tile origin coordinates in cartographic coordinate system */ + cartographicOrigin: Vector3; + /** Model matrix to convert coordinate system of POSITION and NORMAL attributes from METER_OFFSETS to CARTESIAN */ + cartesianModelMatrix: Matrix4; +}; diff --git a/modules/tile-converter/src/workers/3d-tiles-attributes-worker.ts b/modules/tile-converter/src/workers/3d-tiles-attributes-worker.ts deleted file mode 100644 index 7361d4ebcc..0000000000 --- a/modules/tile-converter/src/workers/3d-tiles-attributes-worker.ts +++ /dev/null @@ -1,6 +0,0 @@ -import {createWorker} from '@loaders.gl/worker-utils'; -import B3dmConverter from '../3d-tiles-converter/helpers/b3dm-converter'; - -const b3dmConverter = new B3dmConverter(); - -createWorker(async (data, options = {}) => b3dmConverter.convert(data, options.featureAttributes)); diff --git a/modules/tile-converter/src/workers/i3s-attributes-worker.ts b/modules/tile-converter/src/workers/i3s-attributes-worker.ts deleted file mode 100644 index 3d78b9ce0c..0000000000 --- a/modules/tile-converter/src/workers/i3s-attributes-worker.ts +++ /dev/null @@ -1,7 +0,0 @@ -import {createWorker} from '@loaders.gl/worker-utils'; -import {convertAttributes} from '../i3s-converter/helpers/geometry-converter'; - -createWorker( - async (data, options = {}) => - await convertAttributes(data, options.materialAndTextureList, options.useCartesianPositions) -); diff --git a/modules/tile-converter/test/i3s-converter/helpers/geometry-converter.spec.js b/modules/tile-converter/test/i3s-converter/helpers/geometry-converter.spec.js index d312c4da54..0967712d02 100644 --- a/modules/tile-converter/test/i3s-converter/helpers/geometry-converter.spec.js +++ b/modules/tile-converter/test/i3s-converter/helpers/geometry-converter.spec.js @@ -8,7 +8,6 @@ import convertB3dmToI3sGeometry, { } from '../../../src/i3s-converter/helpers/geometry-converter'; import {PGMLoader} from '../../../src/pgm-loader'; import {createdStorageAttribute} from '../../../src/i3s-converter/helpers/feature-attributes'; -import {I3SAttributesWorker} from '../../../src/i3s-attributes-worker'; import {BoundingSphere} from '@math.gl/culling'; import {Matrix4} from '@math.gl/core'; @@ -514,11 +513,6 @@ async function getWorkersSource() { let source = await sourceResponse.text(); result.draco = source; - const i3sAttributesWorkerUrl = getWorkerURL(I3SAttributesWorker, {...getLoaderOptions()}); - sourceResponse = await fetchFile(i3sAttributesWorkerUrl); - source = await sourceResponse.text(); - result.I3SAttributes = source; - return result; } From 0b4331310b4803cc3bddb9cae78b6d82df5a070d Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Fri, 14 Jul 2023 12:37:16 +0200 Subject: [PATCH 003/214] fix(3d-tiles): implicit tiling v1.1 (#2549) --- .../helpers/parse-3d-implicit-tiles.ts | 68 ++++++++++++------- .../src/lib/parsers/parse-3d-tile-header.ts | 42 ++++++++++-- modules/3d-tiles/src/types.ts | 4 ++ modules/3d-tiles/test/tiles-3d-loader.spec.ts | 2 +- 4 files changed, 85 insertions(+), 31 deletions(-) diff --git a/modules/3d-tiles/src/lib/parsers/helpers/parse-3d-implicit-tiles.ts b/modules/3d-tiles/src/lib/parsers/helpers/parse-3d-implicit-tiles.ts index 9bf60b3018..0223ab8633 100644 --- a/modules/3d-tiles/src/lib/parsers/helpers/parse-3d-implicit-tiles.ts +++ b/modules/3d-tiles/src/lib/parsers/helpers/parse-3d-implicit-tiles.ts @@ -7,6 +7,8 @@ import {getS2CellIdFromToken, getS2ChildCellId, getS2TokenFromCellId} from '../. import type {S2VolumeInfo} from '../../utils/obb/s2-corners-to-obb'; import {convertS2BoundingVolumetoOBB} from '../../utils/obb/s2-corners-to-obb'; import Long from 'long'; +import {Tiles3DLoaderOptions} from '../../../tiles-3d-loader'; +import {ImplicitOptions} from '../parse-3d-tile-header'; const QUADTREE_DEVISION_COUNT = 4; const OCTREE_DEVISION_COUNT = 8; @@ -84,15 +86,16 @@ function getChildS2VolumeBox( // eslint-disable-next-line max-statements export async function parseImplicitTiles(params: { subtree: Subtree; - options: any; + implicitOptions: ImplicitOptions; parentData?: {mortonIndex: number; x: number; y: number; z: number}; childIndex?: number; level?: number; globalData?: {level: number; mortonIndex: number; x: number; y: number; z: number}; s2VolumeBox?: S2VolumeBox; + loaderOptions: Tiles3DLoaderOptions; }) { const { - options, + implicitOptions, parentData = { mortonIndex: 0, x: 0, @@ -107,7 +110,8 @@ export async function parseImplicitTiles(params: { y: 0, z: 0 }, - s2VolumeBox + s2VolumeBox, + loaderOptions } = params; let {subtree, level = 0} = params; const { @@ -117,44 +121,56 @@ export async function parseImplicitTiles(params: { contentUrlTemplate, subtreesUriTemplate, basePath - } = options; - + } = implicitOptions; const tile = {children: [], lodMetricValue: 0, contentUrl: ''}; + if (!maximumLevel) { + // eslint-disable-next-line no-console + log.once( + `Missing 'maximumLevel' or 'availableLevels' property. The subtree ${contentUrlTemplate} won't be loaded...` + ); + return tile; + } + + const lev = level + globalData.level; + if (lev > maximumLevel) { + return tile; + } + const childrenPerTile = SUBDIVISION_COUNT_MAP[subdivisionScheme]; + const bitsPerTile = Math.log2(childrenPerTile); - // childIndex is in range [0, 7] + // childIndex is in range [0,4] for quadtrees and [0, 7] for octrees const childX = childIndex & 0b01; // Get first bit for X const childY = (childIndex >> 1) & 0b01; // Get second bit for Y const childZ = (childIndex >> 2) & 0b01; // Get third bit for Z const levelOffset = (childrenPerTile ** level - 1) / (childrenPerTile - 1); - let childTileMortonIndex = concatBits(parentData.mortonIndex, childIndex); + let childTileMortonIndex = concatBits(parentData.mortonIndex, childIndex, bitsPerTile); let tileAvailabilityIndex = levelOffset + childTileMortonIndex; // Local tile coordinates - let childTileX = concatBits(parentData.x, childX); - let childTileY = concatBits(parentData.y, childY); - let childTileZ = concatBits(parentData.z, childZ); + let childTileX = concatBits(parentData.x, childX, 1); + let childTileY = concatBits(parentData.y, childY, 1); + let childTileZ = concatBits(parentData.z, childZ, 1); let isChildSubtreeAvailable = false; - if (level + 1 > subtreeLevels) { + if (level >= subtreeLevels) { isChildSubtreeAvailable = getAvailabilityResult( subtree.childSubtreeAvailability, childTileMortonIndex ); } - const x = concatBits(globalData.x, childTileX); - const y = concatBits(globalData.y, childTileY); - const z = concatBits(globalData.z, childTileZ); - const lev = level + globalData.level; + const x = concatBits(globalData.x, childTileX, level * bitsPerTile); + const y = concatBits(globalData.y, childTileY, level * bitsPerTile); + const z = concatBits(globalData.z, childTileZ, level * bitsPerTile); if (isChildSubtreeAvailable) { const subtreePath = `${basePath}/${subtreesUriTemplate}`; const childSubtreeUrl = replaceContentUrlTemplate(subtreePath, lev, x, y, z); - const childSubtree = await load(childSubtreeUrl, Tile3DSubtreeLoader); + const childSubtree = await load(childSubtreeUrl, Tile3DSubtreeLoader, loaderOptions); subtree = childSubtree; @@ -174,7 +190,7 @@ export async function parseImplicitTiles(params: { const isTileAvailable = getAvailabilityResult(subtree.tileAvailability, tileAvailabilityIndex); - if (!isTileAvailable || level > maximumLevel) { + if (!isTileAvailable) { return tile; } @@ -200,11 +216,12 @@ export async function parseImplicitTiles(params: { // Recursive calling... const childTileParsed = await parseImplicitTiles({ subtree, - options, + implicitOptions, + loaderOptions, parentData: pData, childIndex: index, level: childTileLevel, - globalData, + globalData: {...globalData}, s2VolumeBox: childS2VolumeBox }); @@ -215,7 +232,7 @@ export async function parseImplicitTiles(params: { childTileParsed, globalLevel, childCoordinates, - options, + implicitOptions, s2VolumeBox ); // @ts-ignore @@ -274,7 +291,7 @@ function formatTileData( tile, level: number, childCoordinates: {childTileX: number; childTileY: number; childTileZ: number}, - options: any, + options: ImplicitOptions, s2VolumeBox?: S2VolumeBox ) { const { @@ -363,11 +380,12 @@ function calculateBoundingVolumeForChildTile( /** * Do binary concatenation - * @param first - * @param second + * @param higher - number to put to higher part of result + * @param lower - number to put to lower part of result + * @param shift - number of bits to shift lower number */ -function concatBits(first: number, second: number): number { - return parseInt(first.toString(2) + second.toString(2), 2); +function concatBits(higher: number, lower: number, shift: number): number { + return (higher << shift) + lower; } /** diff --git a/modules/3d-tiles/src/lib/parsers/parse-3d-tile-header.ts b/modules/3d-tiles/src/lib/parsers/parse-3d-tile-header.ts index 8fd71f4f36..1ea00bc3c9 100644 --- a/modules/3d-tiles/src/lib/parsers/parse-3d-tile-header.ts +++ b/modules/3d-tiles/src/lib/parsers/parse-3d-tile-header.ts @@ -6,6 +6,7 @@ import {LOD_METRIC_TYPE, TILE_REFINEMENT, TILE_TYPE} from '@loaders.gl/tiles'; import { ImplicitTilingExensionData, Subtree, + Tile3DBoundingVolume, Tiles3DTileContentJSON, Tiles3DTileJSON, Tiles3DTileJSONPostprocessed, @@ -16,6 +17,34 @@ import {parseImplicitTiles, replaceContentUrlTemplate} from './helpers/parse-3d- import type {S2VolumeInfo} from '../utils/obb/s2-corners-to-obb'; import {convertS2BoundingVolumetoOBB} from '../utils/obb/s2-corners-to-obb'; +/** Options for recursive loading implicit subtrees */ +export type ImplicitOptions = { + /** Template of the full url of the content template */ + contentUrlTemplate: string; + /** Template of the full url of the subtree */ + subtreesUriTemplate: string; + /** Implicit subdivision scheme */ + subdivisionScheme: 'QUADTREE' | 'OCTREE' | string; + /** Levels per subtree */ + subtreeLevels: number; + /** Maximum implicit level through all subtrees */ + maximumLevel?: number; + /** 3DTiles refine method (add/replace) */ + refine?: string; + /** Tileset base path */ + basePath: string; + /** 3DTiles LOD metric type */ + lodMetricType: LOD_METRIC_TYPE.GEOMETRIC_ERROR; + /** Root metric value of the root tile of the implicit subtrees */ + rootLodMetricValue: number; + /** Bounding volume of the root tile of the implicit subtrees */ + rootBoundingVolume: Tile3DBoundingVolume; + /** Function that detects TILE_TYPE by tile metadata and content URL */ + getTileType: (tile: Tiles3DTileJSON, tileContentUrl?: string) => TILE_TYPE | string; + /** Function that converts string refine method to enum value */ + getRefine: (refine?: string) => TILE_REFINEMENT | string | undefined; +}; + function getTileType(tile: Tiles3DTileJSON, tileContentUrl: string = ''): TILE_TYPE | string { if (!tileContentUrl) { return TILE_TYPE.EMPTY; @@ -158,6 +187,7 @@ export async function normalizeImplicitTileHeaders( const { subdivisionScheme, maximumLevel, + availableLevels, subtreeLevels, subtrees: {uri: subtreesUriTemplate} } = implicitTilingExtension; @@ -179,12 +209,12 @@ export async function normalizeImplicitTileHeaders( const rootBoundingVolume = tile.boundingVolume; - const implicitOptions = { + const implicitOptions: ImplicitOptions = { contentUrlTemplate, subtreesUriTemplate, subdivisionScheme, subtreeLevels, - maximumLevel, + maximumLevel: Number.isFinite(availableLevels) ? availableLevels - 1 : maximumLevel, refine, basePath, lodMetricType: LOD_METRIC_TYPE.GEOMETRIC_ERROR, @@ -194,7 +224,7 @@ export async function normalizeImplicitTileHeaders( getRefine }; - return await normalizeImplicitTileData(tile, basePath, subtree, implicitOptions); + return await normalizeImplicitTileData(tile, basePath, subtree, implicitOptions, options); } /** @@ -208,7 +238,8 @@ export async function normalizeImplicitTileData( tile: Tiles3DTileJSON, basePath: string, rootSubtree: Subtree, - options: any + implicitOptions: ImplicitOptions, + loaderOptions: Tiles3DLoaderOptions ): Promise { if (!tile) { return null; @@ -216,7 +247,8 @@ export async function normalizeImplicitTileData( const {children, contentUrl} = await parseImplicitTiles({ subtree: rootSubtree, - options + implicitOptions, + loaderOptions }); let tileContentUrl: string | undefined; diff --git a/modules/3d-tiles/src/types.ts b/modules/3d-tiles/src/types.ts index b5ca5fdfd8..c5b6df0925 100644 --- a/modules/3d-tiles/src/types.ts +++ b/modules/3d-tiles/src/types.ts @@ -372,6 +372,10 @@ export type BufferView = { * Spec - https://github.com/CesiumGS/3d-tiles/tree/main/extensions/3DTILES_implicit_tiling */ export type ImplicitTilingExensionData = ImplicitTilingData & { + /** This property is not part of the schema + * https://github.com/CesiumGS/3d-tiles/blob/main/extensions/3DTILES_implicit_tiling/schema/tile.3DTILES_implicit_tiling.schema.json + * But it can be seen in some test datasets. It is handled as substitute of `availableLevels` + */ maximumLevel?: number; }; diff --git a/modules/3d-tiles/test/tiles-3d-loader.spec.ts b/modules/3d-tiles/test/tiles-3d-loader.spec.ts index 505efac95a..22985ca69c 100644 --- a/modules/3d-tiles/test/tiles-3d-loader.spec.ts +++ b/modules/3d-tiles/test/tiles-3d-loader.spec.ts @@ -168,7 +168,7 @@ test('Tiles3DLoader#Implicit Octree Tileset with bitstream availability and subt t.equal(tileset.root.children[0].children[0].children.length, 1); // children level 3 - t.equal(tileset.root.children[0].children[0].children[0].content.uri, 'content/3/2/0/1.pnts'); + t.equal(tileset.root.children[0].children[0].children[0].content.uri, 'content/3/8/0/1.pnts'); t.equal(tileset.root.children[0].children[0].children[0].lodMetricValue, 625); t.equal(tileset.root.children[0].children[0].children[0].refine, 1); t.equal(tileset.root.children[0].children[0].children[0].type, 'pointcloud'); From dd885592142ad9c26fc38e0eb0a711cf4806345a Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Fri, 14 Jul 2023 12:58:02 +0200 Subject: [PATCH 004/214] v4.0.0-alpha.14 --- lerna.json | 2 +- modules/3d-tiles/package.json | 12 ++++++------ modules/arrow/package.json | 6 +++--- modules/bson/package.json | 8 ++++---- modules/compression/package.json | 6 +++--- modules/core/package.json | 6 +++--- modules/crypto/package.json | 6 +++--- modules/csv/package.json | 6 +++--- modules/draco/package.json | 10 +++++----- modules/excel/package.json | 6 +++--- modules/flatgeobuf/package.json | 6 +++--- modules/geopackage/package.json | 8 ++++---- modules/geotiff/package.json | 2 +- modules/gis/package.json | 6 +++--- modules/gltf/package.json | 10 +++++----- modules/i3s/package.json | 16 ++++++++-------- modules/images/package.json | 4 ++-- modules/json/package.json | 8 ++++---- modules/kml/package.json | 8 ++++---- modules/las/package.json | 6 +++--- modules/loader-utils/package.json | 4 ++-- modules/math/package.json | 6 +++--- modules/mvt/package.json | 8 ++++---- modules/netcdf/package.json | 4 ++-- modules/obj/package.json | 6 +++--- modules/parquet/package.json | 10 +++++----- modules/pcd/package.json | 6 +++--- modules/ply/package.json | 6 +++--- modules/polyfills/package.json | 2 +- modules/potree/package.json | 4 ++-- modules/schema/package.json | 2 +- modules/shapefile/package.json | 8 ++++---- modules/terrain/package.json | 8 ++++---- modules/textures/package.json | 12 ++++++------ modules/tile-converter/package.json | 26 +++++++++++++------------- modules/tiles/package.json | 6 +++--- modules/video/package.json | 6 +++--- modules/wkt/package.json | 6 +++--- modules/wms/package.json | 10 +++++----- modules/worker-utils/package.json | 2 +- modules/xml/package.json | 6 +++--- modules/zarr/package.json | 2 +- modules/zip/package.json | 2 +- 43 files changed, 147 insertions(+), 147 deletions(-) diff --git a/lerna.json b/lerna.json index f54f8040db..ae9df994a2 100644 --- a/lerna.json +++ b/lerna.json @@ -1,6 +1,6 @@ { "lerna": "2.9.1", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "command": { "publish": {}, "bootstrap": {} diff --git a/modules/3d-tiles/package.json b/modules/3d-tiles/package.json index a5890468f8..ce8664a921 100644 --- a/modules/3d-tiles/package.json +++ b/modules/3d-tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/3d-tiles", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "3D Tiles, an open standard for streaming massive heterogeneous 3D geospatial datasets.", "license": "MIT", "publishConfig": { @@ -34,11 +34,11 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.13", - "@loaders.gl/gltf": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/math": "4.0.0-alpha.13", - "@loaders.gl/tiles": "4.0.0-alpha.13", + "@loaders.gl/draco": "4.0.0-alpha.14", + "@loaders.gl/gltf": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/math": "4.0.0-alpha.14", + "@loaders.gl/tiles": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1", "@math.gl/geospatial": "^3.5.1", "@probe.gl/log": "^4.0.4", diff --git a/modules/arrow/package.json b/modules/arrow/package.json index e7cbd22288..00348f679b 100644 --- a/modules/arrow/package.json +++ b/modules/arrow/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/arrow", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Simple columnar table loader for the Apache Arrow format", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ "build-worker2": "esbuild src/workers/arrow-worker.ts --bundle --outfile=dist/arrow-worker.js --platform=browser --external:{stream}" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/bson/package.json b/modules/bson/package.json index 133ab80b16..da52e9aea7 100644 --- a/modules/bson/package.json +++ b/modules/bson/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/bson", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -34,9 +34,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@types/bson": "4.2.0", "bson": "4.2.0" }, diff --git a/modules/compression/package.json b/modules/compression/package.json index 25d5b71326..c61433231e 100644 --- a/modules/compression/package.json +++ b/modules/compression/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/compression", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Decompression and compression plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "@types/brotli": "^1.3.0", "@types/pako": "^1.0.1", "fflate": "0.7.4", diff --git a/modules/core/package.json b/modules/core/package.json index 5b8323bf29..b88c0823cb 100644 --- a/modules/core/package.json +++ b/modules/core/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/core", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "The core API for working with loaders.gl loaders and writers", "license": "MIT", "publishConfig": { @@ -43,8 +43,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "@probe.gl/log": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/crypto/package.json b/modules/crypto/package.json index e156ad1600..3fd3c9f6be 100644 --- a/modules/crypto/package.json +++ b/modules/crypto/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/crypto", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Cryptographic/hashing plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "@types/crypto-js": "^4.0.2" }, "devDependencies": { diff --git a/modules/csv/package.json b/modules/csv/package.json index 5d77424937..2e22cb7625 100644 --- a/modules/csv/package.json +++ b/modules/csv/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/csv", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for CSV and DSV table formats", "license": "MIT", "publishConfig": { @@ -30,8 +30,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "devDependencies": { "d3-dsv": "^1.2.0" diff --git a/modules/draco/package.json b/modules/draco/package.json index 50402e47d7..727e531291 100644 --- a/modules/draco/package.json +++ b/modules/draco/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/draco", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader and writer for Draco compressed meshes and point clouds", "license": "MIT", "publishConfig": { @@ -42,13 +42,13 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "draco3d": "1.5.5" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.13" + "@loaders.gl/polyfills": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/excel/package.json b/modules/excel/package.json index feafbf4a15..3da3d72f44 100644 --- a/modules/excel/package.json +++ b/modules/excel/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/excel", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for Excel files", "license": "MIT", "publishConfig": { @@ -35,8 +35,8 @@ "build-worker": "esbuild src/workers/excel-worker.ts --bundle --outfile=dist/excel-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "xlsx": "^0.17.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/flatgeobuf/package.json b/modules/flatgeobuf/package.json index fbc1fadabb..f800fd1478 100644 --- a/modules/flatgeobuf/package.json +++ b/modules/flatgeobuf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/flatgeobuf", "description": "Loader for FlatGeobuf", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -32,8 +32,8 @@ "build-worker": "esbuild src/workers/flatgeobuf-worker.ts --bundle --outfile=dist/flatgeobuf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", "@math.gl/proj4": "^3.3.1", "flatgeobuf": "3.6.5" }, diff --git a/modules/geopackage/package.json b/modules/geopackage/package.json index e1fe18682d..6ded8f1c52 100644 --- a/modules/geopackage/package.json +++ b/modules/geopackage/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/geopackage", "description": "GeoPackage data loaders", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -25,9 +25,9 @@ ], "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", - "@loaders.gl/wkt": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/wkt": "4.0.0-alpha.14", "@math.gl/proj4": "^3.5.1", "@types/sql.js": "^1.4.2", "sql.js": "1.5.0" diff --git a/modules/geotiff/package.json b/modules/geotiff/package.json index 8f6d952dbe..14c6bb21bc 100644 --- a/modules/geotiff/package.json +++ b/modules/geotiff/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/geotiff", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for tiff and geotiff", "license": "MIT", "publishConfig": { diff --git a/modules/gis/package.json b/modules/gis/package.json index c43684895f..6d26e25844 100644 --- a/modules/gis/package.json +++ b/modules/gis/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/gis", "description": "Helpers for GIS category data", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -24,8 +24,8 @@ "README.md" ], "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@mapbox/vector-tile": "^1.3.1", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" diff --git a/modules/gltf/package.json b/modules/gltf/package.json index fb45a5b3f2..3c4e6f21c7 100644 --- a/modules/gltf/package.json +++ b/modules/gltf/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/gltf", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the glTF format", "license": "MIT", "publishConfig": { @@ -34,10 +34,10 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.13", - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/textures": "4.0.0-alpha.13", + "@loaders.gl/draco": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/textures": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/i3s/package.json b/modules/i3s/package.json index e5bbcfefd4..a131a63d60 100644 --- a/modules/i3s/package.json +++ b/modules/i3s/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/i3s", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "i3s .", "license": "MIT", "publishConfig": { @@ -33,13 +33,13 @@ "build-worker-node": "esbuild src/workers/i3s-content-worker-node.ts --outfile=dist/i3s-content-worker-node.js --platform=node --target=node16 --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/compression": "4.0.0-alpha.13", - "@loaders.gl/draco": "4.0.0-alpha.13", - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", - "@loaders.gl/textures": "4.0.0-alpha.13", - "@loaders.gl/tiles": "4.0.0-alpha.13", + "@loaders.gl/compression": "4.0.0-alpha.14", + "@loaders.gl/draco": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/textures": "4.0.0-alpha.14", + "@loaders.gl/tiles": "4.0.0-alpha.14", "@luma.gl/constants": "^8.5.4", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", diff --git a/modules/images/package.json b/modules/images/package.json index f1f300eef4..7140cceb6b 100644 --- a/modules/images/package.json +++ b/modules/images/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/images", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders and writers for images (PNG, JPG, ...)", "license": "MIT", "publishConfig": { @@ -32,7 +32,7 @@ "build-bundle": "esbuild src/bundle.ts --outfile=dist/dist.min.js --bundle --minify --sourcemap" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5", "devDependencies": { diff --git a/modules/json/package.json b/modules/json/package.json index 4a62eeb407..6f0f83572d 100644 --- a/modules/json/package.json +++ b/modules/json/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/json", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ "build-worker": "esbuild src/workers/geojson-worker.ts --bundle --outfile=dist/geojson-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/kml/package.json b/modules/kml/package.json index 107aa043cc..f5e660bc56 100644 --- a/modules/kml/package.json +++ b/modules/kml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/kml", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the KML format", "license": "MIT", "publishConfig": { @@ -32,9 +32,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@tmcw/togeojson": "^4.5.0", "@xmldom/xmldom": "^0.7.5" }, diff --git a/modules/las/package.json b/modules/las/package.json index 3f3735d222..29c03cd97f 100644 --- a/modules/las/package.json +++ b/modules/las/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/las", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the LAS and LAZ formats", "license": "MIT", "publishConfig": { @@ -41,8 +41,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/loader-utils/package.json b/modules/loader-utils/package.json index 933e73fc7b..9e62fccd4c 100644 --- a/modules/loader-utils/package.json +++ b/modules/loader-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/loader-utils", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for 3D graphics formats", "license": "MIT", "publishConfig": { @@ -38,7 +38,7 @@ "scripts": {}, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "@probe.gl/stats": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/math/package.json b/modules/math/package.json index f07c355868..81b2b6d492 100644 --- a/modules/math/package.json +++ b/modules/math/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/math", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Experimental math classes for loaders.gl", "license": "MIT", "publishConfig": { @@ -31,8 +31,8 @@ "pre-build": "echo \"Nothing to build in @loaders.gl/math\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/mvt/package.json b/modules/mvt/package.json index b5cc700408..5c54558c2d 100644 --- a/modules/mvt/package.json +++ b/modules/mvt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/mvt", "description": "Loader for Mapbox Vector Tiles", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -32,9 +32,9 @@ "build-worker": "esbuild src/workers/mvt-worker.ts --bundle --outfile=dist/mvt-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" }, diff --git a/modules/netcdf/package.json b/modules/netcdf/package.json index bf71f13e74..324035309d 100644 --- a/modules/netcdf/package.json +++ b/modules/netcdf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/netcdf", "description": "Loader for NetCDF", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -30,7 +30,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/obj/package.json b/modules/obj/package.json index a17ce58469..81a24ab007 100644 --- a/modules/obj/package.json +++ b/modules/obj/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/obj", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the OBJ format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/parquet/package.json b/modules/parquet/package.json index 0c473775b3..491051680f 100644 --- a/modules/parquet/package.json +++ b/modules/parquet/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/parquet", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for Apache Parquet files", "license": "MIT", "publishConfig": { @@ -41,10 +41,10 @@ "./src/lib/wasm/load-wasm/load-wasm-node.ts": "./src/lib/wasm/load-wasm/load-wasm-browser.ts" }, "dependencies": { - "@loaders.gl/bson": "4.0.0-alpha.13", - "@loaders.gl/compression": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/bson": "4.0.0-alpha.14", + "@loaders.gl/compression": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "async-mutex": "^0.2.2", "brotli": "^1.3.2", "int53": "^0.2.4", diff --git a/modules/pcd/package.json b/modules/pcd/package.json index 4a13c82c9e..2f953bbbb8 100644 --- a/modules/pcd/package.json +++ b/modules/pcd/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/pcd", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the PCD format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-worker": "esbuild src/workers/pcd-worker.ts --bundle --outfile=dist/pcd-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/ply/package.json b/modules/ply/package.json index 0c722292f5..72d78b0ae9 100644 --- a/modules/ply/package.json +++ b/modules/ply/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/ply", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for the PLY format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/polyfills/package.json b/modules/polyfills/package.json index adb3617d52..b0e569ab06 100644 --- a/modules/polyfills/package.json +++ b/modules/polyfills/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/polyfills", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Polyfills for TextEncoder/TextDecoder", "license": "MIT", "publishConfig": { diff --git a/modules/potree/package.json b/modules/potree/package.json index 07ee05f1c9..37c1b11b16 100644 --- a/modules/potree/package.json +++ b/modules/potree/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/potree", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "potree loaders for large point clouds.", "license": "MIT", "publishConfig": { @@ -35,7 +35,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/math": "4.0.0-alpha.13", + "@loaders.gl/math": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/schema/package.json b/modules/schema/package.json index d881cf1b61..a154e8f5e6 100644 --- a/modules/schema/package.json +++ b/modules/schema/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/schema", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Table format APIs for JSON, CSV, etc...", "license": "MIT", "publishConfig": { diff --git a/modules/shapefile/package.json b/modules/shapefile/package.json index 5bb2c6a081..414677b90f 100644 --- a/modules/shapefile/package.json +++ b/modules/shapefile/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/shapefile", "description": "Loader for the Shapefile Format", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -37,9 +37,9 @@ "build-worker-dbf": "esbuild src/workers/dbf-worker.ts --bundle --outfile=dist/dbf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/gis": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@math.gl/proj4": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/terrain/package.json b/modules/terrain/package.json index 05f4afc1b8..a1774c81f8 100644 --- a/modules/terrain/package.json +++ b/modules/terrain/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/terrain", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loader for terrain raster formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "@mapbox/martini": "^0.2.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/textures/package.json b/modules/textures/package.json index 8c43b754a2..51bdfbdeb7 100644 --- a/modules/textures/package.json +++ b/modules/textures/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/textures", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for compressed and super compressed (basis) textures ", "license": "MIT", "publishConfig": { @@ -45,15 +45,15 @@ "build-crunch-worker": "esbuild src/workers/crunch-worker.ts --outfile=dist/crunch-worker.js --target=esnext --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "ktx-parse": "^0.0.4", "texture-compressor": "^1.0.2" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.13" + "@loaders.gl/polyfills": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index 220faaa114..aa9ad48f5e 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tile-converter", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Converter", "license": "MIT", "publishConfig": { @@ -44,18 +44,18 @@ "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/3d-tiles": "4.0.0-alpha.13", - "@loaders.gl/crypto": "4.0.0-alpha.13", - "@loaders.gl/draco": "4.0.0-alpha.13", - "@loaders.gl/gltf": "4.0.0-alpha.13", - "@loaders.gl/i3s": "4.0.0-alpha.13", - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/polyfills": "4.0.0-alpha.13", - "@loaders.gl/textures": "4.0.0-alpha.13", - "@loaders.gl/tiles": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", - "@loaders.gl/zip": "4.0.0-alpha.13", + "@loaders.gl/3d-tiles": "4.0.0-alpha.14", + "@loaders.gl/crypto": "4.0.0-alpha.14", + "@loaders.gl/draco": "4.0.0-alpha.14", + "@loaders.gl/gltf": "4.0.0-alpha.14", + "@loaders.gl/i3s": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/polyfills": "4.0.0-alpha.14", + "@loaders.gl/textures": "4.0.0-alpha.14", + "@loaders.gl/tiles": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/zip": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geoid": "^3.5.1", diff --git a/modules/tiles/package.json b/modules/tiles/package.json index 8e863978c4..fc63ad55cf 100644 --- a/modules/tiles/package.json +++ b/modules/tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tiles", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Common components for different tiles loaders.", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/math": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/math": "4.0.0-alpha.14", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geospatial": "^3.5.1", diff --git a/modules/video/package.json b/modules/video/package.json index c9da64377f..ac0e4ff589 100644 --- a/modules/video/package.json +++ b/modules/video/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/video", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders and writers for video (MP4, WEBM, ...)", "license": "MIT", "publishConfig": { @@ -32,8 +32,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/worker-utils": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.14", "gifshot": "^0.4.5" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/wkt/package.json b/modules/wkt/package.json index 6d25dafa0b..79baf3448f 100644 --- a/modules/wkt/package.json +++ b/modules/wkt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/wkt", "description": "Loader and Writer for the WKT (Well Known Text) Format", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "license": "MIT", "publishConfig": { "access": "public" @@ -35,8 +35,8 @@ "fuzzer": "^0.2.1" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13" + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/wms/package.json b/modules/wms/package.json index b7b3ee8fad..80a5843557 100644 --- a/modules/wms/package.json +++ b/modules/wms/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/wms", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for the WMS (Web Map Service) standard", "license": "MIT", "publishConfig": { @@ -39,10 +39,10 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.13", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", - "@loaders.gl/xml": "4.0.0-alpha.13", + "@loaders.gl/images": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/xml": "4.0.0-alpha.14", "@turf/rewind": "^5.1.5", "deep-strict-equal": "^0.2.0", "lerc": "^4.0.1" diff --git a/modules/worker-utils/package.json b/modules/worker-utils/package.json index 5e0f5362ff..21d8d56f1b 100644 --- a/modules/worker-utils/package.json +++ b/modules/worker-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/worker-utils", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Utilities for running tasks on worker threads", "license": "MIT", "publishConfig": { diff --git a/modules/xml/package.json b/modules/xml/package.json index 51ef59b291..950effb7a5 100644 --- a/modules/xml/package.json +++ b/modules/xml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/xml", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for the XML (eXtensible Markup Language) format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.13", - "@loaders.gl/schema": "4.0.0-alpha.13", + "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/schema": "4.0.0-alpha.14", "fast-xml-parser": "^4.2.5" }, "devDependencies": { diff --git a/modules/zarr/package.json b/modules/zarr/package.json index 57c1f8041e..10101b6828 100644 --- a/modules/zarr/package.json +++ b/modules/zarr/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zarr", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Framework-independent loaders for Zarr", "license": "MIT", "publishConfig": { diff --git a/modules/zip/package.json b/modules/zip/package.json index 86a799d18b..fa92382ec4 100644 --- a/modules/zip/package.json +++ b/modules/zip/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zip", - "version": "4.0.0-alpha.13", + "version": "4.0.0-alpha.14", "description": "Zip Archive Loader", "license": "MIT", "publishConfig": { From ed13bdbb2f8cc64590e4b80d8f49be4f4d19ba50 Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Tue, 18 Jul 2023 09:41:54 +0200 Subject: [PATCH 005/214] chore(tile-converter): update i3s-server manual (#2552) --- .../tile-converter/src/i3s-server/README.md | 48 ++++++++++++++++++- 1 file changed, 46 insertions(+), 2 deletions(-) diff --git a/modules/tile-converter/src/i3s-server/README.md b/modules/tile-converter/src/i3s-server/README.md index 27d27d215e..9c010579ce 100644 --- a/modules/tile-converter/src/i3s-server/README.md +++ b/modules/tile-converter/src/i3s-server/README.md @@ -8,12 +8,56 @@ The server provides I3S Rest endpoints per specification https://github.com/Esri ### Serve 3DTiles to I3S converted dataset - Convert data set from 3DTiles to I3S without `--slpk` option -- Serve output folder `I3sLayerPath="./data/BatchedTextured" DEBUG=i3s-server:* npx i3s-server` +- Serve output folder + +Example for path `./data/BatchTextured/SceneServer/layers/0/...`: + +#### Start the server + +```bash +PORT=8080 HTTPS_PORT=4443 I3sLayerPath="./data" DEBUG=i3s-server:* npx i3s-server +``` +#### Check the layer availability + +The layer should be available on URLs: + +- `http://localhost:8080/BatchTextured/SceneServer/layers/0/...` +- `https://localhost:4443/BatchTextured/SceneServer/layers/0/...` + +#### Open in ArcGIS + +`https://www.arcgis.com/home/webscene/viewer.html?url=http://localhost:8080/BatchTextured/SceneServer/layers/0/` + +#### Open in I3S Explorer + +`https://i3s.loaders.gl/viewer?tileset=http://localhost:8080/BatchTextured/SceneServer/layers/0` ### Serve SLPK -- Serve slpk file `I3sLayerPath="../datasets/Rancho_Mesh_mesh_v17_1.slpk" DEBUG=i3s-server:* npx i3s-server` +Example for path `../datasets/Rancho_Mesh_mesh_v17_1.slpk`: + +#### Start the server + +```bash +PORT=8080 HTTPS_PORT=4443 I3sLayerPath="../datasets/Rancho_Mesh_mesh_v17_1.slpk" DEBUG=i3s-server:* npx i3s-server +``` +#### Check the layer availability + +The layer should be available on URLs: + +- `http://localhost:8080/SceneServer/layers/0/...` +- `https://localhost:4443/SceneServer/layers/0/...` + +#### Open in ArcGIS + +`https://www.arcgis.com/home/webscene/viewer.html?url=http://localhost:8080/SceneServer` + +#### Open in I3S Explorer + +`https://i3s.loaders.gl/viewer?tileset=http://localhost:8080/SceneServer/layers/0` ## ENV variables - `I3sLayerPath` - path to converted data or SLPK file. +- `PORT` - HTTP port. Eg for `PORT = 8080 npx i3s-server` the server will work on host `http://localhost:8080/...`. Default value is `80`; +- `HTTPS_PORT` - HTTPS port. Eg for `PORT = 4443 npx i3s-server` the server will work on host `https://localhost:4443/...`. Default value is `443` From 3f3a88aef8c395ee2cd80665c139745c827e11cf Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Tue, 18 Jul 2023 09:42:08 +0200 Subject: [PATCH 006/214] chore: update CHANGELOG.md (#2551) --- CHANGELOG.md | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index ad9b5bdf4c..2422665511 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,11 @@ > The early pre-release track was abandoned due to build incompatibility problems. release info (#2491)) +### v4.0.0-alpha.14 + +- fix(3d-tiles): implicit tiling v1.1 (#2549) +- fix(tile-converter): i3s->3dtiles regression (#2550) + ### v4.0.0-alpha.13 - fix(gltf): 3D tiles extension types & docs (#2542) From 4400d4b91a4fc4b043631078cad5f4d22647f737 Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Wed, 19 Jul 2023 15:30:52 +0200 Subject: [PATCH 007/214] fix(tile-converter): CesiumION tileset URL (#2560) --- modules/tile-converter/src/i3s-converter/i3s-converter.ts | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/modules/tile-converter/src/i3s-converter/i3s-converter.ts b/modules/tile-converter/src/i3s-converter/i3s-converter.ts index 5a5a0f10a1..e972b0e42a 100644 --- a/modules/tile-converter/src/i3s-converter/i3s-converter.ts +++ b/modules/tile-converter/src/i3s-converter/i3s-converter.ts @@ -240,10 +240,14 @@ export default class I3SConverter { try { const preloadOptions = await this._fetchPreloadOptions(); + let tilesetUrl = inputUrl; + if (preloadOptions.url) { + tilesetUrl = preloadOptions.url; + } if (preloadOptions.headers) { this.loadOptions.fetch = {headers: preloadOptions.headers}; } - this.sourceTileset = await load(inputUrl, this.Loader, this.loadOptions); + this.sourceTileset = await load(tilesetUrl, this.Loader, this.loadOptions); const preprocessResult = await this.preprocessConversion(); From 132c45268a6fe985416ba8d39cb28b7ab7fadd2c Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 19 Jul 2023 13:11:41 -0400 Subject: [PATCH 008/214] chore(deps): bump word-wrap in /test/apps/typescript-test (#2559) --- test/apps/typescript-test/yarn.lock | 76 ++++++++++++++--------------- 1 file changed, 38 insertions(+), 38 deletions(-) diff --git a/test/apps/typescript-test/yarn.lock b/test/apps/typescript-test/yarn.lock index 23632837d7..a06c832811 100644 --- a/test/apps/typescript-test/yarn.lock +++ b/test/apps/typescript-test/yarn.lock @@ -43,47 +43,47 @@ resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz#b520529ec21d8e5945a1851dfd1c32e94e39ff45" integrity sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA== -"@loaders.gl/gis@3.3.0-alpha.12": - version "3.3.0-alpha.12" - resolved "https://registry.yarnpkg.com/@loaders.gl/gis/-/gis-3.3.0-alpha.12.tgz#ba6080df9148cefed9e77da8e0753d46053e00d7" - integrity sha512-yNvr90e6b27pmxcIXdwfgEIqv0KSLpKsDu7etkLTgYBeNpQnclf/YHG9tzlP9kPY9HEidm+K10XmSwpUGIknDw== +"@loaders.gl/gis@4.0.0-alpha.14": + version "4.0.0-alpha.14" + resolved "https://registry.yarnpkg.com/@loaders.gl/gis/-/gis-4.0.0-alpha.14.tgz#c9a8e38e6e2b74261dec82026d228604a4e5379a" + integrity sha512-g0X2/7JpCbJCe/xoHedJnYoEsBpKLvaw2jKTd66UA6ZNFCfAy51VmhRCG9A6HhcG25NOd3uunMrsmebSSBsgPg== dependencies: - "@loaders.gl/loader-utils" "3.3.0-alpha.12" - "@loaders.gl/schema" "3.3.0-alpha.12" + "@loaders.gl/loader-utils" "4.0.0-alpha.14" + "@loaders.gl/schema" "4.0.0-alpha.14" "@mapbox/vector-tile" "^1.3.1" "@math.gl/polygon" "^3.5.1" pbf "^3.2.1" -"@loaders.gl/loader-utils@3.3.0-alpha.12": - version "3.3.0-alpha.12" - resolved "https://registry.yarnpkg.com/@loaders.gl/loader-utils/-/loader-utils-3.3.0-alpha.12.tgz#55ce0de14422f64ddf72727716f5ebfa3ddf29f2" - integrity sha512-pBlHaXCbEaAwFe+XbuzoEjwbV4OPLiJ9DS+P3Cpe/ML2R9CT9UUxNDQMJOqM9cW/+hIG+kyUhatP3f67R5SOfQ== +"@loaders.gl/loader-utils@4.0.0-alpha.14": + version "4.0.0-alpha.14" + resolved "https://registry.yarnpkg.com/@loaders.gl/loader-utils/-/loader-utils-4.0.0-alpha.14.tgz#d09d8867c7ef6036e23520e4c1c75690964a58a9" + integrity sha512-T4PCHX5r1ZP3BLyl9VAc9xdtYUgvJzVXdrbchRdCpyeEhkT8FVvOzW2PqYTii8uC7cJkAQZk7lJ13GrMsaMaYA== dependencies: "@babel/runtime" "^7.3.1" - "@loaders.gl/worker-utils" "3.3.0-alpha.12" - "@probe.gl/stats" "^3.5.0" + "@loaders.gl/worker-utils" "4.0.0-alpha.14" + "@probe.gl/stats" "^4.0.2" -"@loaders.gl/schema@3.3.0-alpha.12": - version "3.3.0-alpha.12" - resolved "https://registry.yarnpkg.com/@loaders.gl/schema/-/schema-3.3.0-alpha.12.tgz#bd78d48ae3c7c527bd55afedbc9dc6cb6e0e2afc" - integrity sha512-Kv7Auj2fEHU52kvegjFpK/Hjn83MWni3lVm2EpFEDc+Aw27dDb7U5sH7ow606qq7axiX+lC9JXaEDGYIBALn0Q== +"@loaders.gl/schema@4.0.0-alpha.14": + version "4.0.0-alpha.14" + resolved "https://registry.yarnpkg.com/@loaders.gl/schema/-/schema-4.0.0-alpha.14.tgz#4cc04b834d7a7267cd7748b5a070a40021e813df" + integrity sha512-365OyfUF+Z3gi7XG4y4FYSpsefI+SW+qSMJIVwzE53z0bWNxgncPI5cuGHZPQFd4KYrFg0914z3O/FIj9ISaqA== dependencies: "@types/geojson" "^7946.0.7" -"@loaders.gl/shapefile@3.3.0-alpha.12": - version "3.3.0-alpha.12" - resolved "https://registry.yarnpkg.com/@loaders.gl/shapefile/-/shapefile-3.3.0-alpha.12.tgz#3d5096f74d8c5ebb269d971951a2eb21395792b1" - integrity sha512-2z0OOq+HbTEsH/nIr6Gi5v7fvx5Pdgqyrdk6VohmCgXKsTPupP5q2tX5Wmx8OBA4TfU+Q8XGydUFbLsmBBJS5Q== +"@loaders.gl/shapefile@^4.0.0-alpha.8": + version "4.0.0-alpha.14" + resolved "https://registry.yarnpkg.com/@loaders.gl/shapefile/-/shapefile-4.0.0-alpha.14.tgz#6b9166eb1567c7d2348a6666eb5935e514abd4a0" + integrity sha512-Q2YS6K3lJfdImcYwz3srv+20CcWdX0HOnc9/Qi/jJAPyB9QAGAvAxCufqwE9YVQIET+O40gXYw6S1fSevKDxJQ== dependencies: - "@loaders.gl/gis" "3.3.0-alpha.12" - "@loaders.gl/loader-utils" "3.3.0-alpha.12" - "@loaders.gl/schema" "3.3.0-alpha.12" + "@loaders.gl/gis" "4.0.0-alpha.14" + "@loaders.gl/loader-utils" "4.0.0-alpha.14" + "@loaders.gl/schema" "4.0.0-alpha.14" "@math.gl/proj4" "^3.5.1" -"@loaders.gl/worker-utils@3.3.0-alpha.12": - version "3.3.0-alpha.12" - resolved "https://registry.yarnpkg.com/@loaders.gl/worker-utils/-/worker-utils-3.3.0-alpha.12.tgz#3c5f56be03b1359491e79674c6d5d71b3161e998" - integrity sha512-OOptTemcEvH0K0P/WJi13jXF2eerdcoxIDRVX0Q8sHHahkhD3+oGBwhilFj+oxTy1VQUiE/wienJRqZioQ22dQ== +"@loaders.gl/worker-utils@4.0.0-alpha.14": + version "4.0.0-alpha.14" + resolved "https://registry.yarnpkg.com/@loaders.gl/worker-utils/-/worker-utils-4.0.0-alpha.14.tgz#ac8d6da13f64e0aa2fb089543092cc26fa7307f8" + integrity sha512-ntnJgfPEt4ym9mxXpl1PLOjsGLTlNI/8ivfpzHR0Dp0KvaXjFsL+Vle8g/+2BV2dmbihHhnARygNw9VmYbgsKw== dependencies: "@babel/runtime" "^7.3.1" @@ -145,10 +145,10 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" -"@probe.gl/stats@^3.5.0": - version "3.5.0" - resolved "https://registry.yarnpkg.com/@probe.gl/stats/-/stats-3.5.0.tgz#774495772f06e898aae28c1d315c9edac07f3425" - integrity sha512-IH2M+F3c8HR1DTroBARePUFG7wIewumtKA0UFqx51Z7S4hKrD60wFbpMmg0AcF4FvHAXMBoC+kYi1UKW9XbAOw== +"@probe.gl/stats@^4.0.2": + version "4.0.4" + resolved "https://registry.yarnpkg.com/@probe.gl/stats/-/stats-4.0.4.tgz#b33a47bf192951d0789dfd2044b295c3709386bd" + integrity sha512-SDuSY/D4yDL6LQDa69l/GCcnZLRiGYdyvYkxWb0CgnzTPdPrcdrzGkzkvpC3zsA4fEFw2smlDje370QGHwlisg== dependencies: "@babel/runtime" "^7.0.0" @@ -1607,10 +1607,10 @@ typedarray-to-buffer@^3.1.5: dependencies: is-typedarray "^1.0.0" -typescript@^4.0.3: - version "4.4.4" - resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.4.4.tgz#2cd01a1a1f160704d3101fd5a58ff0f9fcb8030c" - integrity sha512-DqGhF5IKoBl8WNf8C1gu8q0xZSInh9j1kJJMqT3a94w1JzVaBU4EXOSMrz9yDqMT0xt3selp83fuFMQ0uzv6qA== +typescript@^5.0.4: + version "5.1.6" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.1.6.tgz#02f8ac202b6dad2c0dd5e0913745b47a37998274" + integrity sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA== undefsafe@^2.0.3: version "2.0.5" @@ -1683,9 +1683,9 @@ wkt-parser@^1.2.4: integrity sha512-XK5qV+Y5gsygQfHx2/cS5a7Zxsgleaw8iX5UPC5eOXPc0TgJAu1JB9lr0iYYX3zAnN3p0aNiaN5c+1Bdblxwrg== word-wrap@^1.2.3: - version "1.2.3" - resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.3.tgz#610636f6b1f703891bd34771ccb17fb93b47079c" - integrity sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ== + version "1.2.4" + resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.4.tgz#cb4b50ec9aca570abd1f52f33cd45b6c61739a9f" + integrity sha512-2V81OA4ugVo5pRo46hAoD2ivUJx8jXmWXfUkY4KFNw0hEptvN0QfH3K4nHiwzGeKl5rFKedV48QVoqYavy4YpA== wrap-ansi@^7.0.0: version "7.0.0" From 729619bc433b1e34d169b23000f5b661b0c71cd1 Mon Sep 17 00:00:00 2001 From: Nathan Reese Date: Wed, 19 Jul 2023 12:31:46 -0600 Subject: [PATCH 009/214] fix(docs): JSONLoader _rootObjectBatches removed but not mentioned in upgrade guide (#2558) --- docs/whats-new.mdx | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/docs/whats-new.mdx b/docs/whats-new.mdx index 81eb6042a5..55a2a67cd9 100644 --- a/docs/whats-new.mdx +++ b/docs/whats-new.mdx @@ -461,6 +461,12 @@ loaders.gl v3.0 is a major release, that adds a range of new loaders and feature - Supports binary array fields in draco metadata. - Significant performance improvements for loading and decoding. +**@loaders.gl/json** + +- Removed deprecated json parse option `_rootObjectBatches`. Use loader option `metadata` instead. +- Replaced `batch.batchType` `root-object-batch-partial` with `partial-result`. +- Replaced `batch.batchType` `root-object-batch-complete` with `final-result`. + **@loaders.gl/kml** - `KMLLoader` - updated loader, now works under Node.js. From edffd477280a9f71f56bd2a8db0e230d8618e3ad Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 19 Jul 2023 14:32:23 -0400 Subject: [PATCH 010/214] chore(deps): bump semver from 5.7.1 to 5.7.2 in /website (#2546) --- website/yarn.lock | 244 ++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 235 insertions(+), 9 deletions(-) diff --git a/website/yarn.lock b/website/yarn.lock index 875a24c811..c67677ce89 100644 --- a/website/yarn.lock +++ b/website/yarn.lock @@ -2007,6 +2007,25 @@ "@math.gl/geospatial" "^3.5.1" long "^5.2.1" +"@loaders.gl/compression@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/compression/-/compression-4.0.0-alpha.12.tgz#c0f5ac86803d9c9933a0a6fe10d6e13da070790a" + integrity sha512-QRVRfuoCVGhog6CpfEP2R2IBJhQn+gV6nz9+NdqpSDndLmUgsBTO/PGSj2A73qKo3uALVTzLXc007zBX+aGP8A== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/worker-utils" "4.0.0-alpha.12" + "@types/brotli" "^1.3.0" + "@types/pako" "^1.0.1" + fflate "0.7.4" + lzo-wasm "^0.0.4" + pako "1.0.11" + snappyjs "^0.6.1" + optionalDependencies: + brotli "^1.3.2" + lz4js "^0.2.0" + zstd-codec "^0.1" + "@loaders.gl/core@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/core/-/core-3.4.4.tgz#65c182fcf9e7e6536ac7d27d87635a8721843a95" @@ -2017,6 +2036,16 @@ "@loaders.gl/worker-utils" "3.4.4" "@probe.gl/log" "^4.0.1" +"@loaders.gl/core@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/core/-/core-4.0.0-alpha.12.tgz#41439a2fe895a0add40f4e03e5287ac5d305d9ae" + integrity sha512-ovK2pGNKMEGX13agDuZp+KdbNSetaHYus7ilkVOf/eaEIYpo3/Z2Ny0aTpYPPx/icPZFvkB9i5e3OohB5+mojA== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/worker-utils" "4.0.0-alpha.12" + "@probe.gl/log" "^4.0.2" + "@loaders.gl/draco@3.4.4": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/draco/-/draco-3.4.4.tgz#cf2db51cf413a866363475da439382ef6878e579" @@ -2028,6 +2057,17 @@ "@loaders.gl/worker-utils" "3.4.4" draco3d "1.5.5" +"@loaders.gl/draco@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/draco/-/draco-4.0.0-alpha.12.tgz#8a32bff77ab38b34c79539c6b2acadbb8ad1240c" + integrity sha512-woq5PvIxl9Rp4QIPZEIHniSuffbelR9XtjFhvCWkkDrqlAikUSsVuFdkmBVPVqXLEbC0TbsSBpeUXAOzB4jZPw== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" + "@loaders.gl/worker-utils" "4.0.0-alpha.12" + draco3d "1.5.5" + "@loaders.gl/gis@3.4.4", "@loaders.gl/gis@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/gis/-/gis-3.4.4.tgz#74390f4271e26b0bcc83bbe5786947658c3093bd" @@ -2050,6 +2090,24 @@ "@loaders.gl/textures" "3.4.4" "@math.gl/core" "^3.5.1" +"@loaders.gl/i3s@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/i3s/-/i3s-4.0.0-alpha.12.tgz#786565214ca0b97e2325e07714eb063195b9a2eb" + integrity sha512-MejeqP08oRBa8tqvCO/MfRO9QF2ozjs1+7BLrixQT1juhb79bXojxE4lcQ9/8Yzg8oBQ8C1sora7kI0upxN81Q== + dependencies: + "@loaders.gl/compression" "4.0.0-alpha.12" + "@loaders.gl/draco" "4.0.0-alpha.12" + "@loaders.gl/images" "4.0.0-alpha.12" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" + "@loaders.gl/textures" "4.0.0-alpha.12" + "@loaders.gl/tiles" "4.0.0-alpha.12" + "@luma.gl/constants" "^8.5.4" + "@math.gl/core" "^3.5.1" + "@math.gl/culling" "^3.5.1" + "@math.gl/geospatial" "^3.5.1" + md5 "^2.3.0" + "@loaders.gl/images@3.4.4", "@loaders.gl/images@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/images/-/images-3.4.4.tgz#ce0d56caac49ebf63f76a6234f1df4a7b8c99441" @@ -2057,6 +2115,23 @@ dependencies: "@loaders.gl/loader-utils" "3.4.4" +"@loaders.gl/images@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/images/-/images-4.0.0-alpha.12.tgz#ba61b22fc1854c5ef71e95d3e50c53e81521ebd9" + integrity sha512-zdzBt+cPPlSSwV8bGbSyA7vIQpoxK0ysUtIpDSh7EmDGKzEN3S3m2R+S3Jd4NV8VWLTfZXbWpCvl9r/tIB4kgw== + dependencies: + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + +"@loaders.gl/las@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/las/-/las-4.0.0-alpha.12.tgz#34cda83df4f6816ff464a179066a9eaee5c72992" + integrity sha512-vw7dKYpkb/BUL9mF62nkyfHKGSB/QbLazJxC4dMAZk4evLw44AePeQLXsUgx4KnNxiMepnFoz6la0m2VzNoE2Q== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" + apache-arrow "^9.0.0" + "@loaders.gl/loader-utils@3.4.4", "@loaders.gl/loader-utils@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/loader-utils/-/loader-utils-3.4.4.tgz#06a9185e6b9a6457aaaf4f5526dca25c9021ff25" @@ -2066,6 +2141,14 @@ "@loaders.gl/worker-utils" "3.4.4" "@probe.gl/stats" "^4.0.1" +"@loaders.gl/loader-utils@4.0.0-alpha.12", "@loaders.gl/loader-utils@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/loader-utils/-/loader-utils-4.0.0-alpha.12.tgz#a44136be2d5f9c3d1e9ab141ea2554b97d5ca47a" + integrity sha512-1kd8gGgMvrGednJb6ZMp8QDwWtRcbqa8LmCmNl0lyWo85xp5f+Gxo1VzFPjKTUzRAITuqwsndLkEmOeT7hcL9w== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/worker-utils" "4.0.0-alpha.12" + "@probe.gl/stats" "^4.0.2" "@loaders.gl/math@3.4.4": version "3.4.4" @@ -2076,6 +2159,15 @@ "@loaders.gl/loader-utils" "3.4.4" "@math.gl/core" "^3.5.1" +"@loaders.gl/math@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/math/-/math-4.0.0-alpha.12.tgz#985f6c324ab2ca524abbda8369d498180e326165" + integrity sha512-37KID9HC9ZynIQk6DyXnZFEIufyFIg8N0Zn/KYHsk2yNYqcbBt9JWFVH3Aue1WixrvIRWlhNnip+ve32E3ix2Q== + dependencies: + "@loaders.gl/images" "4.0.0-alpha.12" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@math.gl/core" "^3.5.1" + "@loaders.gl/mvt@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/mvt/-/mvt-3.4.4.tgz#7c7559912ed3ef6e995e0baae20cc7c6b090375b" @@ -2087,6 +2179,23 @@ "@math.gl/polygon" "^3.5.1" pbf "^3.2.1" +"@loaders.gl/obj@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/obj/-/obj-4.0.0-alpha.12.tgz#378cd6227be8b9c8b48e75b8889038aedad80ff5" + integrity sha512-Jzfk2vMfODWNUw0welmpMjCS0fQHaEjCknBitItXbzOap8Y8uFLnOkKk480WQ7QkqDHYx2n/EysrJ0F07ouWOg== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" + +"@loaders.gl/ply@^4.0.0-alpha.8": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/ply/-/ply-4.0.0-alpha.12.tgz#770da026ed53e7ffd45f43706dca51bdc519b276" + integrity sha512-EumoRPlzx9GfhC6Xt34/ZbMPmQPMiEU1T+xM0WGNf2AyoeHCuZzPU7hhA+txNb5EZ3HP/fSof5DNFmHEX8SziA== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" "@loaders.gl/schema@3.4.4", "@loaders.gl/schema@^3.4.2": version "3.4.4" @@ -2095,6 +2204,13 @@ dependencies: "@types/geojson" "^7946.0.7" +"@loaders.gl/schema@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/schema/-/schema-4.0.0-alpha.12.tgz#4e1f19f16f84854ef9512b46f861dfae31b0264f" + integrity sha512-d0cWhXRx0KSA06dykNcU1Ct5hfT6qeC7dHbVWqoC3QYQBPxc7VCU+jQfQkCLGyuV9GYEzjj97DNYCA6/GETP0Q== + dependencies: + "@types/geojson" "^7946.0.7" + "@loaders.gl/terrain@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/terrain/-/terrain-3.4.4.tgz#4af56000d7b0722a421282621ad3c625fc7142c4" @@ -2118,6 +2234,18 @@ ktx-parse "^0.0.4" texture-compressor "^1.0.2" +"@loaders.gl/textures@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/textures/-/textures-4.0.0-alpha.12.tgz#8fa0fc728f470eaa5698843fe710b06defee72f8" + integrity sha512-r1QuUIz3vakwx30vhoT5dj9FRDe56ybQeNGsU0cPJ1zPIBRMvophfBPDp7y0xhkp730p6uI5R3Mz61x9AqBnoA== + dependencies: + "@loaders.gl/images" "4.0.0-alpha.12" + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/schema" "4.0.0-alpha.12" + "@loaders.gl/worker-utils" "4.0.0-alpha.12" + ktx-parse "^0.0.4" + texture-compressor "^1.0.2" + "@loaders.gl/tiles@3.4.4", "@loaders.gl/tiles@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/tiles/-/tiles-3.4.4.tgz#8e0597a7cda501e4a1a6a35d6434f27407a487d3" @@ -2131,6 +2259,19 @@ "@math.gl/web-mercator" "^3.5.1" "@probe.gl/stats" "^4.0.1" +"@loaders.gl/tiles@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/tiles/-/tiles-4.0.0-alpha.12.tgz#5e935c55ed4d1794e1d1072ef94ff6f183329100" + integrity sha512-7Z7GsXeJCACwG55FBBTFCrP5N3Fga7krsAjcE5w8sd09F80RTK5Z9o0mMFR16YR2ZzJrPCwDafl2ag9nw4btYA== + dependencies: + "@loaders.gl/loader-utils" "4.0.0-alpha.12" + "@loaders.gl/math" "4.0.0-alpha.12" + "@math.gl/core" "^3.5.1" + "@math.gl/culling" "^3.5.1" + "@math.gl/geospatial" "^3.5.1" + "@math.gl/web-mercator" "^3.5.1" + "@probe.gl/stats" "^4.0.2" + "@loaders.gl/wms@^3.4.2": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/wms/-/wms-3.4.4.tgz#447124ee599d3c45ca668f3aa36776f449b47108" @@ -2152,6 +2293,13 @@ dependencies: "@babel/runtime" "^7.3.1" +"@loaders.gl/worker-utils@4.0.0-alpha.12": + version "4.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/@loaders.gl/worker-utils/-/worker-utils-4.0.0-alpha.12.tgz#6828f0fa768d737c64accc5159ed79cb0cddd499" + integrity sha512-UMOnbKQsuFdPiTPOaXVtZIgAYVtrn5boSSdUW72rV04jFE9iYwCcCRXNSodk+TTsIAd6SFdqaXPRImPRAPKQfQ== + dependencies: + "@babel/runtime" "^7.3.1" + "@loaders.gl/xml@3.4.4": version "3.4.4" resolved "https://registry.yarnpkg.com/@loaders.gl/xml/-/xml-3.4.4.tgz#e15b3bc25c2f60776ab2557752cb0981b98725ec" @@ -3958,6 +4106,13 @@ dependencies: "@types/node" "*" +"@types/brotli@^1.3.0": + version "1.3.1" + resolved "https://registry.yarnpkg.com/@types/brotli/-/brotli-1.3.1.tgz#65dc6c69bb9f4159677032f60e81ffc09faf1fce" + integrity sha512-mGwX0BBQqmpHoX8+b8Oez0X+ZEYnl2gbDL2n0HxYT4imqhTChhj1AAgAKVWNZSuPvXGZXqVoOtBS0071tN6Tkw== + dependencies: + "@types/node" "*" + "@types/color-convert@*": version "2.0.0" resolved "https://registry.yarnpkg.com/@types/color-convert/-/color-convert-2.0.0.tgz#8f5ee6b9e863dcbee5703f5a517ffb13d3ea4e22" @@ -4163,6 +4318,11 @@ resolved "https://registry.yarnpkg.com/@types/pad-left/-/pad-left-2.1.1.tgz#17d906fc75804e1cc722da73623f1d978f16a137" integrity sha512-Xd22WCRBydkGSApl5Bw0PhAOHKSVjNL3E3AwzKaps96IMraPqy5BvZIsBVK6JLwdybUzjHnuWVwpDd0JjTfHXA== +"@types/pako@^1.0.1": + version "1.0.4" + resolved "https://registry.yarnpkg.com/@types/pako/-/pako-1.0.4.tgz#b4262aef92680a9331fcdb8420c69cf3dd98d3f3" + integrity sha512-Z+5bJSm28EXBSUJEgx29ioWeEEHUh6TiMkZHDhLwjc9wVFH+ressbkmX6waUZc5R3Gobn4Qu5llGxaoflZ+yhA== + "@types/parse-json@^4.0.0": version "4.0.0" resolved "https://registry.yarnpkg.com/@types/parse-json/-/parse-json-4.0.0.tgz#2f8bb441434d163b35fb8ffdccd7138927ffb8c0" @@ -4772,6 +4932,11 @@ base16@^1.0.0: resolved "https://registry.yarnpkg.com/base16/-/base16-1.0.0.tgz#e297f60d7ec1014a7a971a39ebc8a98c0b681e70" integrity sha512-pNdYkNPiJUnEhnfXV56+sQy8+AaPcG3POZAUnwr4EeqCUZFz4u2PePbo3e5Gj4ziYPCWGUZT9RHisvJKnwFuBQ== +base64-js@^1.1.2: + version "1.5.1" + resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" + integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== + batch@0.6.1: version "0.6.1" resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16" @@ -4863,6 +5028,13 @@ braces@^3.0.2, braces@~3.0.2: dependencies: fill-range "^7.0.1" +brotli@^1.3.2: + version "1.3.3" + resolved "https://registry.yarnpkg.com/brotli/-/brotli-1.3.3.tgz#7365d8cc00f12cf765d2b2c898716bcf4b604d48" + integrity sha512-oTKjJdShmDuGW94SyyaoQvAjf30dZaHnjJ8uAF+u2/vGJkJbJPJAT1gDiOJP5v1Zb6f9KEyW/1HpuaWIXtGHPg== + dependencies: + base64-js "^1.1.2" + browserslist@^4.0.0, browserslist@^4.14.5, browserslist@^4.18.1, browserslist@^4.21.3, browserslist@^4.21.4, browserslist@^4.21.5: version "4.21.5" resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.21.5.tgz#75c5dae60063ee641f977e00edd3cfb2fb7af6a7" @@ -4999,6 +5171,11 @@ character-reference-invalid@^1.0.0: resolved "https://registry.yarnpkg.com/character-reference-invalid/-/character-reference-invalid-1.1.4.tgz#083329cda0eae272ab3dbbf37e9a382c13af1560" integrity sha512-mKKUkUbhPpQlCOfIuZkvSEgktjPFIsZKRRbC6KWVEMvlzblj3i3asQv5ODsrwt0N3pHAEvjP8KTQPHkp0+6jOg== +charenc@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/charenc/-/charenc-0.0.2.tgz#c0a1d2f3a7092e03774bfa83f14c0fc5790a8667" + integrity sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA== + cheerio-select@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/cheerio-select/-/cheerio-select-2.1.0.tgz#4d8673286b8126ca2a8e42740d5e3c4884ae21b4" @@ -5415,6 +5592,11 @@ cross-spawn@^7.0.3: shebang-command "^2.0.0" which "^2.0.1" +crypt@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/crypt/-/crypt-0.0.2.tgz#88d7ff7ec0dfb86f713dc87bbb42d044d3e6c41b" + integrity sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow== + crypto-random-string@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-2.0.0.tgz#ef2a7a966ec11083388369baa02ebead229b30d5" @@ -6304,6 +6486,11 @@ feed@^4.2.2: dependencies: xml-js "^1.6.11" +fflate@0.7.4: + version "0.7.4" + resolved "https://registry.yarnpkg.com/fflate/-/fflate-0.7.4.tgz#61587e5d958fdabb5a9368a302c25363f4f69f50" + integrity sha512-5u2V/CDW15QM1XbbgS+0DfPxVB+jUKhWEKuuFuHncbk3tEEqzmoXL+2KyOFuKGqOnmdIy0/davWF1CkuwtibCw== + file-loader@^6.2.0: version "6.2.0" resolved "https://registry.yarnpkg.com/file-loader/-/file-loader-6.2.0.tgz#baef7cf8e1840df325e4390b4484879480eebe4d" @@ -7146,6 +7333,11 @@ is-buffer@^2.0.0: resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-2.0.5.tgz#ebc252e400d22ff8d77fa09888821a24a658c191" integrity sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ== +is-buffer@~1.1.6: + version "1.1.6" + resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" + integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== + is-ci@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-2.0.0.tgz#6bc6334181810e04b5c22b3d589fdca55026404c" @@ -7742,6 +7934,16 @@ luxon@~3.2.1: resolved "https://registry.yarnpkg.com/luxon/-/luxon-3.2.1.tgz#14f1af209188ad61212578ea7e3d518d18cee45f" integrity sha512-QrwPArQCNLAKGO/C+ZIilgIuDnEnKx5QYODdDtbFaxzsbZcc/a7WFq7MhsVYgRlwawLtvOUESTlfJ+hc/USqPg== +lz4js@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/lz4js/-/lz4js-0.2.0.tgz#09f1a397cb2158f675146c3351dde85058cb322f" + integrity sha512-gY2Ia9Lm7Ep8qMiuGRhvUq0Q7qUereeldZPP1PMEJxPtEWHJLqw9pgX68oHajBH0nzJK4MaZEA/YNV3jT8u8Bg== + +lzo-wasm@^0.0.4: + version "0.0.4" + resolved "https://registry.yarnpkg.com/lzo-wasm/-/lzo-wasm-0.0.4.tgz#49152521a0b67a1da3a3e113dd38f150f303230a" + integrity sha512-VKlnoJRFrB8SdJhlVKvW5vI1gGwcZ+mvChEXcSX6r2xDNc/Q2FD9esfBmGCuPZdrJ1feO+YcVFd2PTk0c137Gw== + make-dir@^3.0.0, make-dir@^3.0.2, make-dir@^3.1.0: version "3.1.0" resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz#415e967046b3a7f1d185277d84aa58203726a13f" @@ -7816,6 +8018,15 @@ math.gl@^3.6.2: dependencies: "@math.gl/core" "3.6.3" +md5@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/md5/-/md5-2.3.0.tgz#c3da9a6aae3a30b46b7b0c349b87b110dc3bda4f" + integrity sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g== + dependencies: + charenc "0.0.2" + crypt "0.0.2" + is-buffer "~1.1.6" + mdast-squeeze-paragraphs@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/mdast-squeeze-paragraphs/-/mdast-squeeze-paragraphs-4.0.0.tgz#7c4c114679c3bee27ef10b58e2e015be79f1ef97" @@ -8244,6 +8455,11 @@ pad-left@^2.1.0: dependencies: repeat-string "^1.5.4" +pako@1.0.11: + version "1.0.11" + resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf" + integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw== + param-case@^3.0.4: version "3.0.4" resolved "https://registry.yarnpkg.com/param-case/-/param-case-3.0.4.tgz#7d17fe4aa12bde34d4a77d91acfb6219caad01c5" @@ -9558,19 +9774,19 @@ semver-diff@^3.1.1: semver "^6.3.0" semver@^5.4.1: - version "5.7.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" - integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== + version "5.7.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" + integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== semver@^6.0.0, semver@^6.1.1, semver@^6.1.2, semver@^6.2.0, semver@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" - integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== semver@^7.3.2, semver@^7.3.4, semver@^7.3.7, semver@^7.3.8: - version "7.5.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.0.tgz#ed8c5dc8efb6c629c88b23d41dc9bf40c1d96cd0" - integrity sha512-+XC0AD/R7Q2mPSRuy2Id0+CGTZ98+8f+KvwirxOKIEyid+XSx6HbC63p+O4IndTHuX5Z+JxQ0TghCkO5Cg/2HA== + version "7.5.4" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" + integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA== dependencies: lru-cache "^6.0.0" @@ -9750,6 +9966,11 @@ slash@^4.0.0: resolved "https://registry.yarnpkg.com/slash/-/slash-4.0.0.tgz#2422372176c4c6c5addb5e2ada885af984b396a7" integrity sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew== +snappyjs@^0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/snappyjs/-/snappyjs-0.6.1.tgz#9bca9ff8c54b133a9cc84a71d22779e97fc51878" + integrity sha512-YIK6I2lsH072UE0aOFxxY1dPDCS43I5ktqHpeAsuLNYWkE5pGxRGWfDM4/vSUfNzXjC1Ivzt3qx31PCLmc9yqg== + sockjs@^0.3.24: version "0.3.24" resolved "https://registry.yarnpkg.com/sockjs/-/sockjs-0.3.24.tgz#c9bc8995f33a111bea0395ec30aa3206bdb5ccce" @@ -10832,6 +11053,11 @@ yocto-queue@^0.1.0: resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== +zstd-codec@^0.1: + version "0.1.4" + resolved "https://registry.yarnpkg.com/zstd-codec/-/zstd-codec-0.1.4.tgz#6abb311b63cfacbd06e72797ee6c6e1c7c65248c" + integrity sha512-KYnWoFWgGtWyQEKNnUcb3u8ZtKO8dn5d8u+oGpxPlopqsPyv60U8suDyfk7Z7UtAO6Sk5i1aVcAs9RbaB1n36A== + zwitch@^1.0.0: version "1.0.5" resolved "https://registry.yarnpkg.com/zwitch/-/zwitch-1.0.5.tgz#d11d7381ffed16b742f6af7b3f223d5cd9fe9920" From 48e473d7ef9eefdcb2bbc63364999a47d639ee5b Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Thu, 20 Jul 2023 14:17:28 +0200 Subject: [PATCH 011/214] feat(tile-converter): i3s-server bundle (#2555) --- modules/tile-converter/bin/i3s-server.js | 4 ++++ modules/tile-converter/package.json | 7 ++++--- .../src/i3s-server/bin/{www => www.js} | 14 +++++++------- 3 files changed, 15 insertions(+), 10 deletions(-) create mode 100755 modules/tile-converter/bin/i3s-server.js rename modules/tile-converter/src/i3s-server/bin/{www => www.js} (82%) diff --git a/modules/tile-converter/bin/i3s-server.js b/modules/tile-converter/bin/i3s-server.js new file mode 100755 index 0000000000..8e1be9221a --- /dev/null +++ b/modules/tile-converter/bin/i3s-server.js @@ -0,0 +1,4 @@ +#!/usr/bin/env node +// @ts-nocheck +/* eslint-disable import/no-extraneous-dependencies */ +require('@loaders.gl/tile-converter/dist/esm/i3s-server/bin/i3s-server.min'); diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index aa9ad48f5e..53b2160754 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -22,7 +22,7 @@ "bin": { "tile-converter": "./bin/converter.js", "slpk-extractor": "./bin/slpk-extractor.js", - "i3s-server": "./src/i3s-server/bin/www" + "i3s-server": "./bin/i3s-server.js" }, "files": [ "src", @@ -38,10 +38,11 @@ "join-images": false }, "scripts": { - "pre-build": "npm run build-bundle && npm run build-converter-bundle && npm run build-slpk-extractor-bundle", + "pre-build": "npm run build-bundle && npm run build-converter-bundle && npm run build-slpk-extractor-bundle && npm run build-i3s-server-bundle", "build-bundle": "esbuild ./src/index.ts --bundle --outfile=dist/dist.min.js --platform=node --external:join-images", "build-converter-bundle": "esbuild src/converter-cli.ts --outfile=dist/converter.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", - "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" + "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", + "build-i3s-server-bundle": "esbuild src/i3s-server/bin/www.js --outfile=dist/esm/i3s-server/bin/i3s-server.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { "@loaders.gl/3d-tiles": "4.0.0-alpha.14", diff --git a/modules/tile-converter/src/i3s-server/bin/www b/modules/tile-converter/src/i3s-server/bin/www.js similarity index 82% rename from modules/tile-converter/src/i3s-server/bin/www rename to modules/tile-converter/src/i3s-server/bin/www.js index 8b09bdd2a0..40bd66bc39 100755 --- a/modules/tile-converter/src/i3s-server/bin/www +++ b/modules/tile-converter/src/i3s-server/bin/www.js @@ -72,22 +72,22 @@ function formErrorHandler(optionalPort) { throw error; } - const bind = typeof port === 'string' ? `Pipe ${optionalPort}`: `Port ${optionalPort}`; + const bind = typeof global.port === 'string' ? `Pipe ${optionalPort}` : `Port ${optionalPort}`; // handle specific listen errors with friendly messages switch (error.code) { case 'EACCES': - console.error(`${bind} requires elevated privileges`);// eslint-disable-line no-console, no-undef - process.exit(1);// eslint-disable-line no-process-exit, no-undef + console.error(`${bind} requires elevated privileges`); // eslint-disable-line no-console, no-undef + process.exit(1); // eslint-disable-line no-process-exit, no-undef break; case 'EADDRINUSE': - console.error(`${bind} is already in use`);// eslint-disable-line no-console, no-undef - process.exit(1);// eslint-disable-line no-process-exit, no-undef + console.error(`${bind} is already in use`); // eslint-disable-line no-console, no-undef + process.exit(1); // eslint-disable-line no-process-exit, no-undef break; default: throw error; } - } + }; } /** @@ -98,5 +98,5 @@ function formListeningHandler(optionalServer) { const addr = optionalServer.address(); const bind = typeof addr === 'string' ? `pipe ${addr}` : `port ${addr.port}`; debug(`Listening on ${bind}`); - } + }; } From 71e48c3826c2d1db12b30616836e55565af55aea Mon Sep 17 00:00:00 2001 From: dariaterekhovaae <98411986+dariaterekhovaae@users.noreply.github.com> Date: Thu, 20 Jul 2023 16:58:51 +0300 Subject: [PATCH 012/214] chore(tile-converter): Support for SLPKs larger than 2 Gb (#2547) --- modules/i3s/src/i3s-slpk-loader.ts | 20 +++++- modules/i3s/src/index.ts | 3 +- .../src/lib/parsers/parse-slpk/parse-slpk.ts | 51 +++----------- .../parsers/parse-slpk/search-from-the-end.ts | 38 ++++++++++ .../lib/parsers/parse-slpk/slpk-archieve.ts | 21 +++--- .../parsers/parse-zip/buffer-file-provider.ts | 55 --------------- .../lib/parsers/parse-zip/cd-file-header.ts | 60 ++++++++-------- .../parse-zip/data-view-file-provider.ts | 69 +++++++++++++++++++ .../lib/parsers/parse-zip/file-provider.ts | 16 +++-- .../parsers/parse-zip/local-file-header.ts | 60 +++++++++++----- modules/i3s/test/index.js | 4 ++ modules/i3s/test/local-file-header.spec.js | 14 ---- modules/i3s/test/search-from-the-end.spec.js | 15 ++++ .../{ => zip-utils}/cd-file-header.spec.js | 12 ++-- .../test/zip-utils/data-view-provider.spec.ts | 34 +++++++++ .../test/zip-utils/local-file-header.spec.js | 14 ++++ modules/tile-converter/src/i3s-server/app.js | 2 +- .../i3s-server/controllers/slpk-controller.js | 15 ++-- modules/tile-converter/src/index.ts | 1 + .../helpers/file-handle-provider.ts | 54 ++++++++++----- .../src/slpk-extractor/helpers/fs-promises.ts | 66 ++++++++++++++++++ .../src/slpk-extractor/slpk-extractor.ts | 11 ++- modules/tile-converter/test/index.js | 2 + .../file-handle-provider.spec.js | 47 +++++++++++++ .../test/slpk-extractor/file-handle.spec.js | 20 ++++++ 25 files changed, 488 insertions(+), 216 deletions(-) create mode 100644 modules/i3s/src/lib/parsers/parse-slpk/search-from-the-end.ts delete mode 100644 modules/i3s/src/lib/parsers/parse-zip/buffer-file-provider.ts create mode 100644 modules/i3s/src/lib/parsers/parse-zip/data-view-file-provider.ts delete mode 100644 modules/i3s/test/local-file-header.spec.js create mode 100644 modules/i3s/test/search-from-the-end.spec.js rename modules/i3s/test/{ => zip-utils}/cd-file-header.spec.js (50%) create mode 100644 modules/i3s/test/zip-utils/data-view-provider.spec.ts create mode 100644 modules/i3s/test/zip-utils/local-file-header.spec.js create mode 100644 modules/tile-converter/src/slpk-extractor/helpers/fs-promises.ts create mode 100644 modules/tile-converter/test/slpk-extractor/file-handle-provider.spec.js create mode 100644 modules/tile-converter/test/slpk-extractor/file-handle.spec.js diff --git a/modules/i3s/src/i3s-slpk-loader.ts b/modules/i3s/src/i3s-slpk-loader.ts index 63216c431f..e2bdf39865 100644 --- a/modules/i3s/src/i3s-slpk-loader.ts +++ b/modules/i3s/src/i3s-slpk-loader.ts @@ -1,13 +1,17 @@ import {LoaderOptions, LoaderWithParser} from '@loaders.gl/loader-utils'; -import {parseSLPK} from './lib/parsers/parse-slpk/parse-slpk'; +import {parseSLPK as parseSLPKFromProvider} from './lib/parsers/parse-slpk/parse-slpk'; +import {DataViewFileProvider} from './lib/parsers/parse-zip/data-view-file-provider'; // __VERSION__ is injected by babel-plugin-version-inline // @ts-ignore TS2304: Cannot find name '__VERSION__'. const VERSION = typeof __VERSION__ !== 'undefined' ? __VERSION__ : 'latest'; +/** options to load data from SLPK */ export type SLPKLoaderOptions = LoaderOptions & { slpk?: { + /** path inside the slpk archive */ path?: string; + /** mode of the path */ pathMode?: 'http' | 'raw'; }; }; @@ -25,3 +29,17 @@ export const SLPKLoader: LoaderWithParser = { extensions: ['slpk'], options: {} }; + +/** + * returns a single file from the slpk archive + * @param data slpk archive data + * @param options options + * @returns requested file + */ + +async function parseSLPK(data: ArrayBuffer, options: SLPKLoaderOptions = {}) { + return (await parseSLPKFromProvider(new DataViewFileProvider(new DataView(data)))).getFile( + options.slpk?.path ?? '', + options.slpk?.pathMode + ); +} diff --git a/modules/i3s/src/index.ts b/modules/i3s/src/index.ts index c50f71ed3f..78ea82b813 100644 --- a/modules/i3s/src/index.ts +++ b/modules/i3s/src/index.ts @@ -33,6 +33,7 @@ export type { OperationalLayer, TextureSetDefinitionFormats } from './types'; +export type {FileProvider} from './lib/parsers/parse-zip/file-provider'; export {COORDINATE_SYSTEM} from './lib/parsers/constants'; @@ -44,4 +45,4 @@ export {I3SBuildingSceneLayerLoader} from './i3s-building-scene-layer-loader'; export {I3SNodePageLoader} from './i3s-node-page-loader'; export {ArcGisWebSceneLoader} from './arcgis-webscene-loader'; export {parseZipLocalFileHeader} from './lib/parsers/parse-zip/local-file-header'; -export {FileProvider} from './lib/parsers/parse-zip/file-provider'; +export {parseSLPK} from './lib/parsers/parse-slpk/parse-slpk'; diff --git a/modules/i3s/src/lib/parsers/parse-slpk/parse-slpk.ts b/modules/i3s/src/lib/parsers/parse-slpk/parse-slpk.ts index 4ef1aa7cf6..e99ee17705 100644 --- a/modules/i3s/src/lib/parsers/parse-slpk/parse-slpk.ts +++ b/modules/i3s/src/lib/parsers/parse-slpk/parse-slpk.ts @@ -1,45 +1,19 @@ -import type {SLPKLoaderOptions} from '../../../i3s-slpk-loader'; -import {DataViewFileProvider} from '../parse-zip/buffer-file-provider'; import {parseZipCDFileHeader} from '../parse-zip/cd-file-header'; +import {FileProvider} from '../parse-zip/file-provider'; import {parseZipLocalFileHeader} from '../parse-zip/local-file-header'; +import {ZipSignature, searchFromTheEnd} from './search-from-the-end'; import {SLPKArchive} from './slpk-archieve'; /** - * Returns one byte from the provided buffer at the provided position - * @param offset - position where to read - * @param buffer - buffer to read - * @returns one byte from the provided buffer at the provided position + * Creates slpk file handler from raw file + * @param fileProvider raw file data + * @returns slpk file handler */ -const getByteAt = (offset: number, buffer: DataView): number => { - return buffer.getUint8(buffer.byteOffset + offset); -}; - -export async function parseSLPK(data: ArrayBuffer, options: SLPKLoaderOptions = {}) { - const archive = new DataView(data); - const cdFileHeaderSignature = [80, 75, 1, 2]; - const searchWindow = [ - getByteAt(archive.byteLength - 1, archive), - getByteAt(archive.byteLength - 2, archive), - getByteAt(archive.byteLength - 3, archive), - undefined - ]; +export const parseSLPK = async (fileProvider: FileProvider): Promise => { + const cdFileHeaderSignature: ZipSignature = [0x50, 0x4b, 0x01, 0x02]; - let hashCDOffset = 0; - - // looking for the last record in the central directory - for (let i = archive.byteLength - 4; i > -1; i--) { - searchWindow[3] = searchWindow[2]; - searchWindow[2] = searchWindow[1]; - searchWindow[1] = searchWindow[0]; - searchWindow[0] = getByteAt(i, archive); - if (searchWindow.every((val, index) => val === cdFileHeaderSignature[index])) { - hashCDOffset = i; - break; - } - } - - const fileProvider = new DataViewFileProvider(archive); + const hashCDOffset = await searchFromTheEnd(fileProvider, cdFileHeaderSignature); const cdFileHeader = await parseZipCDFileHeader(hashCDOffset, fileProvider); @@ -56,7 +30,7 @@ export async function parseSLPK(data: ArrayBuffer, options: SLPKLoaderOptions = } const fileDataOffset = localFileHeader.fileDataOffset; - const hashFile = archive.buffer.slice( + const hashFile = await fileProvider.slice( fileDataOffset, fileDataOffset + localFileHeader.compressedSize ); @@ -65,8 +39,5 @@ export async function parseSLPK(data: ArrayBuffer, options: SLPKLoaderOptions = throw new Error('No hash file in slpk'); } - return await new SLPKArchive(data, hashFile).getFile( - options.slpk?.path ?? '', - options.slpk?.pathMode - ); -} + return new SLPKArchive(fileProvider, hashFile); +}; diff --git a/modules/i3s/src/lib/parsers/parse-slpk/search-from-the-end.ts b/modules/i3s/src/lib/parsers/parse-slpk/search-from-the-end.ts new file mode 100644 index 0000000000..55ba569159 --- /dev/null +++ b/modules/i3s/src/lib/parsers/parse-slpk/search-from-the-end.ts @@ -0,0 +1,38 @@ +import {FileProvider} from 'modules/i3s/src/lib/parsers/parse-zip/file-provider'; + +/** Description of zip signature type */ +export type ZipSignature = [number, number, number, number]; + +/** + * looking for the last occurrence of the provided + * @param file + * @param target + * @returns + */ +export const searchFromTheEnd = async ( + file: FileProvider, + target: ZipSignature +): Promise => { + const searchWindow = [ + await file.getUint8(file.length - 1n), + await file.getUint8(file.length - 2n), + await file.getUint8(file.length - 3n), + undefined + ]; + + let targetOffset = 0n; + + // looking for the last record in the central directory + for (let i = file.length - 4n; i > -1; i--) { + searchWindow[3] = searchWindow[2]; + searchWindow[2] = searchWindow[1]; + searchWindow[1] = searchWindow[0]; + searchWindow[0] = await file.getUint8(i); + if (searchWindow.every((val, index) => val === target[index])) { + targetOffset = i; + break; + } + } + + return targetOffset; +}; diff --git a/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts b/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts index f689dccda2..e588e8dd09 100644 --- a/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts +++ b/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts @@ -1,7 +1,7 @@ import md5 from 'md5'; import {parseZipLocalFileHeader} from '../parse-zip/local-file-header'; -import {DataViewFileProvider} from '../parse-zip/buffer-file-provider'; import {GZipCompression} from '@loaders.gl/compression'; +import {FileProvider} from '../parse-zip/file-provider'; /** Element of hash array */ type HashElement = { @@ -12,7 +12,7 @@ type HashElement = { /** * File offset in the archive */ - offset: number; + offset: bigint; }; /** Description of real paths for different file types */ @@ -55,10 +55,10 @@ const PATH_DESCRIPTIONS: {test: RegExp; extensions: string[]}[] = [ * Class for handling information about slpk file */ export class SLPKArchive { - slpkArchive: DataView; - hashArray: {hash: Buffer; offset: number}[]; - constructor(slpkArchiveBuffer: ArrayBuffer, hashFile: ArrayBuffer) { - this.slpkArchive = new DataView(slpkArchiveBuffer); + slpkArchive: FileProvider; + hashArray: {hash: Buffer; offset: bigint}[]; + constructor(slpkArchive: FileProvider, hashFile: ArrayBuffer) { + this.slpkArchive = slpkArchive; this.hashArray = this.parseHashFile(hashFile); } @@ -77,7 +77,7 @@ export class SLPKArchive { hashFileBuffer.byteOffset + i + 24 ) ); - const offset = offsetBuffer.getUint32(offsetBuffer.byteOffset, true); + const offset = offsetBuffer.getBigUint64(offsetBuffer.byteOffset, true); hashArray.push({ hash: Buffer.from( hashFileBuffer.subarray(hashFileBuffer.byteOffset + i, hashFileBuffer.byteOffset + i + 16) @@ -155,15 +155,12 @@ export class SLPKArchive { return undefined; } - const localFileHeader = await parseZipLocalFileHeader( - this.slpkArchive.byteOffset + fileInfo?.offset, - new DataViewFileProvider(this.slpkArchive) - ); + const localFileHeader = await parseZipLocalFileHeader(fileInfo?.offset, this.slpkArchive); if (!localFileHeader) { return undefined; } - const compressedFile = this.slpkArchive.buffer.slice( + const compressedFile = this.slpkArchive.slice( localFileHeader.fileDataOffset, localFileHeader.fileDataOffset + localFileHeader.compressedSize ); diff --git a/modules/i3s/src/lib/parsers/parse-zip/buffer-file-provider.ts b/modules/i3s/src/lib/parsers/parse-zip/buffer-file-provider.ts deleted file mode 100644 index 4f1d048c85..0000000000 --- a/modules/i3s/src/lib/parsers/parse-zip/buffer-file-provider.ts +++ /dev/null @@ -1,55 +0,0 @@ -import {FileProvider} from './file-provider'; - -/** - * Provides file data using DataView - */ -export class DataViewFileProvider implements FileProvider { - /** - * The DataView from which data is provided - */ - private file: DataView; - - constructor(file: DataView) { - this.file = file; - } - - /** - * Gets an unsigned 8-bit integer at the specified byte offset from the start of the file. - * @param offset The offset, in bytes, from the start of the file where to read the data. - */ - getUint8(offset: number): Promise { - return Promise.resolve(this.file.getUint8(offset)); - } - - /** - * Gets an unsigned 16-bit integer at the specified byte offset from the start of the file. - * @param offset The offset, in bytes, from the start of the file where to read the data. - */ - getUint16(offset: number): Promise { - return Promise.resolve(this.file.getUint16(offset, true)); - } - - /** - * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. - * @param offset The offset, in bytes, from the start of the file where to read the data. - */ - getUint32(offset: number): Promise { - return Promise.resolve(this.file.getUint32(offset, true)); - } - - /** - * returns an ArrayBuffer whose contents are a copy of this file bytes from startOffset, inclusive, up to endOffset, exclusive. - * @param startOffset The offset, in bytes, from the start of the file where to start reading the data. - * @param endOffset The offset, in bytes, from the start of the file where to end reading the data. - */ - slice(startOffset: number, endOffset: number): Promise { - return Promise.resolve(this.file.buffer.slice(startOffset, endOffset)); - } - - /** - * the length (in bytes) of the data. - */ - get length() { - return this.file.byteLength; - } -} diff --git a/modules/i3s/src/lib/parsers/parse-zip/cd-file-header.ts b/modules/i3s/src/lib/parsers/parse-zip/cd-file-header.ts index ba9f4083a1..b5febad67a 100644 --- a/modules/i3s/src/lib/parsers/parse-zip/cd-file-header.ts +++ b/modules/i3s/src/lib/parsers/parse-zip/cd-file-header.ts @@ -6,17 +6,17 @@ import {FileProvider} from './file-provider'; */ export type ZipCDFileHeader = { /** Compressed size */ - compressedSize: number; + compressedSize: bigint; /** Uncompressed size */ - uncompressedSize: number; + uncompressedSize: bigint; /** File name length */ fileNameLength: number; /** File name */ fileName: string; /** Extra field offset */ - extraOffset: number; + extraOffset: bigint; /** Relative offset of local file header */ - localHeaderOffset: number; + localHeaderOffset: bigint; }; /** @@ -26,22 +26,24 @@ export type ZipCDFileHeader = { * @returns Info from the header */ export const parseZipCDFileHeader = async ( - headerOffset: number, + headerOffset: bigint, buffer: FileProvider ): Promise => { const offsets = { - CD_COMPRESSED_SIZE_OFFSET: 20, - CD_UNCOMPRESSED_SIZE_OFFSET: 24, - CD_FILE_NAME_LENGTH_OFFSET: 28, - CD_EXTRA_FIELD_LENGTH_OFFSET: 30, - CD_LOCAL_HEADER_OFFSET_OFFSET: 42, - CD_FILE_NAME_OFFSET: 46 + CD_COMPRESSED_SIZE_OFFSET: 20n, + CD_UNCOMPRESSED_SIZE_OFFSET: 24n, + CD_FILE_NAME_LENGTH_OFFSET: 28n, + CD_EXTRA_FIELD_LENGTH_OFFSET: 30n, + CD_LOCAL_HEADER_OFFSET_OFFSET: 42n, + CD_FILE_NAME_OFFSET: 46n }; - const compressedSize = await buffer.getUint32(headerOffset + offsets.CD_COMPRESSED_SIZE_OFFSET); + let compressedSize = BigInt( + await buffer.getUint32(headerOffset + offsets.CD_COMPRESSED_SIZE_OFFSET) + ); - const uncompressedSize = await buffer.getUint32( - headerOffset + offsets.CD_UNCOMPRESSED_SIZE_OFFSET + let uncompressedSize = BigInt( + await buffer.getUint32(headerOffset + offsets.CD_UNCOMPRESSED_SIZE_OFFSET) ); const fileNameLength = await buffer.getUint16(headerOffset + offsets.CD_FILE_NAME_LENGTH_OFFSET); @@ -49,29 +51,29 @@ export const parseZipCDFileHeader = async ( const fileName = new TextDecoder().decode( await buffer.slice( headerOffset + offsets.CD_FILE_NAME_OFFSET, - headerOffset + offsets.CD_FILE_NAME_OFFSET + fileNameLength + headerOffset + offsets.CD_FILE_NAME_OFFSET + BigInt(fileNameLength) ) ); - const extraOffset = headerOffset + offsets.CD_FILE_NAME_OFFSET + fileNameLength; + const extraOffset = headerOffset + offsets.CD_FILE_NAME_OFFSET + BigInt(fileNameLength); const oldFormatOffset = await buffer.getUint32( headerOffset + offsets.CD_LOCAL_HEADER_OFFSET_OFFSET ); - let fileDataOffset = oldFormatOffset; - if (fileDataOffset === 0xffffffff) { - let offsetInZip64Data = 4; - // looking for info that might be also be in zip64 extra field - if (compressedSize === 0xffffffff) { - offsetInZip64Data += 8; - } - if (uncompressedSize === 0xffffffff) { - offsetInZip64Data += 8; - } - - // getUint32 needs to be replaced with getBigUint64 for archieves bigger than 2gb - fileDataOffset = await buffer.getUint32(extraOffset + offsetInZip64Data); // setting it to the one from zip64 + let fileDataOffset = BigInt(oldFormatOffset); + let offsetInZip64Data = 4n; + // looking for info that might be also be in zip64 extra field + if (uncompressedSize === BigInt(0xffffffff)) { + uncompressedSize = await buffer.getBigUint64(extraOffset + offsetInZip64Data); + offsetInZip64Data += 8n; + } + if (compressedSize === BigInt(0xffffffff)) { + compressedSize = await buffer.getBigUint64(extraOffset + offsetInZip64Data); + offsetInZip64Data += 8n; + } + if (fileDataOffset === BigInt(0xffffffff)) { + fileDataOffset = await buffer.getBigUint64(extraOffset + offsetInZip64Data); // setting it to the one from zip64 } const localHeaderOffset = fileDataOffset; diff --git a/modules/i3s/src/lib/parsers/parse-zip/data-view-file-provider.ts b/modules/i3s/src/lib/parsers/parse-zip/data-view-file-provider.ts new file mode 100644 index 0000000000..c0a18508f1 --- /dev/null +++ b/modules/i3s/src/lib/parsers/parse-zip/data-view-file-provider.ts @@ -0,0 +1,69 @@ +import {FileProvider} from './file-provider'; + +/** + * Checks if bigint can be converted to number and convert it if possible + * @param bigint bigint to be converted + * @returns number + */ +const toNumber = (bigint: bigint) => { + if (bigint > Number.MAX_SAFE_INTEGER) { + throw new Error('Offset is out of bounds'); + } + return Number(bigint); +}; + +/** Provides file data using DataView */ +export class DataViewFileProvider implements FileProvider { + /** The DataView from which data is provided */ + private file: DataView; + + constructor(file: DataView) { + this.file = file; + } + + /** + * Gets an unsigned 8-bit integer at the specified byte offset from the start of the file. + * @param offset The offset, in bytes, from the start of the file where to read the data. + */ + async getUint8(offset: bigint): Promise { + return this.file.getUint8(toNumber(offset)); + } + + /** + * Gets an unsigned 16-bit intege at the specified byte offset from the start of the file. + * @param offset The offset, in bytes, from the start of the file where to read the data. + */ + async getUint16(offset: bigint): Promise { + return this.file.getUint16(toNumber(offset), true); + } + + /** + * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. + * @param offset The offset, in bytes, from the start of the file where to read the data. + */ + async getUint32(offset: bigint): Promise { + return this.file.getUint32(toNumber(offset), true); + } + + /** + * Gets an unsigned 64-bit integer at the specified byte offset from the start of the file. + * @param offset The offset, in bytes, from the start of the file where to read the data. + */ + async getBigUint64(offset: bigint): Promise { + return this.file.getBigUint64(toNumber(offset), true); + } + + /** + * returns an ArrayBuffer whose contents are a copy of this file bytes from startOffset, inclusive, up to endOffset, exclusive. + * @param startOffset The offset, in bytes, from the start of the file where to start reading the data. + * @param endOffset The offset, in bytes, from the start of the file where to end reading the data. + */ + async slice(startOffset: bigint, endOffset: bigint): Promise { + return this.file.buffer.slice(toNumber(startOffset), toNumber(endOffset)); + } + + /** the length (in bytes) of the data. */ + get length() { + return BigInt(this.file.byteLength); + } +} diff --git a/modules/i3s/src/lib/parsers/parse-zip/file-provider.ts b/modules/i3s/src/lib/parsers/parse-zip/file-provider.ts index 4d0cebf9c7..5f5d2e0455 100644 --- a/modules/i3s/src/lib/parsers/parse-zip/file-provider.ts +++ b/modules/i3s/src/lib/parsers/parse-zip/file-provider.ts @@ -6,29 +6,35 @@ export interface FileProvider { * Gets an unsigned 8-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the start of the file where to read the data. */ - getUint8(offset: number): Promise; + getUint8(offset: bigint): Promise; /** * Gets an unsigned 16-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the start of the file where to read the data. */ - getUint16(offset: number): Promise; + getUint16(offset: bigint): Promise; /** * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the file of the view where to read the data. */ - getUint32(offset: number): Promise; + getUint32(offset: bigint): Promise; + + /** + * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. + * @param offset The offset, in byte, from the file of the view where to read the data. + */ + getBigUint64(offset: bigint): Promise; /** * returns an ArrayBuffer whose contents are a copy of this file bytes from startOffset, inclusive, up to endOffset, exclusive. * @param startOffset The offset, in bytes, from the start of the file where to start reading the data. * @param endOffset The offset, in bytes, from the start of the file where to end reading the data. */ - slice(startOffset: number, endOffset: number): Promise; + slice(startOffset: bigint, endOffset: bigint): Promise; /** * the length (in bytes) of the data. */ - length: number; + length: bigint; } diff --git a/modules/i3s/src/lib/parsers/parse-zip/local-file-header.ts b/modules/i3s/src/lib/parsers/parse-zip/local-file-header.ts index 8e47b0f8c9..9b3833d1e6 100644 --- a/modules/i3s/src/lib/parsers/parse-zip/local-file-header.ts +++ b/modules/i3s/src/lib/parsers/parse-zip/local-file-header.ts @@ -12,19 +12,20 @@ export type ZipLocalFileHeader = { /** Extra field length */ extraFieldLength: number; /** Offset of the file data */ - fileDataOffset: number; + fileDataOffset: bigint; /** Compressed size */ - compressedSize: number; + compressedSize: bigint; }; const offsets = { - COMPRESSED_SIZE_OFFSET: 18, - FILE_NAME_LENGTH_OFFSET: 26, - EXTRA_FIELD_LENGTH_OFFSET: 28, - FILE_NAME_OFFSET: 30 + COMPRESSED_SIZE_OFFSET: 18n, + UNCOMPRESSED_SIZE_OFFSET: 22n, + FILE_NAME_LENGTH_OFFSET: 26n, + EXTRA_FIELD_LENGTH_OFFSET: 28n, + FILE_NAME_OFFSET: 30n }; -const signature = Buffer.from([0x50, 0x4b, 0x03, 0x04]); +export const signature = Buffer.from([0x50, 0x4b, 0x03, 0x04]); /** * Parses local file header of zip file @@ -33,27 +34,52 @@ const signature = Buffer.from([0x50, 0x4b, 0x03, 0x04]); * @returns Info from the header */ export const parseZipLocalFileHeader = async ( - headerOffset: number, + headerOffset: bigint, buffer: FileProvider ): Promise => { - if (Buffer.from(await buffer.slice(headerOffset, headerOffset + 4)).compare(signature) !== 0) { + if (Buffer.from(await buffer.slice(headerOffset, headerOffset + 4n)).compare(signature) !== 0) { return Promise.resolve(undefined); } const fileNameLength = await buffer.getUint16(headerOffset + offsets.FILE_NAME_LENGTH_OFFSET); - const fileName = new TextDecoder().decode( - await buffer.slice( - headerOffset + offsets.FILE_NAME_OFFSET, - headerOffset + offsets.FILE_NAME_OFFSET + fileNameLength + const fileName = new TextDecoder() + .decode( + await buffer.slice( + headerOffset + offsets.FILE_NAME_OFFSET, + headerOffset + offsets.FILE_NAME_OFFSET + BigInt(fileNameLength) + ) ) - ); + .split('\\') + .join('/'); const extraFieldLength = await buffer.getUint16(headerOffset + offsets.EXTRA_FIELD_LENGTH_OFFSET); - const fileDataOffset = - headerOffset + offsets.FILE_NAME_OFFSET + fileNameLength + extraFieldLength; + let fileDataOffset = + headerOffset + offsets.FILE_NAME_OFFSET + BigInt(fileNameLength + extraFieldLength); - const compressedSize = await buffer.getUint32(headerOffset + offsets.COMPRESSED_SIZE_OFFSET); + let compressedSize = BigInt( + await buffer.getUint32(headerOffset + offsets.COMPRESSED_SIZE_OFFSET) + ); // add zip 64 logic + + let uncompressedSize = BigInt( + await buffer.getUint32(headerOffset + offsets.UNCOMPRESSED_SIZE_OFFSET) + ); // add zip 64 logic + + const extraOffset = headerOffset + offsets.FILE_NAME_OFFSET + BigInt(fileNameLength); + + let offsetInZip64Data = 4n; + // looking for info that might be also be in zip64 extra field + if (uncompressedSize === BigInt(0xffffffff)) { + uncompressedSize = await buffer.getBigUint64(extraOffset + offsetInZip64Data); + offsetInZip64Data += 8n; + } + if (compressedSize === BigInt(0xffffffff)) { + compressedSize = await buffer.getBigUint64(extraOffset + offsetInZip64Data); + offsetInZip64Data += 8n; + } + if (fileDataOffset === BigInt(0xffffffff)) { + fileDataOffset = await buffer.getBigUint64(extraOffset + offsetInZip64Data); // setting it to the one from zip64 + } return { fileNameLength, diff --git a/modules/i3s/test/index.js b/modules/i3s/test/index.js index 04a0e240b1..eec73ed793 100644 --- a/modules/i3s/test/index.js +++ b/modules/i3s/test/index.js @@ -8,3 +8,7 @@ import './i3s-attribute-loader.spec'; import './i3s-content-loader.spec'; import './i3s-building-scene-layer-loader.spec'; import './arcgis-webscene-loader.spec'; +import './zip-utils/data-view-provider.spec'; +import './zip-utils/cd-file-header.spec'; +import './zip-utils/local-file-header.spec'; +import './search-from-the-end.spec'; diff --git a/modules/i3s/test/local-file-header.spec.js b/modules/i3s/test/local-file-header.spec.js deleted file mode 100644 index 1a4c5bba79..0000000000 --- a/modules/i3s/test/local-file-header.spec.js +++ /dev/null @@ -1,14 +0,0 @@ -import test from 'tape-promise/tape'; -import {DATA_ARRAY} from './data/test.zip.js'; -import {parseZipLocalFileHeader} from '../src/lib/parsers/parse-zip/local-file-header.js'; -import {DataViewFileProvider} from '../src/lib/parsers/parse-zip/buffer-file-provider.js'; - -test('SLPKLoader#local file header parse', async (t) => { - const localFileHeader = await parseZipLocalFileHeader( - 0, - new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)) - ); - t.deepEqual(localFileHeader?.compressedSize, 39); - t.deepEqual(localFileHeader?.fileNameLength, 9); - t.end(); -}); diff --git a/modules/i3s/test/search-from-the-end.spec.js b/modules/i3s/test/search-from-the-end.spec.js new file mode 100644 index 0000000000..20e886d95d --- /dev/null +++ b/modules/i3s/test/search-from-the-end.spec.js @@ -0,0 +1,15 @@ +import test from 'tape-promise/tape'; +import {DATA_ARRAY} from './data/test.zip.js'; +import {searchFromTheEnd} from '../src/lib/parsers/parse-slpk/search-from-the-end'; +import {DataViewFileProvider} from '../src/lib/parsers/parse-zip/data-view-file-provider'; + +test('SLPKLoader#searchFromTheEnd', async (t) => { + t.equals( + await searchFromTheEnd( + new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)), + [0x50, 0x4b, 0x03, 0x04] + ), + 0n + ); + t.end(); +}); diff --git a/modules/i3s/test/cd-file-header.spec.js b/modules/i3s/test/zip-utils/cd-file-header.spec.js similarity index 50% rename from modules/i3s/test/cd-file-header.spec.js rename to modules/i3s/test/zip-utils/cd-file-header.spec.js index dc5921333d..c06945164a 100644 --- a/modules/i3s/test/cd-file-header.spec.js +++ b/modules/i3s/test/zip-utils/cd-file-header.spec.js @@ -1,16 +1,16 @@ import test from 'tape-promise/tape'; -import {DATA_ARRAY} from './data/test.zip.js'; -import {parseZipCDFileHeader} from '../src/lib/parsers/parse-zip/cd-file-header.js'; -import {DataViewFileProvider} from '../src/lib/parsers/parse-zip/buffer-file-provider.js'; +import {DATA_ARRAY} from '../data/test.zip.js'; +import {parseZipCDFileHeader} from '../../src/lib/parsers/parse-zip/cd-file-header'; +import {DataViewFileProvider} from '../../src/lib/parsers/parse-zip/data-view-file-provider'; test('SLPKLoader#central directory file header parse', async (t) => { const cdFileHeader = await parseZipCDFileHeader( - 78, + 78n, new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)) ); - t.deepEqual(cdFileHeader.compressedSize, 39); + t.deepEqual(cdFileHeader.compressedSize, 39n); t.deepEqual(cdFileHeader.fileNameLength, 9); t.deepEqual(cdFileHeader.fileName, 'test.json'); - t.deepEqual(cdFileHeader.localHeaderOffset, 0); + t.deepEqual(cdFileHeader.localHeaderOffset, 0n); t.end(); }); diff --git a/modules/i3s/test/zip-utils/data-view-provider.spec.ts b/modules/i3s/test/zip-utils/data-view-provider.spec.ts new file mode 100644 index 0000000000..8739a5b094 --- /dev/null +++ b/modules/i3s/test/zip-utils/data-view-provider.spec.ts @@ -0,0 +1,34 @@ +import test from 'tape-promise/tape'; +import {DATA_ARRAY} from '../data/test.zip.js'; +import {DataViewFileProvider} from '../../src/lib/parsers/parse-zip/data-view-file-provider'; +import {signature} from '../../src/lib/parsers/parse-zip/local-file-header'; + +test('DataViewFileProvider#slice', async (t) => { + const provider = new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)); + t.equals(Buffer.from(await provider.slice(0n, 4n)).compare(signature), 0); + t.end(); +}); + +test('DataViewFileProvider#getUint8', async (t) => { + const provider = new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)); + t.equals(await provider.getUint8(0n), 80); + t.end(); +}); + +test('DataViewFileProvider#local file header parse', async (t) => { + const provider = new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)); + t.equals(await provider.getUint16(0n), 19280); + t.end(); +}); + +test('DataViewFileProvider#local file header parse', async (t) => { + const provider = new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)); + t.equals(await provider.getUint32(0n), 67324752); + t.end(); +}); + +test('DataViewFileProvider#local file header parse', async (t) => { + const provider = new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)); + t.equals(await provider.getBigUint64(0n), 563035920091984n); + t.end(); +}); diff --git a/modules/i3s/test/zip-utils/local-file-header.spec.js b/modules/i3s/test/zip-utils/local-file-header.spec.js new file mode 100644 index 0000000000..d3ba69ff38 --- /dev/null +++ b/modules/i3s/test/zip-utils/local-file-header.spec.js @@ -0,0 +1,14 @@ +import test from 'tape-promise/tape'; +import {DATA_ARRAY} from '../data/test.zip.js'; +import {parseZipLocalFileHeader} from '../../src/lib/parsers/parse-zip/local-file-header'; +import {DataViewFileProvider} from '../../src/lib/parsers/parse-zip/data-view-file-provider'; + +test('SLPKLoader#local file header parse', async (t) => { + const localFileHeader = await parseZipLocalFileHeader( + 0n, + new DataViewFileProvider(new DataView(DATA_ARRAY.buffer)) + ); + t.deepEqual(localFileHeader?.compressedSize, 39n); + t.deepEqual(localFileHeader?.fileNameLength, 9); + t.end(); +}); diff --git a/modules/tile-converter/src/i3s-server/app.js b/modules/tile-converter/src/i3s-server/app.js index aa65065d01..8ae6c97691 100644 --- a/modules/tile-converter/src/i3s-server/app.js +++ b/modules/tile-converter/src/i3s-server/app.js @@ -4,7 +4,6 @@ const logger = require('morgan'); const cors = require('cors'); const indexRouter = require('./routes/index'); -const {sceneServerRouter, router} = require('./routes/slpk-router'); const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef const app = express(); @@ -16,6 +15,7 @@ app.use(express.static(path.join(__dirname, 'public'))); app.use(cors()); if (/\.slpk$/.test(I3S_LAYER_PATH)) { + const {sceneServerRouter, router} = require('./routes/slpk-router'); app.use('/SceneServer/layers/0', router); app.use('/SceneServer', sceneServerRouter); } else { diff --git a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js index 590574dd29..6b2e9317da 100644 --- a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js +++ b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js @@ -1,12 +1,12 @@ require('@loaders.gl/polyfills'); -const {fetchFile, parse} = require('@loaders.gl/core'); -const {SLPKLoader} = require('@loaders.gl/i3s'); +const {parseSLPK} = require('@loaders.gl/i3s'); const path = require('path'); +const {FileHandleProvider} = require('@loaders.gl/tile-converter'); let slpkArchive; const loadArchive = async (fullLayerPath) => { - slpkArchive = await (await fetchFile(fullLayerPath)).arrayBuffer(); + slpkArchive = await parseSLPK(await FileHandleProvider.from(fullLayerPath)); }; const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef @@ -19,14 +19,7 @@ async function getFileByUrl(url) { let uncompressedFile; if (trimmedPath) { try { - uncompressedFile = Buffer.from( - await parse(slpkArchive, SLPKLoader, { - slpk: { - path: trimmedPath[1], - pathMode: 'http' - } - }) - ); + uncompressedFile = Buffer.from(await slpkArchive.getFile(trimmedPath[1], 'http')); } catch (e) {} } return uncompressedFile; diff --git a/modules/tile-converter/src/index.ts b/modules/tile-converter/src/index.ts index c2b412ee13..3deecad0e4 100644 --- a/modules/tile-converter/src/index.ts +++ b/modules/tile-converter/src/index.ts @@ -1,2 +1,3 @@ export {default as I3SConverter} from './i3s-converter/i3s-converter'; export {default as Tiles3DConverter} from './3d-tiles-converter/3d-tiles-converter'; +export {FileHandleProvider} from './slpk-extractor/helpers/file-handle-provider'; diff --git a/modules/tile-converter/src/slpk-extractor/helpers/file-handle-provider.ts b/modules/tile-converter/src/slpk-extractor/helpers/file-handle-provider.ts index 777a7ef942..95c9de64a9 100644 --- a/modules/tile-converter/src/slpk-extractor/helpers/file-handle-provider.ts +++ b/modules/tile-converter/src/slpk-extractor/helpers/file-handle-provider.ts @@ -1,5 +1,5 @@ import {FileProvider} from '@loaders.gl/i3s'; -import {promises as fsPromises, PathLike} from 'fs'; +import {FileHandle} from './fs-promises'; /** * Provides file data using node fs library @@ -9,31 +9,31 @@ export class FileHandleProvider implements FileProvider { * Returns a new copy of FileHandleProvider * @param path The path to the file in file system */ - static async from(path: PathLike): Promise { - const fileDescriptor = await fsPromises.open(path); - return new FileHandleProvider(fileDescriptor, (await fileDescriptor.stat()).size); + static async from(path: string): Promise { + const fileDescriptor = await FileHandle.open(path); + return new FileHandleProvider(fileDescriptor, fileDescriptor.stat.size); } /** * The FileHandle from which data is provided */ - private fileDescriptor: fsPromises.FileHandle; + private fileDescriptor: FileHandle; /** * The file length in bytes */ - private size: number; + private size: bigint; - private constructor(fileDescriptor: fsPromises.FileHandle, size: number) { + private constructor(fileDescriptor: FileHandle, size: bigint) { this.fileDescriptor = fileDescriptor; this.size = size; } /** - * Gets an unsigned 8-bit integer (unsigned byte) at the specified byte offset from the start of the file. + * Gets an unsigned 8-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the start of the file where to read the data. */ - async getUint8(offset: number): Promise { + async getUint8(offset: bigint): Promise { const val = new Uint8Array( (await this.fileDescriptor.read(Buffer.alloc(1), 0, 1, offset)).buffer.buffer ).at(0); @@ -44,10 +44,10 @@ export class FileHandleProvider implements FileProvider { } /** - * Gets an unsigned 16-bit integer (unsigned byte) at the specified byte offset from the start of the file. + * Gets an unsigned 16-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the start of the file where to read the data. */ - async getUint16(offset: number): Promise { + async getUint16(offset: bigint): Promise { const val = new Uint16Array( (await this.fileDescriptor.read(Buffer.alloc(2), 0, 2, offset)).buffer.buffer ).at(0); @@ -58,10 +58,10 @@ export class FileHandleProvider implements FileProvider { } /** - * Gets an unsigned 32-bit integer (unsigned byte) at the specified byte offset from the start of the file. + * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. * @param offset The offset, in bytes, from the start of the file where to read the data. */ - async getUint32(offset: number): Promise { + async getUint32(offset: bigint): Promise { const val = new Uint32Array( (await this.fileDescriptor.read(Buffer.alloc(4), 0, 4, offset)).buffer.buffer ).at(0); @@ -71,21 +71,39 @@ export class FileHandleProvider implements FileProvider { return val; } + /** + * Gets an unsigned 32-bit integer at the specified byte offset from the start of the file. + * @param offset The offset, in bytes, from the start of the file where to read the data. + */ + async getBigUint64(offset: bigint): Promise { + const val = new BigInt64Array( + (await this.fileDescriptor.read(Buffer.alloc(8), 0, 4, offset)).buffer.buffer + ).at(0); + if (val === undefined) { + throw new Error('something went wrong'); + } + return val; + } + /** * returns an ArrayBuffer whose contents are a copy of this file bytes from startOffset, inclusive, up to endOffset, exclusive. - * @param startOffset The offset, in bytes, from the start of the file where to start reading the data. + * @param startOffsset The offset, in byte, from the start of the file where to start reading the data. * @param endOffset The offset, in bytes, from the start of the file where to end reading the data. */ - async slice(startOffset: number, endOffset: number): Promise { - const length = endOffset - startOffset; - return (await this.fileDescriptor.read(Buffer.alloc(length), 0, length, startOffset)).buffer + async slice(startOffsset: bigint, endOffset: bigint): Promise { + const bigLength = endOffset - startOffsset; + if (bigLength > Number.MAX_SAFE_INTEGER) { + throw new Error('too big slice'); + } + const length = Number(bigLength); + return (await this.fileDescriptor.read(Buffer.alloc(length), 0, length, startOffsset)).buffer .buffer; } /** * the length (in bytes) of the data. */ - get length(): number { + get length(): bigint { return this.size; } } diff --git a/modules/tile-converter/src/slpk-extractor/helpers/fs-promises.ts b/modules/tile-converter/src/slpk-extractor/helpers/fs-promises.ts new file mode 100644 index 0000000000..785f583a11 --- /dev/null +++ b/modules/tile-converter/src/slpk-extractor/helpers/fs-promises.ts @@ -0,0 +1,66 @@ +import {read, open, stat, BigIntStats} from 'fs'; + +/** file reading result */ +export type FileReadResult = { + /** amount of the bytes read */ + bytesRead: number; + /** the buffer filled with data from file*/ + buffer: Buffer; +}; + +/** Object handling file info */ +export class FileHandle { + private fileDescriptor: number; + private stats: BigIntStats; + private constructor(fileDescriptor: number, stats: BigIntStats) { + this.fileDescriptor = fileDescriptor; + this.stats = stats; + } + /** + * Opens a `FileHandle`. + * + * @param path path to the file + * @return Fulfills with a {FileHandle} object. + */ + + static open = async (path: string): Promise => { + const [fd, stats] = await Promise.all([ + new Promise((s) => { + open(path, undefined, undefined, (_err, fd) => s(fd)); + }), + new Promise((s) => { + stat(path, {bigint: true}, (_err, stats) => s(stats)); + }) + ]); + return new FileHandle(fd, stats); + }; + + /** + * Reads data from the file and stores that in the given buffer. + * + * If the file is not modified concurrently, the end-of-file is reached when the + * number of bytes read is zero. + * @param buffer A buffer that will be filled with the file data read. + * @param offset The location in the buffer at which to start filling. + * @param length The number of bytes to read. + * @param position The location where to begin reading data from the file. If `null`, data will be read from the current file position, and the position will be updated. If `position` is an + * integer, the current file position will remain unchanged. + * @return Fulfills upon success with a FileReadResult object + */ + read = ( + buffer: Buffer, + offset: number, + length: number, + position: number | bigint + ): Promise => { + return new Promise((s) => { + read(this.fileDescriptor, buffer, offset, length, position, (_err, bytesRead, buffer) => + s({bytesRead, buffer}) + ); + }); + }; + + get stat(): BigIntStats { + return this.stats; + } +} diff --git a/modules/tile-converter/src/slpk-extractor/slpk-extractor.ts b/modules/tile-converter/src/slpk-extractor/slpk-extractor.ts index 897a3a48bb..f19bd12174 100644 --- a/modules/tile-converter/src/slpk-extractor/slpk-extractor.ts +++ b/modules/tile-converter/src/slpk-extractor/slpk-extractor.ts @@ -7,9 +7,6 @@ import {path} from '@loaders.gl/loader-utils'; import {GZipCompression} from '@loaders.gl/compression'; import {writeFile} from '../lib/utils/file-utils'; -/** - * names of files that should be changed to index - */ const indexNames = [ '3dSceneLayer.json.gz', '3dNodeIndexDocument.json.gz', @@ -27,9 +24,9 @@ type File = { /** * Converter from slpk to i3s */ -export default class SLPKExtractor { +export default class SLPKConverter { /** - * extract slpk to i3s + * Extract slpk to i3s * @param options * @param options.inputUrl the url to read SLPK file * @param options.outputPath the output filename @@ -43,7 +40,7 @@ export default class SLPKExtractor { const provider = await FileHandleProvider.from(inputUrl); - let localHeader = await parseZipLocalFileHeader(0, provider); + let localHeader = await parseZipLocalFileHeader(0n, provider); while (localHeader) { await this.writeFile( await this.unGzip({ @@ -68,6 +65,7 @@ export default class SLPKExtractor { * Defines file name and path for i3s format * @param fileName initial file name and path */ + private correctIndexNames(fileName: string): string | null { if (indexNames.includes(path.filename(path.join('/', fileName)))) { return path.join(path.dirname(fileName), 'index.json.gz'); @@ -85,6 +83,7 @@ export default class SLPKExtractor { const compression = new GZipCompression(); const decompressedData = await compression.decompress(file.data); + return {data: decompressedData, name: (file.name ?? '').slice(0, -3)}; } return Promise.resolve(file); diff --git a/modules/tile-converter/test/index.js b/modules/tile-converter/test/index.js index aa5222a129..d70c66b0ac 100644 --- a/modules/tile-converter/test/index.js +++ b/modules/tile-converter/test/index.js @@ -13,6 +13,8 @@ import './i3s-converter/helpers/preprocess-3d-tiles.spec'; import './i3s-converter/i3s-converter.spec'; import './slpk-extractor/slpk-extractor.spec'; +import './slpk-extractor/file-handle-provider.spec'; +import './slpk-extractor/file-handle.spec'; import './utils/cli-utils.spec'; import './3d-tiles-converter/helpers/b3dm-converter.spec'; diff --git a/modules/tile-converter/test/slpk-extractor/file-handle-provider.spec.js b/modules/tile-converter/test/slpk-extractor/file-handle-provider.spec.js new file mode 100644 index 0000000000..1ccf3592c6 --- /dev/null +++ b/modules/tile-converter/test/slpk-extractor/file-handle-provider.spec.js @@ -0,0 +1,47 @@ +import test from 'tape-promise/tape'; +import {signature} from '../../../i3s/src/lib/parsers/parse-zip/local-file-header'; +import {FileHandleProvider} from '@loaders.gl/tile-converter'; +import {isBrowser} from '@loaders.gl/core'; + +const SLPKUrl = 'modules/i3s/test/data/DA12_subset.slpk'; + +test('FileHandleProvider#slice', async (t) => { + if (!isBrowser) { + const provider = await FileHandleProvider.from(SLPKUrl); + t.equals(Buffer.from(await provider.slice(0n, 4n)).compare(signature), 0); + } + t.end(); +}); + +test('FileHandleProvider#getUint8', async (t) => { + if (!isBrowser) { + const provider = await FileHandleProvider.from(SLPKUrl); + t.equals(await provider.getUint8(0n), 80); + t.end(); + } + t.end(); +}); + +test('FileHandleProvider#local file header parse', async (t) => { + if (!isBrowser) { + const provider = await FileHandleProvider.from(SLPKUrl); + t.equals(await provider.getUint16(0n), 19280); + } + t.end(); +}); + +test('FileHandleProvider#local file header parse', async (t) => { + if (!isBrowser) { + const provider = await FileHandleProvider.from(SLPKUrl); + t.equals(await provider.getUint32(0n), 67324752); + } + t.end(); +}); + +test('FileHandleProvider#local file header parse', async (t) => { + if (!isBrowser) { + const provider = await FileHandleProvider.from(SLPKUrl); + t.equals(await provider.getBigUint64(0n), 67324752n); + } + t.end(); +}); diff --git a/modules/tile-converter/test/slpk-extractor/file-handle.spec.js b/modules/tile-converter/test/slpk-extractor/file-handle.spec.js new file mode 100644 index 0000000000..15edda5d74 --- /dev/null +++ b/modules/tile-converter/test/slpk-extractor/file-handle.spec.js @@ -0,0 +1,20 @@ +import test from 'tape-promise/tape'; +import {isBrowser} from '@loaders.gl/core'; +import {FileHandle} from '../../src/slpk-extractor/helpers/fs-promises'; +import {promises as fsPromises} from 'fs'; + +const SLPKUrl = 'modules/i3s/test/data/DA12_subset.slpk'; + +test('FileHandle#open and read', async (t) => { + if (!isBrowser) { + const provider = await FileHandle.open(SLPKUrl); + const fsHandler = await fsPromises.open(SLPKUrl); + t.equals( + (await provider.read(Buffer.alloc(4), 0, 4, 1)).buffer.compare( + (await fsHandler.read(Buffer.alloc(4), 0, 4, 1)).buffer + ), + 0 + ); + } + t.end(); +}); From 93e3678dc37859d716789b9c54b849f506437e6a Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Thu, 20 Jul 2023 16:14:13 +0200 Subject: [PATCH 013/214] chore(tile-converter): bump i3s-server deps (#2561) --- modules/tile-converter/package.json | 6 +- yarn.lock | 151 +++------------------------- 2 files changed, 16 insertions(+), 141 deletions(-) diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index 53b2160754..947d6853a5 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -64,13 +64,13 @@ "archiver": "^5.0.0", "cors": "^2.8.5", "crypt": "^0.0.2", - "debug": "~2.6.9", - "express": "~4.17.3", + "debug": "~4.3.4", + "express": "~4.18.2", "json-map-transform": "^1.2.6", "jszip": "^3.5.0", "md5": "^2.3.0", "morgan": "~1.9.1", - "uuid": "^8.1.0" + "uuid": "^9.0.0" }, "peerDependencies": { "@loaders.gl/core": "^4.0.0-alpha.8" diff --git a/yarn.lock b/yarn.lock index 1c741c82b5..f7e201f458 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3658,22 +3658,6 @@ bn.js@^5.0.0, bn.js@^5.1.1: resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70" integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ== -body-parser@1.19.2: - version "1.19.2" - resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.19.2.tgz#4714ccd9c157d44797b8b5607d72c0b89952f26e" - integrity sha512-SAAwOxgoCKMGs9uUAUFHygfLAyaniaoun6I8mFY9pRAJL9+Kec34aU+oIjDhTycub1jozEfEwx1W1IuOYxVSFw== - dependencies: - bytes "3.1.2" - content-type "~1.0.4" - debug "2.6.9" - depd "~1.1.2" - http-errors "1.8.1" - iconv-lite "0.4.24" - on-finished "~2.3.0" - qs "6.9.7" - raw-body "2.4.3" - type-is "~1.6.18" - body-parser@1.20.1: version "1.20.1" resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.1.tgz#b1812a8912c195cd371a3ee5e66faa2338a5c668" @@ -4065,9 +4049,9 @@ camelcase@^5.0.0, camelcase@^5.3.1: integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== caniuse-lite@^1.0.30001449: - version "1.0.30001515" - resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001515.tgz" - integrity sha512-eEFDwUOZbE24sb+Ecsx3+OvNETqjWIdabMy52oOkIgcUtAsQifjUG9q4U9dgTHJM2mfk4uEPxc0+xuFdJ629QA== + version "1.0.30001517" + resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001517.tgz" + integrity sha512-Vdhm5S11DaFVLlyiKu4hiUTkpZu+y1KA/rZZqVQfOD5YdDT/eQKlkt7NaE0WGOFgX32diqt9MiP9CAiFeRklaA== caseless@~0.12.0: version "0.12.0" @@ -4582,11 +4566,6 @@ cookie-signature@1.0.6: resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" integrity sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ== -cookie@0.4.2: - version "0.4.2" - resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.2.tgz#0e41f24de5ecf317947c82fc789e06a884824432" - integrity sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA== - cookie@0.5.0: version "0.5.0" resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.5.0.tgz#d1f5d71adec6558c58f389987c366aa47e994f8b" @@ -4863,7 +4842,7 @@ dateformat@^3.0.0: resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-3.0.3.tgz#a6e37499a4d9a9cf85ef5872044d62901c9889ae" integrity sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q== -debug@2.6.9, debug@^2.2.0, debug@^2.3.3, debug@~2.6.9: +debug@2.6.9, debug@^2.2.0, debug@^2.3.3: version "2.6.9" resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== @@ -4877,7 +4856,7 @@ debug@3.1.0: dependencies: ms "2.0.0" -debug@4, debug@4.3.4, debug@^4.0.0, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1, debug@^4.2.0, debug@^4.3.1: +debug@4, debug@4.3.4, debug@^4.0.0, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1, debug@^4.2.0, debug@^4.3.1, debug@~4.3.4: version "4.3.4" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== @@ -5090,11 +5069,6 @@ destroy@1.2.0: resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015" integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== -destroy@~1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80" - integrity sha512-3NdhDuEXnfun/z7x9GOElY49LoqVHoGScmOKwmxhsS8N5Y+Z8KyPPDnaSzqWgYt/ji4mqwfTS34Htrk0zPIXVg== - detect-file@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/detect-file/-/detect-file-1.0.0.tgz#f0d66d03672a825cb1b73bdb3fe62310c8e552b7" @@ -6007,7 +5981,7 @@ expand-tilde@^2.0.0, expand-tilde@^2.0.2: dependencies: homedir-polyfill "^1.0.1" -express@^4.16.3, express@^4.17.1: +express@^4.16.3, express@^4.17.1, express@~4.18.2: version "4.18.2" resolved "https://registry.yarnpkg.com/express/-/express-4.18.2.tgz#3fabe08296e930c796c19e3c516979386ba9fd59" integrity sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ== @@ -6044,42 +6018,6 @@ express@^4.16.3, express@^4.17.1: utils-merge "1.0.1" vary "~1.1.2" -express@~4.17.3: - version "4.17.3" - resolved "https://registry.yarnpkg.com/express/-/express-4.17.3.tgz#f6c7302194a4fb54271b73a1fe7a06478c8f85a1" - integrity sha512-yuSQpz5I+Ch7gFrPCk4/c+dIBKlQUxtgwqzph132bsT6qhuzss6I8cLJQz7B3rFblzd6wtcI0ZbGltH/C4LjUg== - dependencies: - accepts "~1.3.8" - array-flatten "1.1.1" - body-parser "1.19.2" - content-disposition "0.5.4" - content-type "~1.0.4" - cookie "0.4.2" - cookie-signature "1.0.6" - debug "2.6.9" - depd "~1.1.2" - encodeurl "~1.0.2" - escape-html "~1.0.3" - etag "~1.8.1" - finalhandler "~1.1.2" - fresh "0.5.2" - merge-descriptors "1.0.1" - methods "~1.1.2" - on-finished "~2.3.0" - parseurl "~1.3.3" - path-to-regexp "0.1.7" - proxy-addr "~2.0.7" - qs "6.9.7" - range-parser "~1.2.1" - safe-buffer "5.2.1" - send "0.17.2" - serve-static "1.14.2" - setprototypeof "1.2.0" - statuses "~1.5.0" - type-is "~1.6.18" - utils-merge "1.0.1" - vary "~1.1.2" - extend-shallow@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-2.0.1.tgz#51af7d614ad9a9f610ea1bafbb989d6b1c56890f" @@ -6287,19 +6225,6 @@ finalhandler@1.2.0: statuses "2.0.1" unpipe "~1.0.0" -finalhandler@~1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d" - integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA== - dependencies: - debug "2.6.9" - encodeurl "~1.0.2" - escape-html "~1.0.3" - on-finished "~2.3.0" - parseurl "~1.3.3" - statuses "~1.5.0" - unpipe "~1.0.0" - find-cache-dir@^2.0.0, find-cache-dir@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-2.1.0.tgz#8d0f94cd13fe43c6c7c261a0d86115ca918c05f7" @@ -7178,17 +7103,6 @@ http-deceiver@^1.2.7: resolved "https://registry.yarnpkg.com/http-deceiver/-/http-deceiver-1.2.7.tgz#fa7168944ab9a519d337cb0bec7284dc3e723d87" integrity sha512-LmpOGxTfbpgtGVxJrj5k7asXHCgNZp5nLfp+hWc8QQRqtb7fUy6kRY3BO1h9ddF6yIPYUARgxGOwB42DnxIaNw== -http-errors@1.8.1: - version "1.8.1" - resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.8.1.tgz#7c3f28577cbc8a207388455dbd62295ed07bd68c" - integrity sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g== - dependencies: - depd "~1.1.2" - inherits "2.0.4" - setprototypeof "1.2.0" - statuses ">= 1.5.0 < 2" - toidentifier "1.0.1" - http-errors@2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3" @@ -10361,11 +10275,6 @@ qs@6.11.0: dependencies: side-channel "^1.0.4" -qs@6.9.7: - version "6.9.7" - resolved "https://registry.yarnpkg.com/qs/-/qs-6.9.7.tgz#4610846871485e1e048f44ae3b94033f0e675afe" - integrity sha512-IhMFgUmuNpyRfxA90umL7ByLlgRXu6tIfKPpF5TmcfRLlLCckfP/g3IQmju6jjpu+Hh8rA+2p6A27ZSPOOHdKw== - qs@^6.9.4: version "6.11.1" resolved "https://registry.yarnpkg.com/qs/-/qs-6.11.1.tgz#6c29dff97f0c0060765911ba65cbc9764186109f" @@ -10443,16 +10352,6 @@ range-parser@^1.2.1, range-parser@~1.2.1: resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031" integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg== -raw-body@2.4.3: - version "2.4.3" - resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.4.3.tgz#8f80305d11c2a0a545c2d9d89d7a0286fcead43c" - integrity sha512-UlTNLIcu0uzb4D2f4WltY6cVjLi+/jEN4lgEUj3E04tpMDpUlkBo/eSn6zou9hum2VMNpCCUone0O0WeJim07g== - dependencies: - bytes "3.1.2" - http-errors "1.8.1" - iconv-lite "0.4.24" - unpipe "1.0.0" - raw-body@2.5.1: version "2.5.1" resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.1.tgz#fe1b1628b181b700215e5fd42389f98b71392857" @@ -11078,25 +10977,6 @@ semver@^7.2.1, semver@^7.3.4, semver@^7.3.5: dependencies: lru-cache "^6.0.0" -send@0.17.2: - version "0.17.2" - resolved "https://registry.yarnpkg.com/send/-/send-0.17.2.tgz#926622f76601c41808012c8bf1688fe3906f7820" - integrity sha512-UJYB6wFSJE3G00nEivR5rgWp8c2xXvJ3OPWPhmuteU0IKj8nKbG3DrjiOmLwpnHGYWAVwA69zmTm++YG0Hmwww== - dependencies: - debug "2.6.9" - depd "~1.1.2" - destroy "~1.0.4" - encodeurl "~1.0.2" - escape-html "~1.0.3" - etag "~1.8.1" - fresh "0.5.2" - http-errors "1.8.1" - mime "1.6.0" - ms "2.1.3" - on-finished "~2.3.0" - range-parser "~1.2.1" - statuses "~1.5.0" - send@0.18.0: version "0.18.0" resolved "https://registry.yarnpkg.com/send/-/send-0.18.0.tgz#670167cc654b05f5aa4a767f9113bb371bc706be" @@ -11136,16 +11016,6 @@ serve-index@^1.9.1: mime-types "~2.1.17" parseurl "~1.3.2" -serve-static@1.14.2: - version "1.14.2" - resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.14.2.tgz#722d6294b1d62626d41b43a013ece4598d292bfa" - integrity sha512-+TMNA9AFxUEGuC0z2mevogSnn9MXKb4fa7ngeRMJaaGv8vTwnIEkKi+QGvPt33HSnf8pRS+WGM0EbMtCJLKMBQ== - dependencies: - encodeurl "~1.0.2" - escape-html "~1.0.3" - parseurl "~1.3.3" - send "0.17.2" - serve-static@1.15.0: version "1.15.0" resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.15.0.tgz#faaef08cffe0a1a62f60cad0c4e513cff0ac9540" @@ -11552,7 +11422,7 @@ statuses@2.0.1: resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ== -"statuses@>= 1.4.0 < 2", "statuses@>= 1.5.0 < 2", statuses@~1.5.0: +"statuses@>= 1.4.0 < 2": version "1.5.0" resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c" integrity sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA== @@ -12541,11 +12411,16 @@ uuid@^3.0.1, uuid@^3.3.2: resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee" integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A== -uuid@^8.1.0, uuid@^8.3.2: +uuid@^8.3.2: version "8.3.2" resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2" integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg== +uuid@^9.0.0: + version "9.0.0" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-9.0.0.tgz#592f550650024a38ceb0c562f2f6aa435761efb5" + integrity sha512-MXcSTerfPa4uqyzStbRoTgt5XIe3x5+42+q1sDuy3R5MDk66URdLMOZe5aPX/SQd+kuYAh0FdP/pO28IkQyTeg== + v8-compile-cache-lib@^3.0.1: version "3.0.1" resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" From 47467ab23c65917e4642730e99b5d005af333c76 Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Thu, 20 Jul 2023 16:42:10 +0200 Subject: [PATCH 014/214] v4.0.0-alpha.15 --- CHANGELOG.md | 14 +++++++++++++- lerna.json | 2 +- modules/3d-tiles/package.json | 12 ++++++------ modules/arrow/package.json | 6 +++--- modules/bson/package.json | 8 ++++---- modules/compression/package.json | 6 +++--- modules/core/package.json | 6 +++--- modules/crypto/package.json | 6 +++--- modules/csv/package.json | 6 +++--- modules/draco/package.json | 10 +++++----- modules/excel/package.json | 6 +++--- modules/flatgeobuf/package.json | 6 +++--- modules/geopackage/package.json | 8 ++++---- modules/geotiff/package.json | 2 +- modules/gis/package.json | 6 +++--- modules/gltf/package.json | 10 +++++----- modules/i3s/package.json | 16 ++++++++-------- modules/images/package.json | 4 ++-- modules/json/package.json | 8 ++++---- modules/kml/package.json | 8 ++++---- modules/las/package.json | 6 +++--- modules/loader-utils/package.json | 4 ++-- modules/math/package.json | 6 +++--- modules/mvt/package.json | 8 ++++---- modules/netcdf/package.json | 4 ++-- modules/obj/package.json | 6 +++--- modules/parquet/package.json | 10 +++++----- modules/pcd/package.json | 6 +++--- modules/ply/package.json | 6 +++--- modules/polyfills/package.json | 2 +- modules/potree/package.json | 4 ++-- modules/schema/package.json | 2 +- modules/shapefile/package.json | 8 ++++---- modules/terrain/package.json | 8 ++++---- modules/textures/package.json | 12 ++++++------ modules/tile-converter/package.json | 26 +++++++++++++------------- modules/tiles/package.json | 6 +++--- modules/video/package.json | 6 +++--- modules/wkt/package.json | 6 +++--- modules/wms/package.json | 10 +++++----- modules/worker-utils/package.json | 2 +- modules/xml/package.json | 6 +++--- modules/zarr/package.json | 2 +- modules/zip/package.json | 2 +- 44 files changed, 160 insertions(+), 148 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2422665511..18e92e2a14 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,19 @@ > The official 4.0 alpha track starts with alpha.6 > The early pre-release track was abandoned due to build incompatibility problems. -release info (#2491)) +> release info (#2491)) + +### v4.0.0-alpha.15 + +- chore(tile-converter): bump i3s-server deps (#2561) +- chore(tile-converter): Support for SLPKs larger than 2 Gb (#2547) +- feat(tile-converter): i3s-server bundle (#2555) +- chore(deps): bump semver from 5.7.1 to 5.7.2 in /website (#2546) +- fix(docs): JSONLoader \_rootObjectBatches removed but not mentioned in upgrade guide (#2558) +- chore(deps): bump word-wrap in /test/apps/typescript-test (#2559) +- fix(tile-converter): CesiumION tileset URL (#2560) +- chore: update CHANGELOG.md (#2551) +- chore(tile-converter): update i3s-server manual (#2552) ### v4.0.0-alpha.14 diff --git a/lerna.json b/lerna.json index ae9df994a2..9e8640d532 100644 --- a/lerna.json +++ b/lerna.json @@ -1,6 +1,6 @@ { "lerna": "2.9.1", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "command": { "publish": {}, "bootstrap": {} diff --git a/modules/3d-tiles/package.json b/modules/3d-tiles/package.json index ce8664a921..ade5067bab 100644 --- a/modules/3d-tiles/package.json +++ b/modules/3d-tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/3d-tiles", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "3D Tiles, an open standard for streaming massive heterogeneous 3D geospatial datasets.", "license": "MIT", "publishConfig": { @@ -34,11 +34,11 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.14", - "@loaders.gl/gltf": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/math": "4.0.0-alpha.14", - "@loaders.gl/tiles": "4.0.0-alpha.14", + "@loaders.gl/draco": "4.0.0-alpha.15", + "@loaders.gl/gltf": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/math": "4.0.0-alpha.15", + "@loaders.gl/tiles": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1", "@math.gl/geospatial": "^3.5.1", "@probe.gl/log": "^4.0.4", diff --git a/modules/arrow/package.json b/modules/arrow/package.json index 00348f679b..649e79610a 100644 --- a/modules/arrow/package.json +++ b/modules/arrow/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/arrow", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Simple columnar table loader for the Apache Arrow format", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ "build-worker2": "esbuild src/workers/arrow-worker.ts --bundle --outfile=dist/arrow-worker.js --platform=browser --external:{stream}" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/bson/package.json b/modules/bson/package.json index da52e9aea7..52b0985a8b 100644 --- a/modules/bson/package.json +++ b/modules/bson/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/bson", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -34,9 +34,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@types/bson": "4.2.0", "bson": "4.2.0" }, diff --git a/modules/compression/package.json b/modules/compression/package.json index c61433231e..ebdcdcc008 100644 --- a/modules/compression/package.json +++ b/modules/compression/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/compression", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Decompression and compression plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -39,8 +39,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "@types/brotli": "^1.3.0", "@types/pako": "^1.0.1", "fflate": "0.7.4", diff --git a/modules/core/package.json b/modules/core/package.json index b88c0823cb..855263eda2 100644 --- a/modules/core/package.json +++ b/modules/core/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/core", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "The core API for working with loaders.gl loaders and writers", "license": "MIT", "publishConfig": { @@ -43,8 +43,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "@probe.gl/log": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/crypto/package.json b/modules/crypto/package.json index 3fd3c9f6be..b5ab5dfa42 100644 --- a/modules/crypto/package.json +++ b/modules/crypto/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/crypto", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Cryptographic/hashing plugins for loaders.gl", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "@types/crypto-js": "^4.0.2" }, "devDependencies": { diff --git a/modules/csv/package.json b/modules/csv/package.json index 2e22cb7625..b0271d11f9 100644 --- a/modules/csv/package.json +++ b/modules/csv/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/csv", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for CSV and DSV table formats", "license": "MIT", "publishConfig": { @@ -30,8 +30,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "devDependencies": { "d3-dsv": "^1.2.0" diff --git a/modules/draco/package.json b/modules/draco/package.json index 727e531291..fd1a953853 100644 --- a/modules/draco/package.json +++ b/modules/draco/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/draco", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader and writer for Draco compressed meshes and point clouds", "license": "MIT", "publishConfig": { @@ -42,13 +42,13 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "draco3d": "1.5.5" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.14" + "@loaders.gl/polyfills": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/excel/package.json b/modules/excel/package.json index 3da3d72f44..b702126de5 100644 --- a/modules/excel/package.json +++ b/modules/excel/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/excel", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for Excel files", "license": "MIT", "publishConfig": { @@ -35,8 +35,8 @@ "build-worker": "esbuild src/workers/excel-worker.ts --bundle --outfile=dist/excel-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "xlsx": "^0.17.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/flatgeobuf/package.json b/modules/flatgeobuf/package.json index f800fd1478..bf0d61ce6c 100644 --- a/modules/flatgeobuf/package.json +++ b/modules/flatgeobuf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/flatgeobuf", "description": "Loader for FlatGeobuf", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -32,8 +32,8 @@ "build-worker": "esbuild src/workers/flatgeobuf-worker.ts --bundle --outfile=dist/flatgeobuf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", "@math.gl/proj4": "^3.3.1", "flatgeobuf": "3.6.5" }, diff --git a/modules/geopackage/package.json b/modules/geopackage/package.json index 6ded8f1c52..1614636b80 100644 --- a/modules/geopackage/package.json +++ b/modules/geopackage/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/geopackage", "description": "GeoPackage data loaders", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -25,9 +25,9 @@ ], "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", - "@loaders.gl/wkt": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", + "@loaders.gl/wkt": "4.0.0-alpha.15", "@math.gl/proj4": "^3.5.1", "@types/sql.js": "^1.4.2", "sql.js": "1.5.0" diff --git a/modules/geotiff/package.json b/modules/geotiff/package.json index 14c6bb21bc..69b6dcce66 100644 --- a/modules/geotiff/package.json +++ b/modules/geotiff/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/geotiff", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for tiff and geotiff", "license": "MIT", "publishConfig": { diff --git a/modules/gis/package.json b/modules/gis/package.json index 6d26e25844..fe506393fc 100644 --- a/modules/gis/package.json +++ b/modules/gis/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/gis", "description": "Helpers for GIS category data", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -24,8 +24,8 @@ "README.md" ], "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@mapbox/vector-tile": "^1.3.1", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" diff --git a/modules/gltf/package.json b/modules/gltf/package.json index 3c4e6f21c7..f9173c231b 100644 --- a/modules/gltf/package.json +++ b/modules/gltf/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/gltf", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the glTF format", "license": "MIT", "publishConfig": { @@ -34,10 +34,10 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/draco": "4.0.0-alpha.14", - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/textures": "4.0.0-alpha.14", + "@loaders.gl/draco": "4.0.0-alpha.15", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/textures": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/i3s/package.json b/modules/i3s/package.json index a131a63d60..2e17015007 100644 --- a/modules/i3s/package.json +++ b/modules/i3s/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/i3s", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "i3s .", "license": "MIT", "publishConfig": { @@ -33,13 +33,13 @@ "build-worker-node": "esbuild src/workers/i3s-content-worker-node.ts --outfile=dist/i3s-content-worker-node.js --platform=node --target=node16 --minify --bundle --sourcemap --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/compression": "4.0.0-alpha.14", - "@loaders.gl/draco": "4.0.0-alpha.14", - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", - "@loaders.gl/textures": "4.0.0-alpha.14", - "@loaders.gl/tiles": "4.0.0-alpha.14", + "@loaders.gl/compression": "4.0.0-alpha.15", + "@loaders.gl/draco": "4.0.0-alpha.15", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", + "@loaders.gl/textures": "4.0.0-alpha.15", + "@loaders.gl/tiles": "4.0.0-alpha.15", "@luma.gl/constants": "^8.5.4", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", diff --git a/modules/images/package.json b/modules/images/package.json index 7140cceb6b..3e146d74f3 100644 --- a/modules/images/package.json +++ b/modules/images/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/images", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders and writers for images (PNG, JPG, ...)", "license": "MIT", "publishConfig": { @@ -32,7 +32,7 @@ "build-bundle": "esbuild src/bundle.ts --outfile=dist/dist.min.js --bundle --minify --sourcemap" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5", "devDependencies": { diff --git a/modules/json/package.json b/modules/json/package.json index 6f0f83572d..6bf72bfd66 100644 --- a/modules/json/package.json +++ b/modules/json/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/json", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for JSON and streaming JSON formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ "build-worker": "esbuild src/workers/geojson-worker.ts --bundle --outfile=dist/geojson-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/kml/package.json b/modules/kml/package.json index f5e660bc56..5855491dbe 100644 --- a/modules/kml/package.json +++ b/modules/kml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/kml", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the KML format", "license": "MIT", "publishConfig": { @@ -32,9 +32,9 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@tmcw/togeojson": "^4.5.0", "@xmldom/xmldom": "^0.7.5" }, diff --git a/modules/las/package.json b/modules/las/package.json index 29c03cd97f..9002d89dba 100644 --- a/modules/las/package.json +++ b/modules/las/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/las", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the LAS and LAZ formats", "license": "MIT", "publishConfig": { @@ -41,8 +41,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "apache-arrow": "^9.0.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/loader-utils/package.json b/modules/loader-utils/package.json index 9e62fccd4c..db492cde9b 100644 --- a/modules/loader-utils/package.json +++ b/modules/loader-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/loader-utils", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for 3D graphics formats", "license": "MIT", "publishConfig": { @@ -38,7 +38,7 @@ "scripts": {}, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "@probe.gl/stats": "^4.0.2" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/math/package.json b/modules/math/package.json index 81b2b6d492..37c517f04c 100644 --- a/modules/math/package.json +++ b/modules/math/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/math", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Experimental math classes for loaders.gl", "license": "MIT", "publishConfig": { @@ -31,8 +31,8 @@ "pre-build": "echo \"Nothing to build in @loaders.gl/math\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/mvt/package.json b/modules/mvt/package.json index 5c54558c2d..e687dad8ec 100644 --- a/modules/mvt/package.json +++ b/modules/mvt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/mvt", "description": "Loader for Mapbox Vector Tiles", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -32,9 +32,9 @@ "build-worker": "esbuild src/workers/mvt-worker.ts --bundle --outfile=dist/mvt-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@math.gl/polygon": "^3.5.1", "pbf": "^3.2.1" }, diff --git a/modules/netcdf/package.json b/modules/netcdf/package.json index 324035309d..4d4ca87ba6 100644 --- a/modules/netcdf/package.json +++ b/modules/netcdf/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/netcdf", "description": "Loader for NetCDF", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -30,7 +30,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/obj/package.json b/modules/obj/package.json index 81a24ab007..6cc9c068cf 100644 --- a/modules/obj/package.json +++ b/modules/obj/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/obj", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the OBJ format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/parquet/package.json b/modules/parquet/package.json index 491051680f..def3c5f427 100644 --- a/modules/parquet/package.json +++ b/modules/parquet/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/parquet", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for Apache Parquet files", "license": "MIT", "publishConfig": { @@ -41,10 +41,10 @@ "./src/lib/wasm/load-wasm/load-wasm-node.ts": "./src/lib/wasm/load-wasm/load-wasm-browser.ts" }, "dependencies": { - "@loaders.gl/bson": "4.0.0-alpha.14", - "@loaders.gl/compression": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/bson": "4.0.0-alpha.15", + "@loaders.gl/compression": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "async-mutex": "^0.2.2", "brotli": "^1.3.2", "int53": "^0.2.4", diff --git a/modules/pcd/package.json b/modules/pcd/package.json index 2f953bbbb8..4001a0eaa4 100644 --- a/modules/pcd/package.json +++ b/modules/pcd/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/pcd", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the PCD format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-worker": "esbuild src/workers/pcd-worker.ts --bundle --outfile=dist/pcd-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/ply/package.json b/modules/ply/package.json index 72d78b0ae9..164b6e4e4f 100644 --- a/modules/ply/package.json +++ b/modules/ply/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/ply", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for the PLY format", "license": "MIT", "publishConfig": { @@ -34,8 +34,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/polyfills/package.json b/modules/polyfills/package.json index b0e569ab06..9fa2b7722a 100644 --- a/modules/polyfills/package.json +++ b/modules/polyfills/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/polyfills", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Polyfills for TextEncoder/TextDecoder", "license": "MIT", "publishConfig": { diff --git a/modules/potree/package.json b/modules/potree/package.json index 37c1b11b16..11c2472d13 100644 --- a/modules/potree/package.json +++ b/modules/potree/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/potree", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "potree loaders for large point clouds.", "license": "MIT", "publishConfig": { @@ -35,7 +35,7 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/math": "4.0.0-alpha.14", + "@loaders.gl/math": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/schema/package.json b/modules/schema/package.json index a154e8f5e6..9d3c4d38ed 100644 --- a/modules/schema/package.json +++ b/modules/schema/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/schema", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Table format APIs for JSON, CSV, etc...", "license": "MIT", "publishConfig": { diff --git a/modules/shapefile/package.json b/modules/shapefile/package.json index 414677b90f..1d96bb8c66 100644 --- a/modules/shapefile/package.json +++ b/modules/shapefile/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/shapefile", "description": "Loader for the Shapefile Format", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -37,9 +37,9 @@ "build-worker-dbf": "esbuild src/workers/dbf-worker.ts --bundle --outfile=dist/dbf-worker.js --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/gis": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/gis": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@math.gl/proj4": "^3.5.1" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/terrain/package.json b/modules/terrain/package.json index a1774c81f8..9657aae58c 100644 --- a/modules/terrain/package.json +++ b/modules/terrain/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/terrain", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loader for terrain raster formats", "license": "MIT", "publishConfig": { @@ -35,9 +35,9 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "@mapbox/martini": "^0.2.0" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/textures/package.json b/modules/textures/package.json index 51bdfbdeb7..c2958b7507 100644 --- a/modules/textures/package.json +++ b/modules/textures/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/textures", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for compressed and super compressed (basis) textures ", "license": "MIT", "publishConfig": { @@ -45,15 +45,15 @@ "build-crunch-worker": "esbuild src/workers/crunch-worker.ts --outfile=dist/crunch-worker.js --target=esnext --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "ktx-parse": "^0.0.4", "texture-compressor": "^1.0.2" }, "devDependencies": { - "@loaders.gl/polyfills": "4.0.0-alpha.14" + "@loaders.gl/polyfills": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index 947d6853a5..c2e41bfd22 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tile-converter", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Converter", "license": "MIT", "publishConfig": { @@ -45,18 +45,18 @@ "build-i3s-server-bundle": "esbuild src/i3s-server/bin/www.js --outfile=dist/esm/i3s-server/bin/i3s-server.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { - "@loaders.gl/3d-tiles": "4.0.0-alpha.14", - "@loaders.gl/crypto": "4.0.0-alpha.14", - "@loaders.gl/draco": "4.0.0-alpha.14", - "@loaders.gl/gltf": "4.0.0-alpha.14", - "@loaders.gl/i3s": "4.0.0-alpha.14", - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/polyfills": "4.0.0-alpha.14", - "@loaders.gl/textures": "4.0.0-alpha.14", - "@loaders.gl/tiles": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", - "@loaders.gl/zip": "4.0.0-alpha.14", + "@loaders.gl/3d-tiles": "4.0.0-alpha.15", + "@loaders.gl/crypto": "4.0.0-alpha.15", + "@loaders.gl/draco": "4.0.0-alpha.15", + "@loaders.gl/gltf": "4.0.0-alpha.15", + "@loaders.gl/i3s": "4.0.0-alpha.15", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/polyfills": "4.0.0-alpha.15", + "@loaders.gl/textures": "4.0.0-alpha.15", + "@loaders.gl/tiles": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", + "@loaders.gl/zip": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geoid": "^3.5.1", diff --git a/modules/tiles/package.json b/modules/tiles/package.json index fc63ad55cf..4ba4672900 100644 --- a/modules/tiles/package.json +++ b/modules/tiles/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/tiles", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Common components for different tiles loaders.", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/math": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/math": "4.0.0-alpha.15", "@math.gl/core": "^3.5.1", "@math.gl/culling": "^3.5.1", "@math.gl/geospatial": "^3.5.1", diff --git a/modules/video/package.json b/modules/video/package.json index ac0e4ff589..45db2f20f7 100644 --- a/modules/video/package.json +++ b/modules/video/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/video", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders and writers for video (MP4, WEBM, ...)", "license": "MIT", "publishConfig": { @@ -32,8 +32,8 @@ "build-bundle": "esbuild src/bundle.ts --bundle --outfile=dist/dist.min.js" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/worker-utils": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/worker-utils": "4.0.0-alpha.15", "gifshot": "^0.4.5" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" diff --git a/modules/wkt/package.json b/modules/wkt/package.json index 79baf3448f..deaf599bca 100644 --- a/modules/wkt/package.json +++ b/modules/wkt/package.json @@ -1,7 +1,7 @@ { "name": "@loaders.gl/wkt", "description": "Loader and Writer for the WKT (Well Known Text) Format", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "license": "MIT", "publishConfig": { "access": "public" @@ -35,8 +35,8 @@ "fuzzer": "^0.2.1" }, "dependencies": { - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14" + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15" }, "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" } diff --git a/modules/wms/package.json b/modules/wms/package.json index 80a5843557..b7e1e98eb7 100644 --- a/modules/wms/package.json +++ b/modules/wms/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/wms", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for the WMS (Web Map Service) standard", "license": "MIT", "publishConfig": { @@ -39,10 +39,10 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/images": "4.0.0-alpha.14", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", - "@loaders.gl/xml": "4.0.0-alpha.14", + "@loaders.gl/images": "4.0.0-alpha.15", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", + "@loaders.gl/xml": "4.0.0-alpha.15", "@turf/rewind": "^5.1.5", "deep-strict-equal": "^0.2.0", "lerc": "^4.0.1" diff --git a/modules/worker-utils/package.json b/modules/worker-utils/package.json index 21d8d56f1b..eaa4b1635b 100644 --- a/modules/worker-utils/package.json +++ b/modules/worker-utils/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/worker-utils", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Utilities for running tasks on worker threads", "license": "MIT", "publishConfig": { diff --git a/modules/xml/package.json b/modules/xml/package.json index 950effb7a5..3819c6262b 100644 --- a/modules/xml/package.json +++ b/modules/xml/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/xml", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for the XML (eXtensible Markup Language) format", "license": "MIT", "publishConfig": { @@ -33,8 +33,8 @@ }, "dependencies": { "@babel/runtime": "^7.3.1", - "@loaders.gl/loader-utils": "4.0.0-alpha.14", - "@loaders.gl/schema": "4.0.0-alpha.14", + "@loaders.gl/loader-utils": "4.0.0-alpha.15", + "@loaders.gl/schema": "4.0.0-alpha.15", "fast-xml-parser": "^4.2.5" }, "devDependencies": { diff --git a/modules/zarr/package.json b/modules/zarr/package.json index 10101b6828..5d1a19b66e 100644 --- a/modules/zarr/package.json +++ b/modules/zarr/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zarr", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Framework-independent loaders for Zarr", "license": "MIT", "publishConfig": { diff --git a/modules/zip/package.json b/modules/zip/package.json index fa92382ec4..1e4d83c3be 100644 --- a/modules/zip/package.json +++ b/modules/zip/package.json @@ -1,6 +1,6 @@ { "name": "@loaders.gl/zip", - "version": "4.0.0-alpha.14", + "version": "4.0.0-alpha.15", "description": "Zip Archive Loader", "license": "MIT", "publishConfig": { From c4e099c78f5e67fd02b3543cb638a56af68bf197 Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Fri, 21 Jul 2023 09:55:14 +0200 Subject: [PATCH 015/214] chore(tile-converter): i3s-server convert to ts (#2562) --- modules/tile-converter/package.json | 8 +- .../src/i3s-server/{app.js => app.ts} | 12 ++- .../src/i3s-server/bin/{www.js => www.ts} | 14 ++-- ...ndex-controller.js => index-controller.ts} | 10 +-- ...{slpk-controller.js => slpk-controller.ts} | 15 ++-- .../i3s-server/routes/{index.js => index.ts} | 3 +- .../routes/{slpk-router.js => slpk-router.ts} | 15 ++-- ...scene-server.js => create-scene-server.ts} | 6 +- yarn.lock | 82 +++++++++++++++++++ 9 files changed, 118 insertions(+), 47 deletions(-) rename modules/tile-converter/src/i3s-server/{app.js => app.ts} (76%) rename modules/tile-converter/src/i3s-server/bin/{www.js => www.ts} (92%) rename modules/tile-converter/src/i3s-server/controllers/{index-controller.js => index-controller.ts} (80%) rename modules/tile-converter/src/i3s-server/controllers/{slpk-controller.js => slpk-controller.ts} (70%) rename modules/tile-converter/src/i3s-server/routes/{index.js => index.ts} (91%) rename modules/tile-converter/src/i3s-server/routes/{slpk-router.js => slpk-router.ts} (65%) rename modules/tile-converter/src/i3s-server/utils/{create-scene-server.js => create-scene-server.ts} (63%) diff --git a/modules/tile-converter/package.json b/modules/tile-converter/package.json index c2e41bfd22..d98910057d 100644 --- a/modules/tile-converter/package.json +++ b/modules/tile-converter/package.json @@ -42,7 +42,7 @@ "build-bundle": "esbuild ./src/index.ts --bundle --outfile=dist/dist.min.js --platform=node --external:join-images", "build-converter-bundle": "esbuild src/converter-cli.ts --outfile=dist/converter.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", "build-slpk-extractor-bundle": "esbuild src/slpk-extractor-cli.ts --outfile=dist/slpk-extractor.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"", - "build-i3s-server-bundle": "esbuild src/i3s-server/bin/www.js --outfile=dist/esm/i3s-server/bin/i3s-server.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" + "build-i3s-server-bundle": "esbuild src/i3s-server/bin/www.ts --outfile=dist/esm/i3s-server/bin/i3s-server.min.js --platform=node --target=esnext,node14 --external:join-images --minify --bundle --define:__VERSION__=\\\"$npm_package_version\\\"" }, "dependencies": { "@loaders.gl/3d-tiles": "4.0.0-alpha.15", @@ -79,5 +79,9 @@ "join-images": "^1.1.3", "sharp": "^0.31.3" }, - "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5" + "gitHead": "c95a4ff72512668a93d9041ce8636bac09333fd5", + "devDependencies": { + "@types/express": "^4.17.17", + "@types/node": "^20.4.2" + } } diff --git a/modules/tile-converter/src/i3s-server/app.js b/modules/tile-converter/src/i3s-server/app.ts similarity index 76% rename from modules/tile-converter/src/i3s-server/app.js rename to modules/tile-converter/src/i3s-server/app.ts index 8ae6c97691..510a986bf5 100644 --- a/modules/tile-converter/src/i3s-server/app.js +++ b/modules/tile-converter/src/i3s-server/app.ts @@ -1,12 +1,12 @@ -const express = require('express'); -const path = require('path'); -const logger = require('morgan'); -const cors = require('cors'); +import express from 'express'; +import path from 'path'; +import logger from 'morgan'; +import cors from 'cors'; const indexRouter = require('./routes/index'); const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef -const app = express(); +export const app = express(); app.use(logger('dev')); app.use(express.json()); @@ -21,5 +21,3 @@ if (/\.slpk$/.test(I3S_LAYER_PATH)) { } else { app.use('/', indexRouter); } - -module.exports = app; diff --git a/modules/tile-converter/src/i3s-server/bin/www.js b/modules/tile-converter/src/i3s-server/bin/www.ts similarity index 92% rename from modules/tile-converter/src/i3s-server/bin/www.js rename to modules/tile-converter/src/i3s-server/bin/www.ts index 40bd66bc39..9485d1a2a4 100755 --- a/modules/tile-converter/src/i3s-server/bin/www.js +++ b/modules/tile-converter/src/i3s-server/bin/www.ts @@ -4,12 +4,14 @@ * Module dependencies. */ -const app = require('../app'); -const debug = require('debug')('i3s-server:server'); -const https = require('https'); -const http = require('http'); -const fs = require('fs'); -const path = require('path'); +import {app} from '../app'; +import debugFactory from 'debug'; +import https from 'https'; +import http from 'http'; +import fs from 'fs'; +import path from 'path'; + +const debug = debugFactory('i3s-server:server'); /** * Get port from environment and store in Express. diff --git a/modules/tile-converter/src/i3s-server/controllers/index-controller.js b/modules/tile-converter/src/i3s-server/controllers/index-controller.ts similarity index 80% rename from modules/tile-converter/src/i3s-server/controllers/index-controller.js rename to modules/tile-converter/src/i3s-server/controllers/index-controller.ts index 1cc225d419..61d9d150d8 100644 --- a/modules/tile-converter/src/i3s-server/controllers/index-controller.js +++ b/modules/tile-converter/src/i3s-server/controllers/index-controller.ts @@ -1,12 +1,12 @@ -const path = require('path'); -const fs = require('fs'); +import path from 'path'; +import fs from 'fs'; const {promises} = fs; const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); // eslint-disable-line no-undef -async function getFileNameByUrl(url) { +export async function getFileNameByUrl(url) { const extensions = ['json', 'bin', 'jpg', 'jpeg', 'png', 'bin.dds', 'ktx2']; for (const ext of extensions) { const fileName = `${FULL_LAYER_PATH}${url}/index.${ext}`; @@ -19,7 +19,3 @@ async function getFileNameByUrl(url) { } return null; } - -module.exports = { - getFileNameByUrl -}; diff --git a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts similarity index 70% rename from modules/tile-converter/src/i3s-server/controllers/slpk-controller.js rename to modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts index 6b2e9317da..3ae6d0f0e1 100644 --- a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.js +++ b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts @@ -1,7 +1,7 @@ -require('@loaders.gl/polyfills'); -const {parseSLPK} = require('@loaders.gl/i3s'); -const path = require('path'); -const {FileHandleProvider} = require('@loaders.gl/tile-converter'); +import '@loaders.gl/polyfills'; +import {parseSLPK} from '@loaders.gl/i3s'; +import {FileHandleProvider} from '@loaders.gl/tile-converter'; +import path from 'path'; let slpkArchive; @@ -14,7 +14,7 @@ const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); // eslint-disa loadArchive(FULL_LAYER_PATH); -async function getFileByUrl(url) { +export async function getFileByUrl(url) { const trimmedPath = /^\/?(.*)\/?$/.exec(url); let uncompressedFile; if (trimmedPath) { @@ -24,8 +24,3 @@ async function getFileByUrl(url) { } return uncompressedFile; } - -module.exports = { - loadArchive, - getFileByUrl -}; diff --git a/modules/tile-converter/src/i3s-server/routes/index.js b/modules/tile-converter/src/i3s-server/routes/index.ts similarity index 91% rename from modules/tile-converter/src/i3s-server/routes/index.js rename to modules/tile-converter/src/i3s-server/routes/index.ts index 4d289caf16..3352663fac 100644 --- a/modules/tile-converter/src/i3s-server/routes/index.js +++ b/modules/tile-converter/src/i3s-server/routes/index.ts @@ -1,4 +1,5 @@ -const express = require('express'); +import express from 'express'; + const router = express.Router(); const {getFileNameByUrl} = require('../controllers/index-controller'); diff --git a/modules/tile-converter/src/i3s-server/routes/slpk-router.js b/modules/tile-converter/src/i3s-server/routes/slpk-router.ts similarity index 65% rename from modules/tile-converter/src/i3s-server/routes/slpk-router.js rename to modules/tile-converter/src/i3s-server/routes/slpk-router.ts index e530b49c53..05a89529e6 100644 --- a/modules/tile-converter/src/i3s-server/routes/slpk-router.js +++ b/modules/tile-converter/src/i3s-server/routes/slpk-router.ts @@ -1,8 +1,8 @@ -const express = require('express'); -const {getFileByUrl} = require('../controllers/slpk-controller'); -const createSceneServer = require('../utils/create-scene-server'); +import express from 'express'; +import {getFileByUrl} from '../controllers/slpk-controller'; +import {createSceneServer} from '../utils/create-scene-server'; -const sceneServerRouter = express.Router(); +export const sceneServerRouter = express.Router(); sceneServerRouter.get('*', async function (req, res, next) { const file = await getFileByUrl('/'); if (file) { @@ -15,7 +15,7 @@ sceneServerRouter.get('*', async function (req, res, next) { } }); -const router = express.Router(); +export const router = express.Router(); router.get('*', async function (req, res, next) { console.log(req.path); const file = await getFileByUrl(req.path); @@ -26,8 +26,3 @@ router.get('*', async function (req, res, next) { res.send('File not found'); } }); - -module.exports = { - sceneServerRouter, - router -}; diff --git a/modules/tile-converter/src/i3s-server/utils/create-scene-server.js b/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts similarity index 63% rename from modules/tile-converter/src/i3s-server/utils/create-scene-server.js rename to modules/tile-converter/src/i3s-server/utils/create-scene-server.ts index fc67c8831b..3110a8c818 100644 --- a/modules/tile-converter/src/i3s-server/utils/create-scene-server.js +++ b/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts @@ -1,6 +1,6 @@ -const {v4: uuidv4} = require('uuid'); +import {v4 as uuidv4} from 'uuid'; -const createSceneServer = (name, layer) => { +export const createSceneServer = (name, layer) => { return { serviceItemId: uuidv4().replace(/-/gi, ''), serviceName: name, @@ -11,5 +11,3 @@ const createSceneServer = (name, layer) => { layers: [layer] }; }; - -module.exports = createSceneServer; diff --git a/yarn.lock b/yarn.lock index f7e201f458..183f2e5deb 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2454,6 +2454,14 @@ "@turf/invariant" "^5.1.5" "@turf/meta" "^5.1.5" +"@types/body-parser@*": + version "1.19.2" + resolved "https://registry.yarnpkg.com/@types/body-parser/-/body-parser-1.19.2.tgz#aea2059e28b7658639081347ac4fab3de166e6f0" + integrity sha512-ALYone6pm6QmwZoAgeyNksccT9Q4AWZQ6PvfwR37GT6r6FWUPguq6sUmNGSMV2Wr761oQoBxwGGa6DR5o1DC9g== + dependencies: + "@types/connect" "*" + "@types/node" "*" + "@types/brotli@^1.3.0": version "1.3.1" resolved "https://registry.yarnpkg.com/@types/brotli/-/brotli-1.3.1.tgz#65dc6c69bb9f4159677032f60e81ffc09faf1fce" @@ -2478,6 +2486,13 @@ resolved "https://registry.yarnpkg.com/@types/command-line-usage/-/command-line-usage-5.0.2.tgz#ba5e3f6ae5a2009d466679cc431b50635bf1a064" integrity sha512-n7RlEEJ+4x4TS7ZQddTmNSxP+zziEG0TNsMfiRIxcIVXt71ENJ9ojeXmGO3wPoTdn7pJcU2xc3CJYMktNT6DPg== +"@types/connect@*": + version "3.4.35" + resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.35.tgz#5fcf6ae445e4021d1fc2219a4873cc73a3bb2ad1" + integrity sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ== + dependencies: + "@types/node" "*" + "@types/crypto-js@^4.0.2": version "4.1.1" resolved "https://registry.yarnpkg.com/@types/crypto-js/-/crypto-js-4.1.1.tgz#602859584cecc91894eb23a4892f38cfa927890d" @@ -2488,6 +2503,26 @@ resolved "https://registry.yarnpkg.com/@types/emscripten/-/emscripten-1.39.6.tgz#698b90fe60d44acf93c31064218fbea93fbfd85a" integrity sha512-H90aoynNhhkQP6DRweEjJp5vfUVdIj7tdPLsu7pq89vODD/lcugKfZOsfgwpvM6XUewEp2N5dCg1Uf3Qe55Dcg== +"@types/express-serve-static-core@^4.17.33": + version "4.17.35" + resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.17.35.tgz#c95dd4424f0d32e525d23812aa8ab8e4d3906c4f" + integrity sha512-wALWQwrgiB2AWTT91CB62b6Yt0sNHpznUXeZEcnPU3DRdlDIz74x8Qg1UUYKSVFi+va5vKOLYRBI1bRKiLLKIg== + dependencies: + "@types/node" "*" + "@types/qs" "*" + "@types/range-parser" "*" + "@types/send" "*" + +"@types/express@^4.17.17": + version "4.17.17" + resolved "https://registry.yarnpkg.com/@types/express/-/express-4.17.17.tgz#01d5437f6ef9cfa8668e616e13c2f2ac9a491ae4" + integrity sha512-Q4FmmuLGBG58btUnfS1c1r/NQdlp3DMfGDGig8WhfpA2YRUtEkxAjkZb0yvplJGYdF1fsQ81iMDcH24sSCNC/Q== + dependencies: + "@types/body-parser" "*" + "@types/express-serve-static-core" "^4.17.33" + "@types/qs" "*" + "@types/serve-static" "*" + "@types/flatbuffers@*": version "1.10.0" resolved "https://registry.yarnpkg.com/@types/flatbuffers/-/flatbuffers-1.10.0.tgz#aa74e30ffdc86445f2f060e1808fc9d56b5603ba" @@ -2519,6 +2554,11 @@ resolved "https://registry.yarnpkg.com/@types/hammerjs/-/hammerjs-2.0.41.tgz#f6ecf57d1b12d2befcce00e928a6a097c22980aa" integrity sha512-ewXv/ceBaJprikMcxCmWU1FKyMAQ2X7a9Gtmzw8fcg2kIePI1crERDM818W+XYrxqdBBOdlf2rm137bU+BltCA== +"@types/http-errors@*": + version "2.0.1" + resolved "https://registry.yarnpkg.com/@types/http-errors/-/http-errors-2.0.1.tgz#20172f9578b225f6c7da63446f56d4ce108d5a65" + integrity sha512-/K3ds8TRAfBvi5vfjuz8y6+GiAYBZ0x4tXv1Av6CWBWn0IlADc+ZX9pMq7oU0fNQPnBwIZl3rmeLp6SBApbxSQ== + "@types/json-schema@^7.0.5", "@types/json-schema@^7.0.7": version "7.0.11" resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.11.tgz#d421b6c527a3037f7c84433fd2c4229e016863d3" @@ -2536,6 +2576,16 @@ dependencies: "@types/unist" "*" +"@types/mime@*": + version "3.0.1" + resolved "https://registry.yarnpkg.com/@types/mime/-/mime-3.0.1.tgz#5f8f2bca0a5863cb69bc0b0acd88c96cb1d4ae10" + integrity sha512-Y4XFY5VJAuw0FgAqPNd6NNoV44jbq9Bz2L7Rh/J6jLTiHBSBJa9fxqQIvkIld4GsoDOcCbvzOUAbLPsSKKg+uA== + +"@types/mime@^1": + version "1.3.2" + resolved "https://registry.yarnpkg.com/@types/mime/-/mime-1.3.2.tgz#93e25bf9ee75fe0fd80b594bc4feb0e862111b5a" + integrity sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw== + "@types/minimatch@*": version "5.1.2" resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-5.1.2.tgz#07508b45797cb81ec3f273011b054cd0755eddca" @@ -2573,6 +2623,11 @@ resolved "https://registry.yarnpkg.com/@types/node/-/node-17.0.45.tgz#2c0fafd78705e7a18b7906b5201a522719dc5190" integrity sha512-w+tIMs3rq2afQdsPJlODhoUEKzFP1ayaoyl1CcnwtIlsVe7K7bA1NGm4s3PraqTLlXnbIN84zuBlxBWo1u9BLw== +"@types/node@^20.4.2": + version "20.4.2" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.4.2.tgz#129cc9ae69f93824f92fac653eebfb4812ab4af9" + integrity sha512-Dd0BYtWgnWJKwO1jkmTrzofjK2QXXcai0dmtzvIBhcA+RsG5h8R3xlyta0kGOZRNfL9GuRtb1knmPEhQrePCEw== + "@types/normalize-package-data@^2.4.0": version "2.4.1" resolved "https://registry.yarnpkg.com/@types/normalize-package-data/-/normalize-package-data-2.4.1.tgz#d3357479a0fdfdd5907fe67e17e0a85c906e1301" @@ -2615,6 +2670,33 @@ resolved "https://registry.yarnpkg.com/@types/q/-/q-1.5.5.tgz#75a2a8e7d8ab4b230414505d92335d1dcb53a6df" integrity sha512-L28j2FcJfSZOnL1WBjDYp2vUHCeIFlyYI/53EwD/rKUBQ7MtUUfbQWiyKJGpcnv4/WgrhWsFKrcPstcAt/J0tQ== +"@types/qs@*": + version "6.9.7" + resolved "https://registry.yarnpkg.com/@types/qs/-/qs-6.9.7.tgz#63bb7d067db107cc1e457c303bc25d511febf6cb" + integrity sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw== + +"@types/range-parser@*": + version "1.2.4" + resolved "https://registry.yarnpkg.com/@types/range-parser/-/range-parser-1.2.4.tgz#cd667bcfdd025213aafb7ca5915a932590acdcdc" + integrity sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw== + +"@types/send@*": + version "0.17.1" + resolved "https://registry.yarnpkg.com/@types/send/-/send-0.17.1.tgz#ed4932b8a2a805f1fe362a70f4e62d0ac994e301" + integrity sha512-Cwo8LE/0rnvX7kIIa3QHCkcuF21c05Ayb0ZfxPiv0W8VRiZiNW/WuRupHKpqqGVGf7SUA44QSOUKaEd9lIrd/Q== + dependencies: + "@types/mime" "^1" + "@types/node" "*" + +"@types/serve-static@*": + version "1.15.2" + resolved "https://registry.yarnpkg.com/@types/serve-static/-/serve-static-1.15.2.tgz#3e5419ecd1e40e7405d34093f10befb43f63381a" + integrity sha512-J2LqtvFYCzaj8pVYKw8klQXrLLk7TBZmQ4ShlcdkELFKGwGMfevMLneMMRkMgZxotOD9wg497LpC7O8PcvAmfw== + dependencies: + "@types/http-errors" "*" + "@types/mime" "*" + "@types/node" "*" + "@types/sql.js@^1.4.2": version "1.4.4" resolved "https://registry.yarnpkg.com/@types/sql.js/-/sql.js-1.4.4.tgz#1c4df5045f08a518d48cda6bbb4f5dacc44c503c" From 1f9901861922f4ce2ff4d9f028f81b5b7cdf1818 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 24 Jul 2023 08:46:14 -0400 Subject: [PATCH 016/214] chore(deps): bump semver from 5.7.1 to 5.7.2 (#2545) --- yarn.lock | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/yarn.lock b/yarn.lock index 183f2e5deb..6699058e44 100644 --- a/yarn.lock +++ b/yarn.lock @@ -11043,19 +11043,19 @@ selfsigned@^1.10.8: node-forge "^0.10.0" "semver@2 || 3 || 4 || 5", "semver@2.x || 3.x || 4 || 5", semver@^5.4.1, semver@^5.5.0, semver@^5.5.1, semver@^5.6.0, semver@^5.7.0, semver@^5.7.1: - version "5.7.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" - integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== + version "5.7.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" + integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== semver@^6.0.0, semver@^6.1.1, semver@^6.1.2, semver@^6.2.0, semver@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" - integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== semver@^7.2.1, semver@^7.3.4, semver@^7.3.5: - version "7.5.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.0.tgz#ed8c5dc8efb6c629c88b23d41dc9bf40c1d96cd0" - integrity sha512-+XC0AD/R7Q2mPSRuy2Id0+CGTZ98+8f+KvwirxOKIEyid+XSx6HbC63p+O4IndTHuX5Z+JxQ0TghCkO5Cg/2HA== + version "7.5.4" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" + integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA== dependencies: lru-cache "^6.0.0" From d02c8ffd0fd475f7f751443ab819db12feab4ae6 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 24 Jul 2023 08:46:27 -0400 Subject: [PATCH 017/214] chore(deps): bump semver in /test/apps/typescript-test (#2544) --- test/apps/typescript-test/yarn.lock | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/test/apps/typescript-test/yarn.lock b/test/apps/typescript-test/yarn.lock index a06c832811..36392917ee 100644 --- a/test/apps/typescript-test/yarn.lock +++ b/test/apps/typescript-test/yarn.lock @@ -1445,19 +1445,19 @@ semver-diff@^3.1.1: semver "^6.3.0" semver@^5.7.1: - version "5.7.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" - integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== + version "5.7.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" + integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== semver@^6.0.0, semver@^6.2.0, semver@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" - integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== semver@^7.3.4, semver@^7.3.5: - version "7.3.5" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.5.tgz#0b621c879348d8998e4b0e4be94b3f12e6018ef7" - integrity sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ== + version "7.5.4" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" + integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA== dependencies: lru-cache "^6.0.0" From 1dabc1494cd868ad32aa176ca429ee0a7e068a6e Mon Sep 17 00:00:00 2001 From: Viktor Belomestnov Date: Mon, 24 Jul 2023 16:34:55 +0200 Subject: [PATCH 018/214] chore(tile-converter): i3s-server tests (#2563) --- modules/i3s/src/index.ts | 1 - modules/i3s/src/lib/parsers/constants.ts | 29 ++-- .../lib/parsers/parse-slpk/slpk-archieve.ts | 2 +- modules/i3s/src/types.ts | 65 +++---- modules/tile-converter/src/i3s-server/app.ts | 6 +- .../tile-converter/src/i3s-server/bin/www.ts | 94 ++-------- .../controllers/index-controller.ts | 11 +- .../i3s-server/controllers/slpk-controller.ts | 21 ++- .../i3s-server/utils/create-scene-server.ts | 9 +- .../src/i3s-server/utils/server-utils.ts | 70 ++++++++ .../Frankfurt-md-2/SceneServer/index.json | 1 + .../SceneServer/layers/0/index.json | 1 + .../layers/0/nodepages/0/index.json | 1 + .../layers/0/nodes/1/geometries/0/index.bin | Bin 0 -> 551256 bytes .../layers/0/nodes/1/geometries/1/index.bin | Bin 0 -> 207942 bytes .../SceneServer/layers/0/nodes/1/index.json | 1 + .../layers/0/nodes/1/shared/index.json | 1 + .../layers/0/nodes/1/textures/0/index.jpg | Bin 0 -> 45225 bytes .../layers/0/nodes/1/textures/1/index.ktx2 | Bin 0 -> 2097152 bytes .../layers/0/nodes/root/index.json | 1 + .../controllers/index-controller.spec.ts | 33 ++++ .../controllers/slpk-controller.spec.ts | 45 +++++ .../utils/create-scene-server.spec.ts | 164 ++++++++++++++++++ .../i3s-server/utils/server-utils.spec.ts | 33 ++++ modules/tile-converter/test/index.js | 5 + 25 files changed, 443 insertions(+), 151 deletions(-) create mode 100644 modules/tile-converter/src/i3s-server/utils/server-utils.ts create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/index.json create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/index.json create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodepages/0/index.json create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/0/index.bin create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/1/index.bin create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/index.json create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/shared/index.json create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/textures/0/index.jpg create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/textures/1/index.ktx2 create mode 100644 modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/root/index.json create mode 100644 modules/tile-converter/test/i3s-server/controllers/index-controller.spec.ts create mode 100644 modules/tile-converter/test/i3s-server/controllers/slpk-controller.spec.ts create mode 100644 modules/tile-converter/test/i3s-server/utils/create-scene-server.spec.ts create mode 100644 modules/tile-converter/test/i3s-server/utils/server-utils.spec.ts diff --git a/modules/i3s/src/index.ts b/modules/i3s/src/index.ts index 78ea82b813..9f5930a34f 100644 --- a/modules/i3s/src/index.ts +++ b/modules/i3s/src/index.ts @@ -29,7 +29,6 @@ export type { Histogram, ValueCount, BuildingSceneSublayer, - DATA_TYPE, OperationalLayer, TextureSetDefinitionFormats } from './types'; diff --git a/modules/i3s/src/lib/parsers/constants.ts b/modules/i3s/src/lib/parsers/constants.ts index 7ff2377c82..102c690b5d 100644 --- a/modules/i3s/src/lib/parsers/constants.ts +++ b/modules/i3s/src/lib/parsers/constants.ts @@ -1,17 +1,16 @@ import GL from '@luma.gl/constants'; -import {DATA_TYPE} from '../../types'; export function getConstructorForDataFormat(dataType: string) { switch (dataType) { - case DATA_TYPE.UInt8: + case 'UInt8': return Uint8Array; - case DATA_TYPE.UInt16: + case 'UInt16': return Uint16Array; - case DATA_TYPE.UInt32: + case 'UInt32': return Uint32Array; - case DATA_TYPE.Float32: + case 'Float32': return Float32Array; - case DATA_TYPE.UInt64: + case 'UInt64': return Float64Array; default: throw new Error(`parse i3s tile content: unknown type of data: ${dataType}`); @@ -32,18 +31,18 @@ export const GL_TYPE_MAP: {[key: string]: number} = { */ export function sizeOf(dataType: string): number { switch (dataType) { - case DATA_TYPE.UInt8: + case 'UInt8': return 1; - case DATA_TYPE.UInt16: - case DATA_TYPE.Int16: + case 'UInt16': + case 'Int16': return 2; - case DATA_TYPE.UInt32: - case DATA_TYPE.Int32: - case DATA_TYPE.Float32: + case 'UInt32': + case 'Int32': + case 'Float32': return 4; - case DATA_TYPE.UInt64: - case DATA_TYPE.Int64: - case DATA_TYPE.Float64: + case 'UInt64': + case 'Int64': + case 'Float64': return 8; default: throw new Error(`parse i3s tile content: unknown size of data: ${dataType}`); diff --git a/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts b/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts index e588e8dd09..a84dacf1a1 100644 --- a/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts +++ b/modules/i3s/src/lib/parsers/parse-slpk/slpk-archieve.ts @@ -26,7 +26,7 @@ const PATH_DESCRIPTIONS: {test: RegExp; extensions: string[]}[] = [ extensions: ['.json.gz'] }, { - test: /^nodes\/\d+$/, + test: /^nodes\/(\d+|root)$/, extensions: ['/3dNodeIndexDocument.json.gz'] }, { diff --git a/modules/i3s/src/types.ts b/modules/i3s/src/types.ts index ef74a7ce54..f0e840ffb7 100644 --- a/modules/i3s/src/types.ts +++ b/modules/i3s/src/types.ts @@ -2,18 +2,6 @@ import type {Matrix4, Quaternion, Vector3} from '@math.gl/core'; import type {TypedArray, MeshAttribute, TextureLevel} from '@loaders.gl/schema'; import {Tile3D, Tileset3D} from '@loaders.gl/tiles'; -export enum DATA_TYPE { - UInt8 = 'UInt8', - UInt16 = 'UInt16', - UInt32 = 'UInt32', - UInt64 = 'UInt64', - Int16 = 'Int16', - Int32 = 'Int32', - Int64 = 'Int64', - Float32 = 'Float32', - Float64 = 'Float64' -} - export type COLOR = [number, number, number, number]; /** @@ -690,20 +678,24 @@ type Domain = { * spec - https://github.com/Esri/i3s-spec/blob/master/docs/1.8/store.cmn.md */ type Store = { - id: string | number; + id?: string | number; profile: string; version: number | string; - resourcePattern: string[]; - rootNode: string; - extent: number[]; - indexCRS: string; - vertexCRS: string; - normalReferenceFrame: string; - attributeEncoding: string; - textureEncoding: string[]; - lodType: string; - lodModel: string; + resourcePattern?: string[]; + rootNode?: string; + extent?: number[]; + indexCRS?: string; + vertexCRS?: string; + normalReferenceFrame?: string; + lodType?: string; + lodModel?: string; defaultGeometrySchema: DefaultGeometrySchema; + nidEncoding?: string; + textureEncoding?: string[]; + featureEncoding?: string; + geometryEncoding?: string; + attributeEncoding?: string; + indexingScheme?: string; }; /** * Spec - https://github.com/Esri/i3s-spec/blob/master/docs/1.8/defaultGeometrySchema.cmn.md @@ -726,15 +718,15 @@ type DefaultGeometrySchema = { export type HeaderAttribute = { property: HeaderAttributeProperty.vertexCount | HeaderAttributeProperty.featureCount | string; type: - | DATA_TYPE.UInt8 - | DATA_TYPE.UInt16 - | DATA_TYPE.UInt32 - | DATA_TYPE.UInt64 - | DATA_TYPE.Int16 - | DATA_TYPE.Int32 - | DATA_TYPE.Int64 - | DATA_TYPE.Float32 - | DATA_TYPE.Float64; + | 'UInt8' + | 'UInt16' + | 'UInt32' + | 'UInt64' + | 'Int16' + | 'Int32' + | 'Int64' + | 'Float32' + | 'Float64'; }; export enum HeaderAttributeProperty { vertexCount = 'vertexCount', @@ -749,14 +741,7 @@ export type VertexAttribute = { }; export type GeometryAttribute = { byteOffset?: number; - valueType: - | DATA_TYPE.UInt8 - | DATA_TYPE.UInt16 - | DATA_TYPE.Int16 - | DATA_TYPE.Int32 - | DATA_TYPE.Int64 - | DATA_TYPE.Float32 - | DATA_TYPE.Float64; + valueType: 'UInt8' | 'UInt16' | 'Int16' | 'Int32' | 'Int64' | 'Float32' | 'Float64'; valuesPerElement: number; }; export type I3SMeshAttributes = { diff --git a/modules/tile-converter/src/i3s-server/app.ts b/modules/tile-converter/src/i3s-server/app.ts index 510a986bf5..24753af161 100644 --- a/modules/tile-converter/src/i3s-server/app.ts +++ b/modules/tile-converter/src/i3s-server/app.ts @@ -2,10 +2,14 @@ import express from 'express'; import path from 'path'; import logger from 'morgan'; import cors from 'cors'; +import {loadArchive} from './controllers/slpk-controller'; + +const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef +const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); // eslint-disable-line no-undef +loadArchive(FULL_LAYER_PATH); const indexRouter = require('./routes/index'); -const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef export const app = express(); app.use(logger('dev')); diff --git a/modules/tile-converter/src/i3s-server/bin/www.ts b/modules/tile-converter/src/i3s-server/bin/www.ts index 9485d1a2a4..ad98d2ec35 100755 --- a/modules/tile-converter/src/i3s-server/bin/www.ts +++ b/modules/tile-converter/src/i3s-server/bin/www.ts @@ -1,29 +1,25 @@ #!/usr/bin/env node -/** - * Module dependencies. - */ - import {app} from '../app'; -import debugFactory from 'debug'; import https from 'https'; import http from 'http'; import fs from 'fs'; import path from 'path'; +import {formErrorHandler, formListeningHandler, normalizePort} from '../utils/server-utils'; -const debug = debugFactory('i3s-server:server'); - -/** - * Get port from environment and store in Express. - */ - -const httpPort = normalizePort(process.env.PORT || '80'); // eslint-disable-line no-process-env, no-undef -const httpsPort = normalizePort(process.env.HTTPS_PORT || '443'); // eslint-disable-line no-process-env, no-undef - -/** - * Create HTTP server. - */ +/** Get port from environment and store in Express. */ +const httpPort = normalizePort(process.env.PORT || '80'); +if (httpPort === false) { + console.error(`Incorrect HTTP port`); + process.exit(1); +} +const httpsPort = normalizePort(process.env.HTTPS_PORT || '443'); +if (httpsPort === false) { + console.error(`Incorrect HTTPs port`); + process.exit(1); +} +/** Create HTTP server. */ const options = { key: fs.readFileSync(path.join(__dirname, '../certs/key.pem')), cert: fs.readFileSync(path.join(__dirname, '../certs/cert.pem')) @@ -32,10 +28,7 @@ const options = { const httpServer = http.createServer(app); const httpsServer = https.createServer(options, app); -/** - * Listen on provided port, on all network interfaces. - */ - +/** Listen on provided port, on all network interfaces. */ httpServer.listen(httpPort); httpServer.on('error', formErrorHandler(httpPort)); httpServer.on('listening', formListeningHandler(httpServer)); @@ -43,62 +36,3 @@ httpServer.on('listening', formListeningHandler(httpServer)); httpsServer.listen(httpsPort); httpsServer.on('error', formErrorHandler(httpsPort)); httpsServer.on('listening', formListeningHandler(httpsServer)); - -/** - * Normalize a port into a number, string, or false. - */ - -function normalizePort(val) { - const chkPort = parseInt(val, 10); - - if (isNaN(chkPort)) { - // named pipe - return val; - } - - if (chkPort >= 0) { - // port number - return chkPort; - } - - return false; -} - -/** - * Event listener for HTTP/HTTPS server "error" event. - */ - -function formErrorHandler(optionalPort) { - return function onError(error) { - if (error.syscall !== 'listen') { - throw error; - } - - const bind = typeof global.port === 'string' ? `Pipe ${optionalPort}` : `Port ${optionalPort}`; - - // handle specific listen errors with friendly messages - switch (error.code) { - case 'EACCES': - console.error(`${bind} requires elevated privileges`); // eslint-disable-line no-console, no-undef - process.exit(1); // eslint-disable-line no-process-exit, no-undef - break; - case 'EADDRINUSE': - console.error(`${bind} is already in use`); // eslint-disable-line no-console, no-undef - process.exit(1); // eslint-disable-line no-process-exit, no-undef - break; - default: - throw error; - } - }; -} - -/** - * Event listener for HTTP/HTTPS server "listening" event. - */ -function formListeningHandler(optionalServer) { - return function onListening() { - const addr = optionalServer.address(); - const bind = typeof addr === 'string' ? `pipe ${addr}` : `port ${addr.port}`; - debug(`Listening on ${bind}`); - }; -} diff --git a/modules/tile-converter/src/i3s-server/controllers/index-controller.ts b/modules/tile-converter/src/i3s-server/controllers/index-controller.ts index 61d9d150d8..fdc4ade617 100644 --- a/modules/tile-converter/src/i3s-server/controllers/index-controller.ts +++ b/modules/tile-converter/src/i3s-server/controllers/index-controller.ts @@ -3,11 +3,16 @@ import fs from 'fs'; const {promises} = fs; -const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef -const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); // eslint-disable-line no-undef +const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; -export async function getFileNameByUrl(url) { +/** + * Get local file name by input HTTP URL + * @param url - I3S HTTP service url + * @returns - local file name + */ +export async function getFileNameByUrl(url: string): Promise { const extensions = ['json', 'bin', 'jpg', 'jpeg', 'png', 'bin.dds', 'ktx2']; + const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); for (const ext of extensions) { const fileName = `${FULL_LAYER_PATH}${url}/index.${ext}`; try { diff --git a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts index 3ae6d0f0e1..679ad70e25 100644 --- a/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts +++ b/modules/tile-converter/src/i3s-server/controllers/slpk-controller.ts @@ -1,22 +1,25 @@ import '@loaders.gl/polyfills'; import {parseSLPK} from '@loaders.gl/i3s'; import {FileHandleProvider} from '@loaders.gl/tile-converter'; -import path from 'path'; let slpkArchive; -const loadArchive = async (fullLayerPath) => { +/** + * Open SLPK file for reading and load HASH file + * @param fullLayerPath - full path to SLPK file + */ +export const loadArchive = async (fullLayerPath: string): Promise => { slpkArchive = await parseSLPK(await FileHandleProvider.from(fullLayerPath)); }; -const I3S_LAYER_PATH = process.env.I3sLayerPath || ''; // eslint-disable-line no-process-env, no-undef -const FULL_LAYER_PATH = path.join(process.cwd(), I3S_LAYER_PATH); // eslint-disable-line no-undef - -loadArchive(FULL_LAYER_PATH); - -export async function getFileByUrl(url) { +/** + * Get a file from SLPK + * @param url - I3S HTTP URL + * @returns - file content + */ +export async function getFileByUrl(url: string) { const trimmedPath = /^\/?(.*)\/?$/.exec(url); - let uncompressedFile; + let uncompressedFile: Buffer | null = null; if (trimmedPath) { try { uncompressedFile = Buffer.from(await slpkArchive.getFile(trimmedPath[1], 'http')); diff --git a/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts b/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts index 3110a8c818..40675fe69f 100644 --- a/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts +++ b/modules/tile-converter/src/i3s-server/utils/create-scene-server.ts @@ -1,6 +1,13 @@ +import {SceneLayer3D} from '@loaders.gl/i3s'; import {v4 as uuidv4} from 'uuid'; -export const createSceneServer = (name, layer) => { +/** + * Create `/SceneServer` response + * @param name - service name, custom user-friendly name of the service + * @param layer - I3S layer JSON + * @returns reponse JSON for `/SceneServer` route + */ +export const createSceneServer = (name: string, layer: SceneLayer3D) => { return { serviceItemId: uuidv4().replace(/-/gi, ''), serviceName: name, diff --git a/modules/tile-converter/src/i3s-server/utils/server-utils.ts b/modules/tile-converter/src/i3s-server/utils/server-utils.ts new file mode 100644 index 0000000000..e3b2acbb66 --- /dev/null +++ b/modules/tile-converter/src/i3s-server/utils/server-utils.ts @@ -0,0 +1,70 @@ +import type {Server as HttpsServer} from 'https'; +import type {Server as HttpServer} from 'http'; + +import debugFactory from 'debug'; +const debug = debugFactory('i3s-server:server'); + +/** + * Normalize a port into a number, string, or false. + * @param val - port value from env variables + * @returns - `number` for port, `string` for a named pipe, or `false` if the port number is not correct + */ +export function normalizePort(val: string): number | string | false { + const chkPort = parseInt(val, 10); + + if (Number.isNaN(chkPort)) { + // named pipe + return val; + } + + if (chkPort >= 0) { + // port number + return chkPort; + } + + return false; +} + +/** + * Event listener creator for HTTP/HTTPS server "error" event. + * @param optionalPort - the port/named pipe the server is started on + * @return callback to handle server errors + */ +export function formErrorHandler( + optionalPort: string | number +): (error: NodeJS.ErrnoException) => void { + return function onError(error: NodeJS.ErrnoException) { + if (error.syscall !== 'listen') { + throw error; + } + + const bind = typeof optionalPort === 'string' ? `Pipe ${optionalPort}` : `Port ${optionalPort}`; + + // handle specific listen errors with friendly messages + switch (error.code) { + case 'EACCES': + console.error(`${bind} requires elevated privileges`); // eslint-disable-line no-console, no-undef + process.exit(1); // eslint-disable-line no-process-exit, no-undef + break; + case 'EADDRINUSE': + console.error(`${bind} is already in use`); // eslint-disable-line no-console, no-undef + process.exit(1); // eslint-disable-line no-process-exit, no-undef + break; + default: + throw error; + } + }; +} + +/** + * Event listener for HTTP/HTTPS server "listening" event. + * @param optionalServer - http or https NodeJS server + * @return callback that is triggered when the server has started + */ +export function formListeningHandler(optionalServer: HttpsServer | HttpServer): () => void { + return function onListening() { + const addr = optionalServer.address(); + const bind = typeof addr === 'string' ? `pipe ${addr}` : `port ${addr?.port}`; + debug(`Listening on ${bind}`); + }; +} diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/index.json b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/index.json new file mode 100644 index 0000000000..58ff5b1405 --- /dev/null +++ b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/index.json @@ -0,0 +1 @@ +{"serviceItemId":"82a6203820c740e88bbafb94132f6b03","serviceName":"Frankfurt-md-2","name":"Frankfurt-md-2","currentVersion":10.7,"serviceVersion":"1.8","supportedBindings":["REST"],"layers":[{"version":"{1F518F6A-9621-4476-8194-DD034FF34614}","id":0,"name":"Frankfurt-md-2","href":"./layers/0","layerType":"IntegratedMesh","spatialReference":{"wkid":4326,"latestWkid":4326,"vcsWkid":5773,"latestVcsWkid":5773},"capabilities":["View","Query"],"store":{"id":"{F23F2D74-EE90-4C42-AFF3-56CE04640CE6}","profile":"meshpyramids","version":"1.8","resourcePattern":["3dNodeIndexDocument","Attributes","SharedResource","Geometry"],"rootNode":"./nodes/root","extent":[8.638486093840738,50.10162577927056,8.714439334060673,50.11520548628318],"indexCRS":"http://www.opengis.net/def/crs/EPSG/0/4326","vertexCRS":"http://www.opengis.net/def/crs/EPSG/0/4326","normalReferenceFrame":"east-north-up","attributeEncoding":"application/octet-stream; version=1.6","textureEncoding":["image/jpeg","image/ktx2"],"lodType":"MeshPyramid","lodModel":"node-switching","defaultGeometrySchema":{"geometryType":"triangles","header":[{"property":"vertexCount","type":"UInt32"},{"property":"featureCount","type":"UInt32"}],"topology":"PerAttributeArray","ordering":["position","normal","uv0","color"],"vertexAttributes":{"position":{"valueType":"Float32","valuesPerElement":3},"normal":{"valueType":"Float32","valuesPerElement":3},"uv0":{"valueType":"Float32","valuesPerElement":2},"color":{"valueType":"UInt8","valuesPerElement":4}},"featureAttributeOrder":["id","faceRange"],"featureAttributes":{"id":{"valueType":"UInt64","valuesPerElement":1},"faceRange":{"valueType":"UInt32","valuesPerElement":2}}}},"fullExtent":{"xmin":8.638486093840738,"ymin":50.10162577927056,"xmax":8.714439334060673,"ymax":50.11520548628318,"zmin":-4668.025435449932,"zmax":5048.2198269797045},"heightModelInfo":{"heightModel":"gravity_related_height","vertCRS":"EGM96_Geoid","heightUnit":"meter"},"nodePages":{"nodesPerPage":64,"lodSelectionMetricType":"maxScreenThresholdSQ"},"materialDefinitions":[{"doubleSided":true,"emissiveFactor":[255,255,255],"alphaMode":"opaque","pbrMetallicRoughness":{"roughnessFactor":1,"metallicFactor":1,"baseColorTexture":{"textureSetDefinitionId":0}}}],"textureSetDefinitions":[{"formats":[{"name":"0","format":"jpg"},{"name":"1","format":"ktx2"}]},{"formats":[{"name":"0","format":"jpg"},{"name":"1","format":"ktx2"}],"atlas":true}],"geometryDefinitions":[{"geometryBuffers":[{"offset":8,"position":{"type":"Float32","component":3},"normal":{"type":"Float32","component":3},"uv0":{"type":"Float32","component":2},"color":{"type":"UInt8","component":4},"featureId":{"binding":"per-feature","type":"UInt64","component":1},"faceRange":{"binding":"per-feature","type":"UInt32","component":2}},{"compressedAttributes":{"encoding":"draco","attributes":["position","normal","uv0","color","feature-index"]}}]}],"attributeStorageInfo":[],"fields":[]}]} \ No newline at end of file diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/index.json b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/index.json new file mode 100644 index 0000000000..0750502d87 --- /dev/null +++ b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/index.json @@ -0,0 +1 @@ +{"version":"{1F518F6A-9621-4476-8194-DD034FF34614}","id":0,"name":"Frankfurt-md-2","href":"./layers/0","layerType":"IntegratedMesh","spatialReference":{"wkid":4326,"latestWkid":4326,"vcsWkid":5773,"latestVcsWkid":5773},"capabilities":["View","Query"],"store":{"id":"{F23F2D74-EE90-4C42-AFF3-56CE04640CE6}","profile":"meshpyramids","version":"1.8","resourcePattern":["3dNodeIndexDocument","Attributes","SharedResource","Geometry"],"rootNode":"./nodes/root","extent":[8.638486093840738,50.10162577927056,8.714439334060673,50.11520548628318],"indexCRS":"http://www.opengis.net/def/crs/EPSG/0/4326","vertexCRS":"http://www.opengis.net/def/crs/EPSG/0/4326","normalReferenceFrame":"east-north-up","attributeEncoding":"application/octet-stream; version=1.6","textureEncoding":["image/jpeg","image/ktx2"],"lodType":"MeshPyramid","lodModel":"node-switching","defaultGeometrySchema":{"geometryType":"triangles","header":[{"property":"vertexCount","type":"UInt32"},{"property":"featureCount","type":"UInt32"}],"topology":"PerAttributeArray","ordering":["position","normal","uv0","color"],"vertexAttributes":{"position":{"valueType":"Float32","valuesPerElement":3},"normal":{"valueType":"Float32","valuesPerElement":3},"uv0":{"valueType":"Float32","valuesPerElement":2},"color":{"valueType":"UInt8","valuesPerElement":4}},"featureAttributeOrder":["id","faceRange"],"featureAttributes":{"id":{"valueType":"UInt64","valuesPerElement":1},"faceRange":{"valueType":"UInt32","valuesPerElement":2}}}},"fullExtent":{"xmin":8.638486093840738,"ymin":50.10162577927056,"xmax":8.714439334060673,"ymax":50.11520548628318,"zmin":-4668.025435449932,"zmax":5048.2198269797045},"heightModelInfo":{"heightModel":"gravity_related_height","vertCRS":"EGM96_Geoid","heightUnit":"meter"},"nodePages":{"nodesPerPage":64,"lodSelectionMetricType":"maxScreenThresholdSQ"},"materialDefinitions":[{"doubleSided":true,"emissiveFactor":[255,255,255],"alphaMode":"opaque","pbrMetallicRoughness":{"roughnessFactor":1,"metallicFactor":1,"baseColorTexture":{"textureSetDefinitionId":0}}}],"textureSetDefinitions":[{"formats":[{"name":"0","format":"jpg"},{"name":"1","format":"ktx2"}]},{"formats":[{"name":"0","format":"jpg"},{"name":"1","format":"ktx2"}],"atlas":true}],"geometryDefinitions":[{"geometryBuffers":[{"offset":8,"position":{"type":"Float32","component":3},"normal":{"type":"Float32","component":3},"uv0":{"type":"Float32","component":2},"color":{"type":"UInt8","component":4},"featureId":{"binding":"per-feature","type":"UInt64","component":1},"faceRange":{"binding":"per-feature","type":"UInt32","component":2}},{"compressedAttributes":{"encoding":"draco","attributes":["position","normal","uv0","color","feature-index"]}}]}],"attributeStorageInfo":[],"fields":[]} \ No newline at end of file diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodepages/0/index.json b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodepages/0/index.json new file mode 100644 index 0000000000..f908de3a64 --- /dev/null +++ b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodepages/0/index.json @@ -0,0 +1 @@ +{"nodes":[{"index":0,"lodThreshold":0,"obb":{"center":[8.676496951388435,50.10841667136257,141.39774178531218],"halfSize":[2311.620410505966,2062.4371056037326,959.9614393664732],"quaternion":[0.2227952683502938,0.25834836931465127,0.7144546321993598,0.6109373295678506]},"children":[1]},{"index":1,"obb":{"center":[8.676496951388435,50.108416671362576,141.39774178662847],"halfSize":[2311.620410505966,2062.4371056037326,959.9614393664732],"quaternion":[0.2227952683502938,0.25834836931465127,0.7144546321993598,0.6109373295678506]},"lodThreshold":217717.37806230574,"children":[],"mesh":{"geometry":{"definition":0,"resource":1,"vertexCount":15312,"featureCount":1},"attribute":{"resource":1},"material":{"definition":0,"resource":1,"texelCountHint":524288}}}]} \ No newline at end of file diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/0/index.bin b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/0/index.bin new file mode 100644 index 0000000000000000000000000000000000000000..39930c244973e8cd40158730c06442f730876017 GIT binary patch literal 551256 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%0AOxJR$Uo5;r=Bx9Z$d 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 W-2vSJ-GTov9kBVwzvm7%HvbQZbXHFQ literal 0 HcmV?d00001 diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/1/index.bin b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/geometries/1/index.bin new file mode 100644 index 0000000000000000000000000000000000000000..06e10510966f50e5e66119e371b0771382111df1 GIT binary patch literal 207942 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�NUF@|fK}j2iNWyM&^qz0q~wqrlsx_@DUNi9 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 Qst%lqikOkuJJk>W4>;!({Qv*} literal 0 HcmV?d00001 diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/index.json b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/index.json new file mode 100644 index 0000000000..7d2b60e1eb --- /dev/null +++ b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/index.json @@ -0,0 +1 @@ +{"version":"{9CF428F9-E6D2-47C1-86C7-20589A1510B2}","id":"1","level":1,"mbs":[8.676496951388435,50.108416671362576,141.39774178662847,3243.2640505992454],"obb":{"center":[8.676496951388435,50.108416671362576,141.39774178662847],"halfSize":[2311.620410505966,2062.4371056037326,959.9614393664732],"quaternion":[0.2227952683502938,0.25834836931465127,0.7144546321993598,0.6109373295678506]},"lodSelection":[{"metricType":"maxScreenThreshold","maxError":526.503917672968},{"metricType":"maxScreenThresholdSQ","maxError":217717.37806230574}],"children":[],"neighbors":[],"parentNode":{"id":"root","href":"../root","mbs":[8.676496951388435,50.10841667136257,141.39774178531218,3243.2640505992454],"obb":{"center":[8.676496951388435,50.10841667136257,141.39774178531218],"halfSize":[2311.620410505966,2062.4371056037326,959.9614393664732],"quaternion":[0.2227952683502938,0.25834836931465127,0.7144546321993598,0.6109373295678506]}},"sharedResource":{"href":"./shared"},"geometryData":[{"href":"./geometries/0"}],"textureData":[{"href":"./textures/0"},{"href":"./textures/1"}]} \ No newline at end of file diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/shared/index.json b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/shared/index.json new file mode 100644 index 0000000000..5051cb2d18 --- /dev/null +++ b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/shared/index.json @@ -0,0 +1 @@ +{"materialDefinitions":{"Mat10":{"name":"standard","type":"standard","params":{"renderMode":"solid","shininess":1,"reflectivity":0,"ambient":[1,1,1],"diffuse":[1,1,1],"specular":[0,0,0],"useVertexColorAlpha":false,"vertexRegions":false,"vertexColors":true}}},"textureDefinitions":{"10":{"encoding":["image/jpeg"],"wrap":["none"],"atlas":false,"uvSet":"uv0","channels":"rgb","images":[{"id":"1170920505658572802","size":1024,"href":["../textures/0"],"length":[2097152]}]}}} \ No newline at end of file diff --git a/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/textures/0/index.jpg b/modules/tile-converter/test/data/i3s-server/Frankfurt-md-2/SceneServer/layers/0/nodes/1/textures/0/index.jpg new file mode 100644 index 0000000000000000000000000000000000000000..cf7f57c1190274a89a6af14b5e3f4dea43a60e44 GIT binary patch literal 45225 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