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

Design how *blessed* implementation of GeoInterface would work as generic fallback. #141

Open
evetion opened this issue Jul 7, 2024 · 9 comments
Assignees

Comments

@evetion
Copy link
Member

evetion commented Jul 7, 2024

Or how we could use GeometryOps as a default implementation of methods such GeoInterface.area, for packages that do define a geometry, but not the method.

If GeometryOps now would implement GeoInterface.area(::geomtrait, geom), it would be type piracy.

@evetion evetion converted this from a draft issue Jul 7, 2024
@evetion evetion changed the title Design blessed implementation Design how *blessed* implementation of GeoInterface would work as generic fallback. Jul 7, 2024
@evetion
Copy link
Member Author

evetion commented Jul 7, 2024

Random thought:

We choose a single blessed package, and implement the "type piracy" method as a package extension here at GeoInterface.

So something like GeometryOpsExt.jl will have area(::geomtrait, geom::Any) = GeometryOps.area(::geomtrait, convert(:GeometryOps, geom))?

Then again, single blessed packages are restrictive (and no single one might've fully implemented the spec) so we could also go with the above package extension route, but add dispatch on the module (possible in 1.10, will be LTS soon):

area(::geomtrait, geom::Any) = area(::Val{backend}, ::geomtrait, convert(backend, geom)),
and we introduce a get_backend() method that users can set/call, or is set by the package extension (last or first package loaded comes first?).

edit: The Val{backend} way also enables implementations of several packages directly on our Wrapper geometries, without introducing type piracy, so we don't need to do the convert.

@evetion
Copy link
Member Author

evetion commented Jul 7, 2024

Might be useful for inspiration on what to do, or probably, what not to do from Plots: https://github.com/JuliaPlots/Plots.jl/blob/master/src/backends.jl

@asinghvi17
Copy link
Member

The package extension here would be a circular dependency, since any such package would have depend on GeoInterface. This is the same issue with GeoInterfaceMakie and GeoInterface geoms.

@asinghvi17
Copy link
Member

asinghvi17 commented Jul 7, 2024

set by the package extension (last or first package loaded comes first?)

seems pretty fragile - what if a dependency loads one of the packages that sets a backend?

Maybe a set priority list would be better but I would rather centralize all of this in GeometryOps, at least until we understand how people use the design. It already has a LibGEOS backend for example, we could simply add more (GDAL, S2) as required.

@evetion
Copy link
Member Author

evetion commented Jul 7, 2024

The package extension here would be a circular dependency, since any such package would have depend on GeoInterface. This is the same issue with GeoInterfaceMakie and GeoInterface geoms.

Hmm, I keep forgetting about that. :/

Well then, packages could implement area(::Val{package}, ::geomtrait, geom::Any/Wrapper), and we have a method here that you can set? Either user facing like Plots with setting gr() or pyplot(), or something in the blessed packages?

@rafaqz
Copy link
Member

rafaqz commented Jul 7, 2024

I think we should instead deprecate GeoInterface methods like area.

Using LibGEOS and GeometryOps methods directly is simpler and works on all objects without confusion over what package is doing what.

But I think we should standardise the function form and arguments in the implementing packages.

@evetion
Copy link
Member Author

evetion commented Jul 8, 2024

But I think we should standardise the function form and arguments in the implementing packages.

That seems like a no-regret action to me, like #134 for read/write.

However, ideally we would somehow have a (correct for crs trait) default fallback that just works, just like Plots works out of the box. But it might be too hard to do correctly 🤷🏻 ?

@rafaqz
Copy link
Member

rafaqz commented Jul 8, 2024

read/write is a good example! It's basically the same problem.

So we're talking about an "interface" where package specific functions have an identical definition following a specification, rather than methods extending a single function.

That's an interesting kind of interface. For ecosystem level user experience consistency rather than anything technical.

We could test it with a test function that accepts the function as an argument and calls it on generic GeoInterface geoms and keywords.

@evetion
Copy link
Member Author

evetion commented Jul 11, 2024

Ok, after our nice little talk and some reflection, let's deprecate the operations here 💥.

Motivation; accessing geometries is fundamentally different from operations on geometries. Different type of geometries often come from file drivers (99% of our implementations), and have no operations. Operations are hard and scarce.

Impact of doing this is probably minimal, and for the packages implementing area and such we should still implement GeoInterface.area(::Any) and related interface methods as fallback to not "break" the contract. But we do not document it, and only document GO for operations.

In terms of operations we keep the package/module trait (area(LibGEOS(), stuff), so we can compare our own implementations, users can choose which implementation they want, and in rare cases other developers can extent these operations and make another package extension.

We might want to reconsider LibGEOS() and go for Val{LibGEOS} (as in the module) if that produces a more elegant design.

@asinghvi17 asinghvi17 moved this from Todo to In Progress in JuliaCon 2024 Hackathon Jul 13, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
Status: In Progress
Development

No branches or pull requests

3 participants