Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: STAC Render Extension support #1038

Merged

Conversation

alekzvik
Copy link
Contributor

@alekzvik alekzvik commented Nov 26, 2024

What I am changing

  • Adding support of STAC render extension to titiler. This is a continuation of work started at satsummit stac sprint.

How I did it

  • Adding two endpoints:
    • /renders/ lists all renders
    • /renders/<render_id> returns a specific render with its metadata and template links to tilejson.json and WMTS

How you can test it

Related Issues

Example responses

Those are generated based on this STAC item
Request: http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json
Response:

{
  "params": {
    "assets": [
      "B12",
      "B08",
      "B04"
    ],
    "title": "Shortwave Infra-red",
    "rescale": [
      [
        0,
        5000
      ],
      [
        0,
        7000
      ],
      [
        0,
        9000
      ]
    ],
    "resampling": "nearest"
  },
  "links": [
    {
      "href": "http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
      "rel": "self",
      "type": "application/json",
      "title": "sir render item"
    },
    {
      "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/tilejson.json?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
      "rel": "tilesets-map",
      "templated": true,
      "title": "tilejson file for sir"
    },
    {
      "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/WMTSCapabilities.xml?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
      "rel": "tilesets-map",
      "templated": true,
      "title": "WMTS service for sir"
    }
  ]
}

Request: http://127.0.0.1:8000/stac/renders?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json
Response:

{
  "renders": {
    "sir": {
      "params": {
        "assets": [
          "B12",
          "B08",
          "B04"
        ],
        "title": "Shortwave Infra-red",
        "rescale": [
          [
            0,
            5000
          ],
          [
            0,
            7000
          ],
          [
            0,
            9000
          ]
        ],
        "resampling": "nearest"
      },
      "links": [
        {
          "href": "http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
          "rel": "self",
          "type": "application/json",
          "title": "sir render item"
        },
        {
          "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/tilejson.json?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
          "rel": "tilesets-map",
          "templated": true,
          "title": "tilejson file for sir"
        },
        {
          "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/WMTSCapabilities.xml?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
          "rel": "tilesets-map",
          "templated": true,
          "title": "WMTS service for sir"
        }
      ]
    }
  },
  "links": [
    {
      "href": "http://127.0.0.1:8000/stac/renders?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
      "rel": "self",
      "type": "application/json",
      "title": "List Render Items"
    }
  ]
}

emmanuelmathot and others added 3 commits November 19, 2024 17:45
Render extension started during STAC render sprint in SatSummit Lisbon 2024.

- listing (or showing to please Vincent)

Please contribute to complete the feature to

- generate the final XYZ link for rendering following the rules in STAC extensions
- add a dedicated endpoint for render XYZ
@alekzvik
Copy link
Contributor Author

alekzvik commented Nov 26, 2024

Latest changes:

  • fix typing import that did not work on python 3.8
  • rename renderExtention -> stacRenderExtension
  • new response structure, examples below

Both are based on item-steninel2.json
Request: http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json
Response:

{
  "params": {
    "assets": [
      "B12",
      "B08",
      "B04"
    ],
    "title": "Shortwave Infra-red",
    "rescale": [
      [
        0,
        5000
      ],
      [
        0,
        7000
      ],
      [
        0,
        9000
      ]
    ],
    "resampling": "nearest"
  },
  "links": [
    {
      "href": "http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
      "rel": "self",
      "type": "application/json",
      "title": "sir render item"
    },
    {
      "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/tilejson.json?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
      "rel": "tilesets-map",
      "templated": true,
      "title": "tilejson file for sir"
    },
    {
      "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/WMTSCapabilities.xml?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
      "rel": "tilesets-map",
      "templated": true,
      "title": "WMTS service for sir"
    }
  ]
}

Request: http://127.0.0.1:8000/stac/renders?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json
Response:

{
  "renders": {
    "sir": {
      "params": {
        "assets": [
          "B12",
          "B08",
          "B04"
        ],
        "title": "Shortwave Infra-red",
        "rescale": [
          [
            0,
            5000
          ],
          [
            0,
            7000
          ],
          [
            0,
            9000
          ]
        ],
        "resampling": "nearest"
      },
      "links": [
        {
          "href": "http://127.0.0.1:8000/stac/renders/sir?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
          "rel": "self",
          "type": "application/json",
          "title": "sir render item"
        },
        {
          "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/tilejson.json?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
          "rel": "tilesets-map",
          "templated": true,
          "title": "tilejson file for sir"
        },
        {
          "href": "http://127.0.0.1:8000/stac/{tileMatrixSetId}/WMTSCapabilities.xml?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json&assets=B12&assets=B08&assets=B04&resampling_method=nearest",
          "rel": "tilesets-map",
          "templated": true,
          "title": "WMTS service for sir"
        }
      ]
    }
  },
  "links": [
    {
      "href": "http://127.0.0.1:8000/stac/renders?url=https%3A%2F%2Fraw.luolix.top%2Fstac-extensions%2Frender%2F65edf9bcc7dfe0a02e0191a277c1ac81d86b698e%2Fexamples%2Fitem-sentinel2.json",
      "rel": "self",
      "type": "application/json",
      "title": "List Render Items"
    }
  ]
}

For the /renders/ endpoint, I have not yet figured out how to deal with the response model and put renders in additional field renders.

@alekzvik alekzvik changed the title WIP: stac render extension support feat: STAC Render Extension support Nov 26, 2024
@alekzvik alekzvik marked this pull request as ready for review November 26, 2024 18:31
dep.query_params, render
)
_ = dependency(**query_values)
query.update(query_values)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what do you think about moving this outside the extensions package to titiler.core.utils?

so we can replace reuse this outside (in titiler-pgstac and titiler-stacapi)

https://github.com/developmentseed/titiler-stacapi/blob/270c08d7ebbb26d48bddd9c573957cbf7641da7b/titiler/stacapi/factory.py#L74-L90

Copy link
Contributor Author

@alekzvik alekzvik Nov 30, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ouch, I started doing it and found out that all three use cases are slightly different.

The difference between usage here and in titiler-stacapi is evident on BidxParams:

get_dependency_params(
    dependency=BidxParams, query_params={"bidx": [1, 2, 3]}
)

titiler-stacapi wants to receive {"index": [1,2,3]}, I need {"bidx": [1,2,3]}, and the key here is in alias - code

Each project is doing a slightly different task:

  • titiler-pgstac - make sure that passed params pass validation for query params
  • titiler-stacapi - validate and conform passed params based on query params (or do the same thing fastapi does with incoming params)
  • render extension - filter passed params to the ones that can be in query params and make sure they validate

I can write a single function to cover two use cases, but not three. I will think about it a bit more and see if something comes up

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I thin one function could be used for the 3 projects to do validation and translation.

in titiler-pgstac and titiler we want to validate the render metadata but in titiler-stacapi we want to validate and also translate to internal key=values.

In titiler and titiler-pgstac we return None or at least we don't care what the return is

We could have those 2 functions in titiler.core.utils

# titiler (will be used in titiler-stacapi)
def get_dependency_params(*, dependency: Callable, query_params: Dict) -> Any:
    """Check QueryParams for Query dependency.

    1. `get_dependant` is used to get the query-parameters required by the `callable`
    2. we use `request_params_to_args` to construct arguments needed to call the `callable`
    3. we call the `callable` and catch any errors

    Important: We assume the `callable` in not a co-routine

    """
    dep = get_dependant(path="", call=dependency)
    if dep.query_params:
        # call the dependency with the query-parameters values
        query_values, _ = request_params_to_args(dep.query_params, query_params)
        return dependency(**query_values)

    return


# titiler-pgstac and titiler
def check_query_params(
    *, dependencies: List[Callable], query_params: Union[QueryParams, Dict]
) -> None:
    """Check QueryParams for Query dependency.

    1. `get_dependant` is used to get the query-parameters required by the `callable`
    2. we use `request_params_to_args` to construct arguments needed to call the `callable`
    3. we call the `callable` and catch any errors

    Important: We assume the `callable` in not a co-routine

    """
    return [
        get_dependency_params(dependency, query_params=query_params) 
        for dependency in dependencies
    ]

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree, will move to utils.

One more note, initial implementation in render extension, does not only validate but also picks (or filters) only params that would be valid in the tiling endpoint. For example, if we have in render item of the stac {"bidx": [1, 2, 3], "other": "value"}, now I would pass on to the tiling endpoint only {"bidx": [1, 2, 3]}.

I prefer this approach because we are returning a URL we can guarantee will work. Otherwise, we can pass on a param that will fail on the validation, e.g. tilescale=5.

I am bringing this up to make sure we are making the decision knowing the details.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I moved it out to utils, but I also understood that it works a bit differently than I thought before. I will add some more comments near the relevant code to explain more.

@vincentsarago
Copy link
Member

This is great work @alekzvik 🙏
just couple notes

Copy link
Contributor Author

@alekzvik alekzvik left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added a few comments on the decisions made.

"""
dep = get_dependant(path="", call=dependency)
return request_params_to_args(
dep.query_params, QueryParams(urlencode(params, doseq=True))
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Notice the addition of QueryParams here.
The reason is that request_params_to_args expects to see the result of this function.
An example that helped me catch it is rescale param - link to def.

rescale: Annotated[
    Optional[List[str]],
    Query(...)
]

If I pass things directly from STAC renders block, it would not pass validation, since it is defined as [float]
To mitigate this, I first serialize params to the state that fastapi would receive them, and then deserialize how it would do it.

Comment on lines 93 to 95
errors = _validate_params(render)

if not errors:
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

errors here is a list of all validation errors for STAC render block.
I decided to exclude links if there are errors, as I know they are invalid and will fail if you follow the link.
An alternative is to fail louder, e.g. with HTTP 400 and error details. Let me know if you prefer it.

Copy link
Member

@vincentsarago vincentsarago left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is great 🙏

Thanks a lot for the PR @alekzvik.

The overall is perfect and I'm going to merge this to be included in the next release.

The only thing that we could maybe change in the future is that by default we will put all render parameters in the tilejson.json query parameter (e.g width=1024 or minmax_zoom) which will be ignored by the endpoint. I think it's fine letting those in, in case someone as a custom endpoint.

@vincentsarago vincentsarago merged commit 72a8df9 into developmentseed:main Dec 20, 2024
10 checks passed
@vincentsarago vincentsarago mentioned this pull request Dec 20, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

add support for the STAC Render extension
3 participants