- fix possible bug when querystring parameter are case insensitive (developmentseed#323)
- update
tilejson
andWMTSCapabilities.xml
endpoints to allow list querystrings (as done previously in developmentseed#319)
- update rio-tiler dependency to
>=2.1
version and updaterescale
query-parameter (developmentseed#319)
# before
# previously, rio-tiler was splitting a list of input range in tuple of 2
rescale=0,1000,0,1000,0,1000
# now
# rio-tiler 2.1 now expect sequence of tuple in form of Sequence[Tuple[Num, Num]]
rescale=0,1000&rescale=0,1000&rescale=0,1000
- update
cogeo-mosaic
version to>=3.0,<3.1
.
- re-order middlewares (developmentseed#311)
- update rio-cogeo version to
>=2.2
and userio_cogeo.models
instead of custom ones.
-
add
exclude_path
options intitiler.application.middleware.CacheControlMiddleware
to avoid adding cache-control headers to specific paths. -
allow
histogram_bins
to be a single value or a,
delimited scalar (developmentseed#307) -
change error status from
404
to500
forRasterioIOError
exception (author @kylebarron, developmentseed#300)Sometimes GDAL/Rasterio can lose track of the file handler (might be related to cache issue + threading) and raise
RasterioIOError: file not found
, while the file exists for real. To avoid caching this, we changed the error code to 500 (errors >= 500 do not getcache-control
header on titiler.application).
- add support for
.jpg
and.jpeg
extensions (developmentseed#271) - better error message when parsing the colormap value fails (developmentseed#279)
breaking change
-
split
titiler
into a set of namespaces packages (developmentseed#284)titiler.core
The
core
package host the low level tiler factories.# before from titiler.endpoints.factory import TilerFactory # now from titiler.core.factory import TilerFactory
titiler.mosaic
The
mosaic
package is a plugin totitiler.core
which adds support for MosaicJSON# before from titiler.endpoints.factory import MosaicTilerFactory # now from titiler.mosaic.factory import MosaicTilerFactory
titiler.application
The
application
package is a fullready to use
FastAPI application with support of STAC, COG and MosaicJSON.# before $ pip install titiler $ uvicorn titiler.main:app --reload # now $ pip install titiler.application uvicorn $ uvicorn titiler.application.main:app --reload
- adapt for cogeo-mosaic
3.0.0rc2
and addbackend_options
attribute in MosaicTilerFactory (developmentseed#247) - update FastAPI requirements
- update minimal python version to 3.6
- add
**render_params.kwargs
to pass custom render params inimage.render
method (developmentseed#259) - Changed probe url from
/ping
to/healthz
in k8s deployment
breaking change
-
renamed
OptionalHeaders
,MimeTypes
andImageDrivers
enums to the singular form (developmentseed#258) -
renamed titiler.dependencies's Enums (
ColorMapName
,ResamplingName
andTileMatrixSetName
) to the singular form (developmentseed#260) -
renamed
MimeType
toMediaType
(developmentseed#258) -
add
ColorMapParams
dependency to ease the creation of custom colormap dependency (developmentseed#252) -
renamed
PathParams
toDatasetPathParams
and also made it a simple callable (developmentseed#260) -
renamed
colormap
query-parameter tocolormap_name
(developmentseed#262)# before /cog/preview.png?colormap=viridis # now /cog/preview.png?colormap_name=viridis
-
use
colormap
query-parameter to pass custom colormap (developmentseed#262)/cog/preview.png?colormap={"0": "#FFFF00FF", ...}
- update FastAPI requirements
- add
validate
inMosaicTilerFactory
(developmentseed#206, author @drnextgis) - rename
ressources
package toresources
(developmentseed#210, author @drnextgis) - renamed environment variables prefixes for API and STACK configurations:
TITILER_STACK
as prefix to CDK andTITILER_API
as prefix to API (developmentseed#211, author @fredliporace) - remove MosaicTilerFactory
create
andupdate
endpoints (developmentseed#218) - deleted
titiler.models.mosaics
because the models are not used anymore (developmentseed#221) - update rio-tiler and cogeo-mosaic minimal versions (developmentseed#220, developmentseed#213)
- move STAC related dependencies to `titiler.dependencies (developmentseed#225)
- add
rio_tiler.io.MultiBandReader
bands dependencies (developmentseed#226) - add
MultiBaseTilerFactory
andMultiBandTilerFactory
custom tiler factories (developmentseed#230) - Update STAC tiler to use the new
MultiBaseTilerFactory
factory - depreciate empty GET endpoint for MosaicTilerFactory read (developmentseed#232)
- better
debug
configuration and make reponse headers metadata optional (developmentseed#232)
breaking change
- update
titiler.dependencies.AssetsBidxParams
to makeasset
a required parameter (developmentseed#230 - the STAC
/info
endpoint now expect theassets
parameter to be passed. To ge the list of available assets we added a new/assets
endpoint within the tiler factory - remove
COGReader
as defaultreader
intitiler.endpoints.factory.BaseTilerFactory
- add
rio_tiler.errors.MissingBands
in known errors. - add
titiler.endpoints.factory.TMSFactory
to enable custom TMS endpoints. - breaking rename
BaseFactory
toBaseTilerFactory
intitiler.endpoints.factory
- allow
API_DISABLE_{COG/STAC/MOSAIC}
environment variables to control default endpoints in titiler main app (developmentseed#156) - add
overwriting=False/True
on MosaicJSON creation (developmentseed#164) - add
gdal_config
option to Tiler factories to replace customAPIRoute
class (developmentseed#168) - add
info.geojson
endpoint to return dataset info as a GeoJSON feature (developmentseed#166) - update
rio-tiler
,cogeo-mosaic
and optional dependencies
- require
rio-tiler>=2.0.0rc2
- update Enums for Image types. (breaking)
- Add more output datatype (jpeg2000, pngraw)
- add
width/height
in/crop
endpoint path
/crop/{minx},{miny},{maxx},{maxy}.{format}
/crop/{minx},{miny},{maxx},{maxy}/{width}x{height}.{format}
- relax version for rio-tiler and cogeo-mosaic
"rio-cogeo~=2.0"
"rio-tiler>=2.0.0rc1,<2.1"
"cogeo-mosaic>=3.0.0a17,<3.1"
- split
tile()
forMosaicTilerFactory
method (developmentseed#147)
- update for rio-tiler==2.0.0rc1, cogeo-mosaic==3.0.0a17 and morecantile==2.0
- split
tile()
factory method (developmentseed#141, author @fredliporace)
- avoid changing mutable TMS and Colormap list by using deepcopy.
- quiet/turn off logs
- add logger middleware (developmentseed#139)
- update for rio-tiler 2.0.0b17, which now support TMS (morecantile) by default.
- update fastapi minimum version to 0.61
breaking changes
- removed TMSTilerFactory (because default reader built with rio_tiler BaseReader should support TMS).
Note: We changed the versioning scheme to {major}.{minor}.{path}{pre}{prenum}
- remove
pkg_resources
(pypa/setuptools#510)
- refactor CacheControl Middleware
- rename headers value
X-Server-Timings
toServer-Timing
. - add
total;dur={}
in response headerServer-Timing
, using newtitiler.middleware.TotalTimeMiddleware
middleware (113)
from titiler.middleware import CacheControlMiddleware, TotalTimeMiddleware
from fastapi import FastAPI
app.add_middleware(CacheControlMiddleware, cachecontrol="public, max-age=3600")
app.add_middleware(TotalTimeMiddleware)
- Add Brotli compression support (#126, author @kylebarron)
- Numerous fix to CDK app.py (co-author @kylebarron)
- exclude
tests/
anstack/
in titiler python package. - add
EPSG6933
in TMS
breaking changes
-
[FACTORY] the
additional_dependency
should be a Callable which return a dict.@dataclass # type: ignore class BaseFactory(metaclass=abc.ABCMeta): """BaseTiler Factory.""" ... # provide custom dependency additional_dependency: Callable[..., Dict] = field(default=lambda: dict())
def AssetsParams( assets: Optional[str] = Query( None, title="Asset indexes", description="comma (',') delimited asset names (might not be an available options of some readers)", ) ) -> Dict: """Assets Dependency.""" kwargs = {} if assets: kwargs["assets"] = assets.split(",") return kwargs
-
[FACTORY] remove
_
prefix in factory methods (e.g_tile
->tile
) -
[FACTORY] refactor dependencies to better align with rio_tiler.io.BaseReader method definition.
Example:
In the
metadata
, theMetadataParams
will be used to passpmin
andpmax
because they are the only required parameters for the metadata method. All other params will be passed to akwargs
dict.@dataclass class MetadataParams(DefaultDependency): """Common Metadada parameters.""" # Required params pmin: float = Query(2.0, description="Minimum percentile") pmax: float = Query(98.0, description="Maximum percentile") # Optional parameters bidx: Optional[str] = Query( None, title="Band indexes", description="comma (',') delimited band indexes", ) ... def __post_init__(self): """Post Init.""" if self.bidx is not None: self.kwargs["indexes"] = tuple( int(s) for s in re.findall(r"\d+", self.bidx) ) ... # metadata method in factory def metadata( src_path=Depends(self.path_dependency), metadata_params=Depends(self.metadata_dependency), kwargs: Dict = Depends(self.additional_dependency), ): """Return metadata.""" reader = src_path.reader or self.reader with reader(src_path.url, **self.reader_options) as src_dst: info = src_dst.metadata( metadata_params.pmin, metadata_params.pmax, **metadata_params.kwargs, **kwargs, ) return info
-
[FACTORY] refactor dependencies definition
@dataclass # type: ignore class BaseFactory(metaclass=abc.ABCMeta): """BaseTiler Factory.""" reader: default_readers_type = field(default=COGReader) reader_options: Dict = field(default_factory=dict) # FastAPI router router: APIRouter = field(default_factory=APIRouter) # Path Dependency path_dependency: Type[PathParams] = field(default=PathParams) # Rasterio Dataset Options (nodata, unscale, resampling) dataset_dependency: default_deps_type = field(default=DatasetParams) # Indexes/Expression Dependencies layer_dependency: default_deps_type = field(default=BidxExprParams) # Image rendering Dependencies render_dependency: default_deps_type = field(default=RenderParams) # TileMatrixSet dependency tms_dependency: Callable[..., TileMatrixSet] = WebMercatorTMSParams # provide custom dependency additional_dependency: Callable[..., Dict] = field(default=lambda: dict())
-
remove
PathParams.reader
attribute. This option was not used and would have been technically difficult to use.@dataclass class PathParams: """Create dataset path from args""" url: str = Query(..., description="Dataset URL")
-
Update
.npy
output format to follow the numpyTile format (#103)import numpy import requests from io import BytesIO endpoint = ... url = "https://opendata.digitalglobe.com/events/mauritius-oil-spill/post-event/2020-08-12/105001001F1B5B00/105001001F1B5B00.tif" r = requests.get(f"{endpoint}/cog/tiles/14/10818/9146.npy", params = { "url": url, } ) data = numpy.load(BytesIO(r.content)) print(data.shape) > (4, 256, 256)
-
Add
titiler.custom.routing.apiroute_factory
. This function enable the creation of custom fastapi.routing.APIRoute class withrasterio.Env()
block.from fastapi import FastAPI, APIRouter from rasterio._env import get_gdal_config from titiler.custom.routing import apiroute_factory app = FastAPI() route_class = apiroute_factory({"GDAL_DISABLE_READDIR_ON_OPEN": "FALSE"}) router = APIRouter(route_class=route_class) @router.get("/simple") def simple(): """should return FALSE.""" res = get_gdal_config("GDAL_DISABLE_READDIR_ON_OPEN") return {"env": res} app.include_router(router)
Note: This has only be tested for python 3.6 and 3.7.
-
add custom
url_for
method in TilerFactory to retrieveprefixed
endpoint URL (#95) -
remove magic
titiler.dependencies.PathParams
mosaicid path translation, where a user could passurl=mosaicid://
to the endpoint. -
switch to
pydantic.BaseSettings
for FastAPI application setting management.List of Settings:
name: str = "titiler" cors_origins: str = "*" cachecontrol: str = "public, max-age=3600"
API Settings can now be set by adding a .env
file in your local project or by setting environment variables (e.g API_CORS_ORIGIN="https://mywebsite.com/*"
)
- add Transform and CRS information in
/part
GeoTIFF output - pin rio-tiler-crs to
>=3.0b4,<3.1
and cogeo-mosaic to>=3.0a10,<3.1
- rename titiler.models.cog.py to titiler.models.dataset.py
- remove cog* prefix to Bounds, Info and Metadata models
- allow Union[str, int] for key in Metadata.statistics (as defined in rio-tiler-pds)
e.g Create a Landsat 8 Tiler
from titiler.endpoints.factory import TilerFactory, MosaicTilerFactory
from titiler.dependencies import BandsParams
from rio_tiler_pds.landsat.aws.landsat8 import L8Reader # Not in TiTiler dependencies
from fastapi import FastAPI
app = FastAPI(title="Landsat Tiler", openapi_url="/api/v1/openapi.json")
scene = TilerFactory(
reader=L8Reader, additional_dependency=BandsParams, router_prefix="scenes"
)
mosaic = MosaicTilerFactory(
dataset_reader=L8Reader,
additional_dependency=BandsParams,
add_update=False,
add_create=False,
router_prefix="mosaic",
)
app.include_router(scene.router, prefix="/scenes", tags=["Scenes"])
app.include_router(mosaic.router, prefix="/mosaic", tags=["Mosaic"])
First release on pypi
For this release we created new Tiler Factories class which handle creation of FastAPI routers for a given rio_tiler Readers.
from titiler.endpoints.factory import TilerFactory
from rio_tiler.io import COGReader, STACReader
from fastapi import FastAPI
app = FastAPI()
cog = TilerFactory()
app.include_router(cog.router, tags=["Cloud Optimized GeoTIFF"])
The titiler.endpoints.factory.TilerFactory
class will create a tiler with Web Mercator
as uniq supported Tile Matrix Set.
For other TMS support, tiler needs to be created with titiler.endpoints.factory.TMSTilerFactory
and with a TMS friendly reader (e.g rio_tiler_crs.COGReader
).
Simple tiler with only Web Mercator support
from rio_tiler.io import COGReader
from titiler.endpoints import factory
from titiler.dependencies import WebMercatorTMSParams
app = factory.TilerFactory(reader=COGReader)
assert app.tms_dependency == WebMercatorTMSParams
Tiler with more TMS support (from morecantile)
from rio_tiler_crs import COGReader
from titiler.endpoints import factory
from titiler.dependencies import TMSParams
app = factory.TMSTilerFactory(reader=COGReader)
assert app.tms_dependency == TMSParams
- add mosaic support (#17 author @geospatial-jeff)
- update to rio-tiler-crs>=3.0b* and rio-tiler>=2.0b*
- Pin fastapi version to 0.60.1
- Remove titiler.core in favor of starlette settings (#55, author @geospatial-jeff)
- Add fastapi exception handlers (#56, author @geospatial-jeff)
- Remove intermediary routers (#57, author @geospatial-jeff)
- Remove /titiler/api submodule (e.g titiler.api.utils -> titiler.utils)
- Add Cache-Control middleware. Endpoints do not define any cache-control headers. (part of #43, co-author with @geospatial-jeff)
- Add 'X-Assets' in response headers for mosaic tiles (#51)
- add cog validation via rio-cogeo (co-author with @geospatial-jeff, #37)
- default tiler to Web Mercator only
- removed cache layer for tiles
- updated html templates
template_dir = pkg_resources.resource_filename("titiler", "templates")
templates = Jinja2Templates(directory=template_dir)
cog_template = templates.TemplateResponse(
name="cog_index.html",
context={
"request": request,
"tilejson": request.url_for("cog_tilejson"),
"metadata": request.url_for("cog_metadata"),
},
media_type="text/html",
)
stac_template = templates.TemplateResponse(
name="stac_index.html",
context={
"request": request,
"tilejson": request.url_for("stac_tilejson"),
"metadata": request.url_for("stac_info"),
},
media_type="text/html",
)
- add
width
&height
parameters in API docs to force output size for part/preview endpoints. - add
resampling_method
in API docs.
link: https://github.com/developmentseed/titiler/commit/725da5fa1bc56d8e192ae8ff0ad107493ca93378
- add minimum fastapi version (0.54.0) and update docker config
link: https://github.com/developmentseed/titiler/commit/95b98a32ffb3274d546dd52f99a3920091029b4c
- add
/preview
,/crop
,/point
endpoints
link: https://github.com/developmentseed/titiler/commit/8b63fc6b6141b9c9361c95d80897d77b5e2d47c3
- support STAC items (#16)
- better API documentation via response models
- update UI (
/stac/viewer
,/cog/viewer
) - re-order OpenAPI route tags
- update documentation
link: https://github.com/developmentseed/titiler/commit/fa2cb78906b0fd88506b89bace8174969be8cd4f
Initial release
link: https://github.com/developmentseed/titiler/commit/f4fdc02ea0235470589eeb34a4da8e5aae74e696