-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
janf
committed
Mar 25, 2024
1 parent
69198a9
commit 4ae6d63
Showing
3 changed files
with
321 additions
and
0 deletions.
There are no files selected for viewing
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 |
---|---|---|
@@ -1,3 +1,4 @@ | ||
rio-tiler>=6.4.3 | ||
titiler.application==0.18.0 | ||
asyncpg==0.29.0 | ||
postgis==1.0.4 | ||
|
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,14 @@ | ||
import httpx | ||
import urllib | ||
from urllib.parse import unquote | ||
import json | ||
titiler = 'http://0.0.0.0:8001/cog/statistics?url=https%253A%252F%252Fundpgeohub.blob.core.windows.net%252Fuserdata%252Fa85516c81c0b78d3e89d3f00099b8b15%252Fdatasets%252FDem_Rwanda_10m_allt_20230921150153.tif%252FDem_Rwanda_10m_allt_20230921150153_band1.tif%253Fc3Y9MjAyMy0wOC0wMyZzcz1iJnNydD1vJnNlPTIwMjQtMTAtMjZUMjElM0EwMSUzQTAxWiZzcD1yJnNpZz1FYzJreFBtQnA2NyUyQmdOcUwyWURHOUQxSUJUWEV4RnB4c0tjZkYlMkZpYzlrMCUzRA%253D%253D' | ||
#titiler = 'https://titiler.xyz/cog/info?url=https%253A%252F%252Fundpgeohub.blob.core.windows.net%252Fuserdata%252Fa85516c81c0b78d3e89d3f00099b8b15%252Fdatasets%252FDem_Rwanda_10m_allt_20230921150153.tif%252FDem_Rwanda_10m_allt_20230921150153_band1.tif%253Fc3Y9MjAyMy0wOC0wMyZzcz1iJnNydD1vJnNlPTIwMjQtMTAtMjZUMjElM0EwMSUzQTAxWiZzcD1yJnNpZz1FYzJreFBtQnA2NyUyQmdOcUwyWURHOUQxSUJUWEV4RnB4c0tjZkYlMkZpYzlrMCUzRA%253D%253D' | ||
|
||
u = urllib.parse.urlparse(titiler) | ||
qsd = dict(urllib.parse.parse_qsl(u.query)) | ||
qsd = {key:urllib.parse.unquote(value) for (key,value) in qsd.items()} | ||
r = httpx.get(f'{u.scheme}://{u.netloc}{u.path}',params=qsd) | ||
d = r.json() | ||
print(json.dumps(d, indent=4)) | ||
|
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,306 @@ | ||
import logging | ||
import base64 | ||
import rasterio | ||
from urllib.request import urlopen | ||
import json | ||
from rio_tiler.utils import get_array_statistics | ||
from titiler.core.factory import TilerFactory, MultiBandTilerFactory | ||
from titiler.application.routers import mosaic, stac, tms | ||
from titiler.application.settings import ApiSettings | ||
from titiler.application import __version__ as titiler_version | ||
from titiler.core.errors import DEFAULT_STATUS_CODES, add_exception_handlers | ||
from cogeo_mosaic.mosaic import MosaicJSON | ||
from titiler.application.custom import templates | ||
from titiler.mosaic.errors import MOSAIC_STATUS_CODES | ||
from geojson_pydantic.features import Feature, FeatureCollection | ||
from fastapi import FastAPI, Query, Depends, Request | ||
from fastapi.middleware.cors import CORSMiddleware | ||
from rio_tiler.models import BandStatistics | ||
from titiler.core.resources.responses import JSONResponse | ||
from starlette.responses import HTMLResponse | ||
|
||
from typing import Any, Dict, List, Type, Optional, Union | ||
from pydantic import BaseModel | ||
|
||
import attr | ||
from morecantile import TileMatrixSet | ||
from rio_tiler.constants import WEB_MERCATOR_TMS | ||
from rio_tiler.errors import InvalidBandName | ||
from rio_tiler.io import BaseReader, COGReader, MultiBandReader | ||
|
||
logging.getLogger("botocore.credentials").disabled = True | ||
logging.getLogger("botocore.utils").disabled = True | ||
logging.getLogger("rio-tiler").setLevel(logging.ERROR) | ||
|
||
api_settings = ApiSettings() | ||
|
||
|
||
def fetch_admin_geojson(url: str = None) -> dict: | ||
with urlopen(url) as response: | ||
return json.loads(response.read().decode('utf-8')) | ||
|
||
|
||
def DatasetPathParams( | ||
url: str = Query(..., description="Signed raster dataset URL"), | ||
|
||
) -> str: | ||
""" | ||
FastAPI dependency function that enables | ||
COG endpoint to handle Azure Blob located&signed raster files | ||
and combine them into one multiband GDAL VRT file. | ||
This allows one to use titiler's expression parameter | ||
to perform simple but powerful multiband computations. | ||
Example | ||
expression=((b1>0.5)&(b3<100))/b4; | ||
Each url consists of a base64 encoded SAS token: | ||
http://localhost:8000/cog/statistics? \ | ||
url=https://undpngddlsgeohubdev01.blob.core.windows.net/testforgeohub/HREA_Algeria_2012_v1%2FAlgeria_rade9lnmu_2012.tif?c3Y9MjAyMC0xMC0wMiZzZT0yMDIyLTAzLTA0VDE0JTNBMzIlM0E0M1omc3I9YiZzcD1yJnNpZz0lMkJUZVd0UnRScG1uNmpTQ1ZHY2JuV3dhUWMlMkJjRlp5c05ENjRDUDlyMERNRSUzRA==& \ | ||
url=https://undpngddlsgeohubdev01.blob.core.windows.net/testforgeohub/HREA_Algeria_2012_v1%2FAlgeria_set_zscore_sy_2012.tif?c3Y9MjAyMC0xMC0wMiZzZT0yMDIyLTAzLTA0VDE0JTNBMzIlM0E0M1omc3I9YiZzcD1yJnNpZz1DTFRBQmI5aVRmV3UlMkZ0VHElMkZ4MCUyQm1hUDFUYVM5eUtDMnB3UTBjOUZmMlNBJTNE \ | ||
The returned value is a str representing a RAM stored GDAL VRT file | ||
which Titiler will use to resolve the request | ||
Obviously the rasters need to spatially overlap. Additionaly, the VRT can be created with various params | ||
(spatial align, resolution, resmapling) that, to some extent can influence the performace of the server | ||
""" | ||
if '?' in url: | ||
furl, b64token = url.split('?') | ||
try: | ||
decoded_token = base64.b64decode(b64token).decode() | ||
except Exception: | ||
decoded_token = b64token | ||
return f'{furl}?{decoded_token}' | ||
else: | ||
return f'{url}' | ||
|
||
|
||
def admin_parameters(admin_id_url: str = Query(..., description='the Azure hosted url of admin id geojson')): | ||
return admin_id_url | ||
|
||
|
||
ccog = TilerFactory(path_dependency=DatasetPathParams) | ||
|
||
|
||
@attr.s | ||
class MultiFilesBandsReader(MultiBandReader): | ||
"""Multiple Files as Bands.""" | ||
|
||
input: List[str] = attr.ib(kw_only=True) | ||
reader_options: Dict = attr.ib(factory=dict) | ||
tms: TileMatrixSet = attr.ib(default=WEB_MERCATOR_TMS) | ||
reader: Type[BaseReader] = attr.ib(default=COGReader) | ||
|
||
def __attrs_post_init__(self): | ||
"""Fetch Reference band to get the bounds.""" | ||
|
||
self.bands = [f"b{ix + 1}" for ix in range(len(self.input))] | ||
|
||
# We assume the files are similar so we use the first one to | ||
# get the bounds/crs/min,maxzoom | ||
# Note: you can skip that and hard code values | ||
with self.reader(self.input[0], tms=self.tms, **self.reader_options) as cog: | ||
self.bounds = cog.bounds | ||
self.crs = cog.crs | ||
self.minzoom = cog.minzoom | ||
self.maxzoom = cog.maxzoom | ||
|
||
def _get_band_url(self, band: str) -> str: | ||
"""Validate band's name and return band's url.""" | ||
if band not in self.bands: | ||
raise InvalidBandName(f"{band} is not valid") | ||
|
||
index = self.bands.index(band) | ||
return self.input[index] | ||
|
||
|
||
# Forward list of urls to the reader (MultiFilesBandsReader) | ||
def MultibandDatasetPathParams(url: List = Query(..., description="Dataset URL")) -> List[str]: | ||
decoded_urls = list() | ||
for in_url in url: | ||
if '?' in in_url: | ||
furl, b64token = in_url.split('?') | ||
try: | ||
decoded_token = base64.b64decode(b64token).decode() | ||
except Exception: | ||
decoded_token = b64token | ||
decoded_url = f'/vsicurl/{furl}?{decoded_token}' | ||
else: | ||
decoded_url = f'/vsicurl/{in_url}' | ||
|
||
decoded_urls.append(decoded_url) | ||
return decoded_urls | ||
|
||
|
||
multi_band = MultiBandTilerFactory( | ||
reader=MultiFilesBandsReader, | ||
path_dependency=MultibandDatasetPathParams | ||
) | ||
|
||
|
||
class MosaicJsonCreateItem(BaseModel): | ||
url: List[str] = Query(..., description="Dataset URL") | ||
minzoom: int = 0 | ||
maxzoom: int = 22 | ||
attribution: str = None | ||
|
||
|
||
@mosaic.router.post( | ||
"/create", | ||
response_model=MosaicJSON, | ||
response_model_exclude_none=True, | ||
response_class=JSONResponse, | ||
responses={ | ||
200: {"description": "Return a MosaicJSON from multiple COGs."}}, | ||
) | ||
def create_mosaicJSON_post(payload: MosaicJsonCreateItem): | ||
url = MultibandDatasetPathParams(payload.url) | ||
minzoom = payload.minzoom | ||
maxzoom = payload.maxzoom | ||
attribution = payload.attribution | ||
|
||
mosaicjson = MosaicJSON.from_urls(urls=url, minzoom=minzoom, maxzoom=maxzoom, ) | ||
if attribution is not None: | ||
mosaicjson.attribution = attribution | ||
return mosaicjson | ||
|
||
|
||
@mosaic.router.get( | ||
"/create", | ||
response_model=MosaicJSON, | ||
response_model_exclude_none=True, | ||
response_class=JSONResponse, | ||
responses={ | ||
200: {"description": "Return a MosaicJSON from multiple COGs."}}, | ||
) | ||
def create_mosaicJSON_get( | ||
url=Depends(MultibandDatasetPathParams), | ||
minzoom: Optional[int] = 0, | ||
maxzoom: Optional[int] = 22, | ||
attribution: Optional[str] = None | ||
|
||
): | ||
mosaicjson = MosaicJSON.from_urls(urls=url, minzoom=minzoom, maxzoom=maxzoom, ) | ||
if attribution is not None: | ||
mosaicjson.attribution = attribution | ||
return mosaicjson | ||
|
||
|
||
@ccog.router.get( | ||
"/geojsonstats", | ||
# response_model=BandStatistics, | ||
response_model_exclude_none=True, | ||
response_class=JSONResponse, | ||
responses={ | ||
200: {"description": "Return dataset's band stats for 1-3 admin levels ."}}, | ||
) | ||
# def adminstats(src_path=Depends(ccog.path_dependency), admin_params: dict = Depends(admin_parameters)): | ||
def adminstats( | ||
src_path=Depends(ccog.path_dependency), | ||
geojson_url: str = None | ||
): | ||
# fetch the admin level boundary as geoJSON | ||
|
||
try: | ||
geojson_str = fetch_admin_geojson(geojson_url) | ||
except Exception as e: | ||
raise | ||
geojson = None | ||
try: | ||
geojson = FeatureCollection(**geojson_str) | ||
except Exception: | ||
geojson = Feature(**geojson_str) | ||
|
||
with rasterio.Env(**ccog.gdal_config): | ||
with ccog.reader(src_path) as src_dst: | ||
|
||
if isinstance(geojson, FeatureCollection): | ||
for i, feature in enumerate(geojson): | ||
data = src_dst.feature( | ||
feature.dict(exclude_none=True) | ||
) | ||
stats = get_array_statistics( | ||
data.as_masked() | ||
) | ||
|
||
return dict([(data.band_names[ix], BandStatistics(**stats[ix])) for ix in range(len(stats))]) | ||
else: # simple feature | ||
data = src_dst.feature( | ||
geojson.dict(exclude_none=True), | ||
) | ||
stats = get_array_statistics( | ||
data.as_masked(), | ||
) | ||
return dict([(data.band_names[ix], BandStatistics(**stats[ix])) for ix in range(len(stats))]) | ||
|
||
|
||
app = FastAPI( | ||
title=api_settings.name, | ||
description="A lightweight Cloud Optimized GeoTIFF tile server", | ||
version=titiler_version, | ||
root_path=api_settings.root_path, | ||
) | ||
|
||
if not api_settings.disable_cog: | ||
app.include_router(ccog.router, prefix="/cog", | ||
tags=["Cloud Optimized GeoTIFF"]) | ||
|
||
app.include_router(multi_band.router, prefix="/cogs", | ||
tags=["Multibands Cloud Optimized GeoTIFF"]) | ||
|
||
if not api_settings.disable_stac: | ||
app.include_router( | ||
stac.router, prefix="/stac", tags=["SpatioTemporal Asset Catalog"] | ||
) | ||
|
||
if not api_settings.disable_mosaic: | ||
app.include_router(mosaic.router, prefix="/mosaicjson", | ||
tags=["MosaicJSON"]) | ||
|
||
app.include_router(tms.router, tags=["TileMatrixSets"]) | ||
add_exception_handlers(app, DEFAULT_STATUS_CODES) | ||
add_exception_handlers(app, MOSAIC_STATUS_CODES) | ||
|
||
|
||
# if api_settings.debug: | ||
# app.add_middleware(LoggerMiddleware, headers=True, querystrings=True) | ||
# app.add_middleware(TotalTimeMiddleware) | ||
# | ||
# if api_settings.lower_case_query_parameters: | ||
# app.add_middleware(LowerCaseQueryStringMiddleware) | ||
|
||
|
||
@app.get("/health", description="Health Check", tags=["Health Check"]) | ||
def ping(): | ||
"""Health check.""" | ||
return {"ping": "pong!"} | ||
|
||
|
||
@app.get("/", response_class=HTMLResponse, include_in_schema=False) | ||
def landing(request: Request): | ||
"""TiTiler Landing page""" | ||
return templates.TemplateResponse( | ||
name="index.html", | ||
context={"request": request}, | ||
media_type="text/html", | ||
) | ||
|
||
|
||
@app.get("/routes") | ||
def get_all_urls(): | ||
url_list = [{"path": route.path, "name": route.name} | ||
for route in app.routes] | ||
return url_list | ||
|
||
|
||
# Set all CORS enabled origins | ||
if api_settings.cors_origins: | ||
app.add_middleware( | ||
CORSMiddleware, | ||
allow_origins=api_settings.cors_origins, | ||
allow_credentials=True, | ||
allow_methods=['*'], | ||
allow_headers=['*'], | ||
) |