Opened 7 months ago

Closed 7 months ago

#5703 closed defect (wontfix)

ST_Intersects function is returning the wrong results for a simple rectangle that crosses any two quarter longitude lines

Reported by: hangstrap Owned by: pramsey
Priority: medium Milestone: PostGIS 3.4.3
Component: postgis Version: 3.4.x
Keywords: Cc: hangstrap

Description

the ST_Intersects function is returning the wrong results for a simple rectangle that crosses any two quarter longitude lines (e.g 0, 90, 180, -90).

Running the following query returns the points outside the polygon as well as the expected points inside

select postgis.st_astext( point_) from 
(
	with foo ( point_  ) as ( values  
		( postgis.ST_GeogFromText( 'POINT (-6 0)')), 
		( postgis.ST_GeogFromText( 'POINT (-4 0)')),
		( postgis.ST_GeogFromText( 'POINT (94 0)')),
		( postgis.ST_GeogFromText( 'POINT (96 0)'))
	)
	select * from foo
) bar
where postgis.ST_Intersects( point_, postgis.ST_GeogFromText('POLYGON ((  -5 -10,  -5 10,  95.0 10, 95.0 -10 , -5 -10))'));

The select statement should return the points (-4 0) and (94 0), but returns all four points.

This select statement, that does not cross two quarter longitude lines, works correctly

--works only closses 0 line
select postgis.st_astext( point_) from 
(
	with foo ( point_  ) as ( values  
		( postgis.ST_GeogFromText( 'POINT (-51 0)')), 
		( postgis.ST_GeogFromText( 'POINT (-49 0)')),
		( postgis.ST_GeogFromText( 'POINT (49 0)')),
		( postgis.ST_GeogFromText( 'POINT (51 0)'))
	)
	select * from foo
) bar
where postgis.ST_Intersects( point_, postgis.ST_GeogFromText('POLYGON ((  -50 -10,  -50 10,  50 10, 50 -10 , -50 -10))'));

Tested on PostgreSQL 12.18 on x86_64-pc-linux-gnu, compiled by gcc (GCC) 8.5.0 20210514 (Red Hat 8.5.0-20), 64-bit

With Postgis

POSTGIS="3.1.11 ca03d62" [EXTENSION] PGSQL="120" GEOS="3.12.1-CAPI-1.18.1" PROJ="9.2.1" LIBXML="2.9.7" LIBJSON="0.13.1" LIBPROTOBUF="1.3.0" WAGYU="0.5.0 (Internal)"

and

POSTGIS="3.4.2 c19ce56" [EXTENSION] PGSQL="120" GEOS="3.12.1-CAPI-1.18.1" PROJ="9.2.1 NETWORK_ENABLED=OFF URL_ENDPOINT=https://cdn.proj.org USER_WRITABLE_DIRECTORY=/var/lib/pgsql/.local/share/proj DATABASE_PATH=/usr/proj92/share/proj/proj.db" LIBXML="2.9.7" LIBJSON="0.13.1" LIBPROTOBUF="1.3.0" WAGYU="0.5.0 (Internal)" (core procs from "3.1.11 ca03d62" need upgrade)

The queries do work as expected on postgres 10, postgis 2.4.8

Attachments (1)

foo.jpg (52.5 KB ) - added by pramsey 7 months ago.

Download all attachments as: .zip

Change History (7)

comment:1 by pramsey, 7 months ago

Seems true.

SELECT 
  ST_AsText(g) AS geog,
  ST_Intersects(ST_GeogFromText('POLYGON ((  -5 -10,  -5 10,  95.0 10, 95.0 -10 , -5 -10))'), v.g) 
FROM
  (VALUES
    ( ST_GeogFromText( 'POINT (-6 0)')), 
    ( ST_GeogFromText( 'POINT (-4 0)')),
    ( ST_GeogFromText( 'POINT (94 0)')),
    ( ST_GeogFromText( 'POINT (96 0)'))
    ) v(g);

by pramsey, 7 months ago

Attachment: foo.jpg added

comment:2 by pramsey, 7 months ago

Curiouser and curiouser, you can get wrong answers one at a time, on both sides of the line...

SELECT 
  ST_AsText(g) AS geog,
  ST_Intersects(ST_GeogFromText('POLYGON ((  -5 -10,  -5 10,  95.0 10, 95.0 -10 , -5 -10))'), v.g) 
FROM
  (VALUES
    ( ST_GeogFromText( 'POINT (-4 0)'))
    ) v(g);

    geog     | st_intersects 
-------------+---------------
 POINT(-4 0) | f

comment:3 by pramsey, 7 months ago

Peeling back the union further, the caching code can be taken out of consideration too,

SELECT 
  _ST_DWithinUnCached(
  'POLYGON ((  -5 -10,  -5 10,  95.0 10, 95.0 -10 , -5 -10))'::geography, 
  'POINT (-6 0)'::geography,
  0.0,
  true
  );

 _st_dwithinuncached 
---------------------
 t
Version 0, edited 7 months ago by pramsey (next)

comment:4 by pramsey, 7 months ago

This is an effect of ring orientation?

-- CCW (bounds small area, pt is outside)
SELECT _ST_DistanceUnCached(
  'POLYGON ((  -5 10,  -5 -10,  95.0 -10, 95.0 10 , -5 10))'::geography, 
  'POINT (-6 0)'::geography
  );

-- CW (bounds everything except small area, pt is inside)
SELECT _ST_DistanceUnCached(
  'POLYGON ((  -5 -10,  -5 10,  95.0 10, 95.0 -10 , -5 -10))'::geography, 
  'POINT (-6 0)'::geography
  );

But I cannot remember handling ring orientation.

comment:5 by pramsey, 7 months ago

And in general we do not handle ring orientation, but, when multiple corner cases drive us to extremes... we fall back on ring orientation.

https://github.com/postgis/postgis/blob/8c30a25acb166de09a7d647e1e02dc6d71eae9ea/liblwgeom/lwgeodetic.c#L1429-L1441

So the "answer" is: even though we don't, in general respect ring orientation in geography, if you fail to respect it there are corner cases where you'll get the wrong answer, like this case.

comment:6 by pramsey, 7 months ago

Resolution: wontfix
Status: newclosed
Note: See TracTickets for help on using tickets.