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

Coordinate axis order in northing/easting (Y/X) in WFS/GML #16

Open
travenin opened this issue Jan 10, 2023 · 6 comments
Open

Coordinate axis order in northing/easting (Y/X) in WFS/GML #16

travenin opened this issue Jan 10, 2023 · 6 comments

Comments

@travenin
Copy link

My project for City of Helsinki uses coordinate reference system EPSG:3879. We have a problem with latitude and longitude axis order in WFS GML output when using django-gisserver version 1.2.4.

WFS 2.0.0 (with GML 3.2) should respects the axis order defined by the EPSG definition (unlike WFS 1.0.0 which is always easting/northing or (X,Y)). EPSG:3879 defines coordinate axis order as northing/easting (Y,X).

For query <server url>/wfs/?SERVICE=WFS&VERSION=2.0.0&REQUEST=GetFeature&TYPENAMES=signpost&OUTPUTFORMAT=application/gml+xml&srsName=urn:ogc:def:crs:EPSG::3879 we are expecting:

*snip*
<app:location>
  <gml:Point srsName="urn:ogc:def:crs:EPSG::3879" gml:id="signpost.1">
    <gml:pos srsDimension="3">6673364.2700978 25494172.7053077 0</gml:pos>
  </gml:Point>
</app:location>
*snip*

where 6673364.2700978 is latitude (Y) and 25494172.7053077 is longitude (X).

However, the actual result has invert latitude and longitude:

*snip*
<app:location>
  <gml:Point srsName="urn:ogc:def:crs:EPSG::3879" gml:id="signpost.1">
    <gml:pos srsDimension="3">25494172.7053077 6673364.2700978 0</gml:pos>
  </gml:Point>
</app:location>
*snip*

Do I have something wrong in my configuration? Or is this a missing feature or bug in django-gisserver?

@lbam
Copy link
Contributor

lbam commented Jan 11, 2023

Looks like missing functionality. We just render the coordinates in the order django.contrib.gis.geos.Point.coords has them.

@vdboor
Copy link
Contributor

vdboor commented Mar 13, 2023

Another factor here might be the used Django version and GDAL version. As of GDAL 3, the axis ordering was changed. Django 3.1 and later have support for that.

In gisserver/geometries.py you'll find some comments regarding this issue. What you might be able to do, is pass a custom backend to the constructed CRS object. For example:

from django.contrib.gis.gdal import AxisOrder, SpatialReference
from gisserver.geometries import CRS

# Define the actual GDAL backend yourself, so the CRS object will use that:
gk25fin = CRS("EPSG:3879", backend=SpatialReference(3879, srs_type="epsg", axis_order=AxisOrder.AUTHORITY))

# Pass the CRS object to the feature type:
feature_types = [
    FeatureType(
        ...,
        crs=gk25fin,
        other_crs=[..., gk25fin],  # or this one.
    )
]

The rest of the code really just reads what GDAL will provide (like @lbam mentioned).

For some background info, see:

Hope this helps!

and bye the way, we love that you guys started using this project too! (and for that matter completely extended it as well, exactly as the design intended).

@jforsman
Copy link

jforsman commented Dec 9, 2024

Well, atleast does not work like this, any ideas? I will dig in further soonish anyways...

DEFAULT_CRS = CRS.from_srid(3879, backend=SpatialReference(3879, srs_type="epsg", axis_order=AxisOrder.AUTHORITY))

@jforsman
Copy link

jforsman commented Dec 9, 2024

Well, I gave up and decided to just overwrite a couple of more functions from GML32Renderer

@jforsman
Copy link

jforsman commented Dec 9, 2024

btw. do you consider this as a feature or a bug?
@register_geos_type(geos.MultiPolygon) def render_gml_multi_geometry(self, value: geos.GeometryCollection, base_attrs): children = "".join(self._render_gml_type(child) for child in value) return f"<gml:MultiGeometry{base_attrs}>{children}</gml:MultiGeometry>"
Currently there is not any rendered for GeometryCollections

@vdboor
Copy link
Contributor

vdboor commented Jan 16, 2025

@jforsman are you using the Python-based rendering, or database based rendering? By default (GISSERVER_USE_DB_RENDERING=True), so the GML bits will be rendered in PostgresSQL. This turned out to be way more efficient than letting Python do it. The GeoDjango code performs a C-API call for every coordinate of the geometry, hence it's much slower.

If the Python code doesn't properly render a <gml:MultiGeometry>, that's indeed a bug.

I'm also very curious what value.ewkb will return. This should reveal how the value is held in memory

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

No branches or pull requests

4 participants