-
-
Notifications
You must be signed in to change notification settings - Fork 221
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Try to fix #875 . Still a draft and a lot to discuss and finish. - [ ] Remote file on like S3. We will make it in next PRs. Let's make this PR just a basic start. - [x] Made the padded part transparent - [x] Made the [gdal nodata](https://gdal.org/en/stable/drivers/raster/gtiff.html#nodata-value) transparent - [x] Add tests - [x] Add doc - [x] Support rgb/rgba u8 tif tiled image. We will do others in next PRs and leave this PR simple. ## Color types and bits per smaple |colory type|bits per sample|supported|status| |----|----|----|----| |rgb/rgba|8|✅|| |rgb/rgba|16/32...|🛠️|working on, will be added in next PRs| |gray|8/16/32...|🛠️|working on, will be added in next PRs| ## Comporession The crate we used support these methods. | Compression Format | tiff crate(This PR based on this) | GDAL creation optinos | |--------------------|------------------|----------------| | None | ✓ | ✓ | | LZW | ✓ | ✓ (gdal default) | | JPEG | | ✓ | | Deflate | ✓ | ✓ | | ZSTD | | ✓ | | WEBP | | ✓ | | LERC | | ✓ | | LERC_DEFLATE | | ✓ | | LERC_ZSTD | | ✓ | | LZMA | | ✓ | | PackBits | ✓ | Not sure but couldn't find this on gdal doc | ## Some terms and ref links [7.1.2 Tiles in COG spec](https://docs.ogc.org/is/21-026/21-026.html#_tiles) [Tiff 6.0 spec page 67](https%3A%2F%2Fwww.itu.int%2Fitudoc%2Fitu-t%2Fcom16%2Ftiff-fx%2Fdocs%2Ftiff6.pdf) [Chapter 4 of COG spec](https://docs.ogc.org/is/21-026/21-026.html#_terms_and_definitions) ## Tile and padding > In the context for a TIFF file, Tiling is a strategy for dividing the content in the TIFF file differently than using the classical Strips. Tiles, as defined in the TIFF version 6.0 specification, can be mapped to the ones defined in the OGC Two Dimensional Tile Matrix Set Standard (2D-TMS). For example in 2D-TMS, the TIFF 6.0 forces all tiles to be of the same size. This is possible with the introduction of the concept of padding: if necessary extra blank rows or columns are added to the right-most and bottom-most tile to make them the same shape as other tiles. However, the naming of the TIFF tags used version 6.0 and the property names used in the 2D-TMS differ. The following table provides a mapping between the two standards. | OGC 2D-TMS | TIFF v. 6.0 | Definition | |---------------|--------------|-------------------------------------------------| | TileWidth | TileWidth | The tile width in pixels. The number of columns in each tile | | TileHeight | TileLength | The tile height in pixels. The number of rows in each tile | | MatrixWidth | TilesAcross | Number of tiles in the width direction | | MatrixHeight | TilesDown | Number of tiles in the height direction --------- Co-authored-by: Yuri Astrakhan <[email protected]>
- Loading branch information
1 parent
09535cf
commit 1ed46b5
Showing
45 changed files
with
894 additions
and
218 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,104 @@ | ||
# Cloud Optimized GeoTIFF File Sources | ||
|
||
Martin can also serve raster sources like local [COG(Cloud Optimized GeoTIFF)](https://cogeo.org/) files. For cog on remote like S3 and other improvements, you could track them on [issue 875](https://github.com/maplibre/martin/issues/875), we are working on and welcome any assistance. | ||
|
||
## Supported colortype and bits per sample | ||
|
||
| colory type | bits per sample | supported | status | | ||
| ----------- | --------------- | --------- | ---------- | | ||
| rgb/rgba | 8 | ✅ | | | ||
| rgb/rgba | 16/32... | 🛠️ | working on | | ||
| gray | 8/16/32... | 🛠️ | working on | | ||
|
||
## Supported compression | ||
|
||
* None | ||
* LZW | ||
* Deflate | ||
* PackBits | ||
|
||
## Run Martin with CLI to serve cog files | ||
|
||
```bash | ||
# Configured with a directory containing TIFF files. | ||
martin /with/tiff/dir1 /with/tiff/dir2 | ||
# Configured with dedicated TIFF file. | ||
martin /path/to/target1.tif /path/to/target1.tif | ||
# Configured with a combination of directories and dedicated TIFF files. | ||
martin /with/tiff/files /path/to/target.tif | ||
``` | ||
|
||
## Run Martin with configuration file | ||
|
||
```yml | ||
keep_alive: 75 | ||
|
||
# The socket address to bind [default: 0.0.0.0:3000] | ||
listen_addresses: '0.0.0.0:3000' | ||
|
||
# Number of web server workers | ||
worker_processes: 8 | ||
|
||
# Amount of memory (in MB) to use for caching tiles [default: 512, 0 to disable] | ||
cache_size_mb: 8 | ||
|
||
# Database configuration. This can also be a list of PG configs. | ||
|
||
cog: | ||
paths: | ||
# scan this whole dir, matching all *.tif files | ||
- /dir-path | ||
# specific TIFF file will be published as a cog source | ||
- /path/to/target1.tif | ||
- /path/to/target2.tif | ||
sources: | ||
# named source matching source name to a single file | ||
cog-src1: /path/to/cog1.tif | ||
cog-src2: /path/to/cog2.tif | ||
``` | ||
## About COG | ||
[COG](https://cogeo.org/) is just Cloud Optimized GeoTIFF file. | ||
TIFF is an image file format. TIFF tags are something like key-value pairs inside to describe the metadata about a TIFF file, ike `ImageWidth`, `ImageLength`, etc. | ||
|
||
GeoTIFF is a valid TIFF file with a set of TIFF tags to describe the 'Cartographic' information associated with it. | ||
|
||
COG is a valid GeoTIFF file with some requirements for efficient reading. That is, all COG files are valid GeoTIFF files, but not all GeoTIFF files are valid COG files. For quick access to tiles in TIFF files, Martin relies on the requirements/recommendations(like the [requirement about Reduced-Resolution Subfiles](https://docs.ogc.org/is/21-026/21-026.html#_requirement_reduced_resolution_subfiles) and [the content dividing strategy](https://docs.ogc.org/is/21-026/21-026.html#_tiles)) so we use the term `COG` over `GeoTIFF` in our documentation and configuration files. | ||
|
||
You may want to visit these specs: | ||
|
||
* [TIFF 6.0](https://www.itu.int/itudoc/itu-t/com16/tiff-fx/docs/tiff6.pdf) | ||
* [GeoTIFF](https://docs.ogc.org/is/19-008r4/19-008r4.html) | ||
* [Cloud Optimized GeoTIFF](https://docs.ogc.org/is/21-026/21-026.html) | ||
|
||
### COG generation with GDAL | ||
|
||
You could generate cog with `gdal_translate` or `gdalwarp`. See more details in [gdal doc](https://gdal.org/en/latest/drivers/raster/cog.html). | ||
|
||
```bash | ||
# gdal-bin installation | ||
# sudo apt update | ||
# sudo apt install gdal-bin | ||
# gdalwarp | ||
gdalwarp src1.tif src2.tif out.tif -of COG | ||
# or gdal_translate | ||
gdal_translate input.tif output_cog.tif -of COG | ||
``` | ||
|
||
### The mapping from ZXY to tiff chunk | ||
|
||
* A single TIFF file could contains many sub-file about same spatial area, each has different resolution | ||
* A sub file is organized with many tiles | ||
|
||
So basically there's a mapping from zxy to tile of sub-file of TIFF. | ||
|
||
| zxy | mapping to | | ||
| ---------- | --------------------------- | | ||
| Zoom level | which sub-file in TIFF file | | ||
| X and Y | which tile in subfile | | ||
|
||
Clients could read only the header part of COG to figure out the mapping from zxy to the chunk number and the subfile number. Martin get tile to frontend by this mapping. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
use std::path::PathBuf; | ||
|
||
use png::EncodingError; | ||
use tiff::TiffError; | ||
|
||
#[derive(thiserror::Error, Debug)] | ||
pub enum CogError { | ||
#[error("Couldn't decode {1} as tiff file: {0}")] | ||
InvalidTiffFile(TiffError, PathBuf), | ||
|
||
#[error("Requested zoom level {0} from file {1} is out of range. Possible zoom levels are {2} to {3}")] | ||
ZoomOutOfRange(u8, PathBuf, u8, u8), | ||
|
||
#[error("Couldn't find any image in the tiff file: {0}")] | ||
NoImagesFound(PathBuf), | ||
|
||
#[error("Couldn't seek to ifd number {1} (0 based indexing) in tiff file {2}: {0}")] | ||
IfdSeekFailed(TiffError, usize, PathBuf), | ||
|
||
#[error("Too many images in the tiff file: {0}")] | ||
TooManyImages(PathBuf), | ||
|
||
#[error("Couldn't find tags {1:?} at ifd {2} of tiff file {3}: {0}")] | ||
TagsNotFound(TiffError, Vec<u16>, usize, PathBuf), | ||
|
||
#[error( | ||
"Unsupported planar configuration {2} at IFD {1} in TIFF file {0}. Only planar configuration 1 is supported." | ||
)] | ||
PlanarConfigurationNotSupported(PathBuf, usize, u16), | ||
|
||
#[error("Failed to read {1}th chunk(0 based index) at ifd {2} from tiff file {3}: {0}")] | ||
ReadChunkFailed(TiffError, u32, usize, PathBuf), | ||
|
||
#[error("Failed to write header of png file at {0}: {1}")] | ||
WritePngHeaderFailed(PathBuf, EncodingError), | ||
|
||
#[error("Failed to write pixel bytes to png file at {0}: {1}")] | ||
WriteToPngFailed(PathBuf, EncodingError), | ||
|
||
#[error("The color type {0:?} and its bit depth of the tiff file {1} is not supported yet")] | ||
NotSupportedColorTypeAndBitDepth(tiff::ColorType, PathBuf), | ||
|
||
#[error("Couldn't parse the {0} value in gdal metadata(tiff tag 42112) from {1}")] | ||
ParseSTATISTICSValueFailed(String, PathBuf), | ||
|
||
#[error("The gdal metadata(tiff tag 42112) from {1} is not valid: {0}")] | ||
InvalidGdalMetaData(String, PathBuf), | ||
|
||
#[error("Striped tiff file is not supported, the tiff file is {0}")] | ||
NotSupportedChunkType(PathBuf), | ||
} |
Oops, something went wrong.