diff --git a/planetiler-core/pom.xml b/planetiler-core/pom.xml index ed9403936f..cbfcadf738 100644 --- a/planetiler-core/pom.xml +++ b/planetiler-core/pom.xml @@ -181,6 +181,11 @@ parquet-floor 1.48 + + org.lz4 + lz4-java + 1.8.0 + diff --git a/planetiler-core/src/main/java/com/onthegomap/planetiler/reader/osm/PbfDecoder.java b/planetiler-core/src/main/java/com/onthegomap/planetiler/reader/osm/PbfDecoder.java index dc6c925552..ddfad722e1 100644 --- a/planetiler-core/src/main/java/com/onthegomap/planetiler/reader/osm/PbfDecoder.java +++ b/planetiler-core/src/main/java/com/onthegomap/planetiler/reader/osm/PbfDecoder.java @@ -22,6 +22,9 @@ import java.util.function.IntUnaryOperator; import java.util.zip.DataFormatException; import java.util.zip.Inflater; +import net.jpountz.lz4.LZ4Exception; +import net.jpountz.lz4.LZ4Factory; +import net.jpountz.lz4.LZ4FastDecompressor; import org.locationtech.jts.geom.Envelope; /** @@ -74,8 +77,24 @@ private static byte[] readBlobContent(Fileformat.Blob blob) { throw new FileFormatException("PBF blob contains incomplete compressed data."); } inflater.end(); + } else if (blob.hasLz4Data()) { + final int decompressedLength = blob.getRawSize(); + LZ4Factory factory = LZ4Factory.fastestInstance(); + LZ4FastDecompressor decompressor = factory.fastDecompressor(); + blobData = new byte[decompressedLength]; + try { + int compressedBytesRead = + decompressor.decompress(blob.getLz4Data().toByteArray(), 0, blobData, 0, decompressedLength); + int compressedBytesExpected = blob.getLz4Data().size(); + if (compressedBytesRead != compressedBytesExpected) { + throw new FileFormatException("Unable to decompress PBF blob. read %d compressed bytes but expected %d" + .formatted(decompressedLength, compressedBytesExpected)); + } + } catch (LZ4Exception e) { + throw new FileFormatException("Unable to decompress PBF blob.", e); + } } else { - throw new FileFormatException("PBF blob uses unsupported compression, only raw or zlib may be used."); + throw new FileFormatException("PBF blob uses unsupported compression, only lz4, zlib, or raw may be used."); } return blobData; diff --git a/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java b/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java index 112190cebb..0be94acf28 100644 --- a/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java +++ b/planetiler-core/src/test/java/com/onthegomap/planetiler/PlanetilerTests.java @@ -2088,10 +2088,13 @@ private static TileCompression extractTileCompression(String args) { "--tile-compression=none", "--tile-compression=gzip", "--output-layerstats", - "--max-point-buffer=1" + "--max-point-buffer=1", + "--osm-test-path=monaco-latest.lz4.osm.pbf", }) void testPlanetilerRunner(String args) throws Exception { - Path originalOsm = TestUtils.pathToResource("monaco-latest.osm.pbf"); + var argParsed = Arguments.fromArgs(args.split(" ")); + Path originalOsm = + TestUtils.pathToResource(argParsed.getString("osm-test-path", "osm-test-path", "monaco-latest.osm.pbf")); Path tempOsm = tempDir.resolve("monaco-temp.osm.pbf"); final TileCompression tileCompression = extractTileCompression(args); diff --git a/planetiler-core/src/test/resources/monaco-latest.lz4.osm.pbf b/planetiler-core/src/test/resources/monaco-latest.lz4.osm.pbf new file mode 100644 index 0000000000..9d4e0a34f0 Binary files /dev/null and b/planetiler-core/src/test/resources/monaco-latest.lz4.osm.pbf differ