Skip to content

Commit

Permalink
Add support for parsing IVOA POS string
Browse files Browse the repository at this point in the history
  • Loading branch information
timj committed May 8, 2024
1 parent bd99824 commit f36e789
Show file tree
Hide file tree
Showing 2 changed files with 194 additions and 0 deletions.
126 changes: 126 additions & 0 deletions python/lsst/sphgeom/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@

"""lsst.sphgeom
"""
import typing

from ._healpixPixelization import *
from ._sphgeom import *
Expand All @@ -37,3 +38,128 @@
from .version import *

PixelizationABC.register(Pixelization)

# Copy and paste from lsst.utils.wrappers
_INTRINSIC_SPECIAL_ATTRIBUTES = frozenset(
(
"__qualname__",
"__module__",
"__metaclass__",
"__dict__",
"__weakref__",
"__class__",
"__subclasshook__",
"__name__",
"__doc__",
)
)


def _isAttributeSafeToTransfer(name: str, value: typing.Any) -> bool:
if name.startswith("__") and (
value is getattr(object, name, None) or name in _INTRINSIC_SPECIAL_ATTRIBUTES
):
return False
return True


def _continueClass(cls):
import sys
orig = getattr(sys.modules[cls.__module__], cls.__name__)
for name in dir(cls):
# Common descriptors like classmethod and staticmethod can only be
# accessed without invoking their magic if we use __dict__; if we use
# getattr on those we'll get e.g. a bound method instance on the dummy
# class rather than a classmethod instance we can put on the target
# class.
attr = cls.__dict__.get(name, None) or getattr(cls, name)
if _isAttributeSafeToTransfer(name, attr):
setattr(orig, name, attr)
return orig


def _inf_to_lat(lat: float) -> float:
"""Map +Inf to +90 and -Inf to -90 degrees."""
import math
if not math.isinf(lat):
return lat
if lat > 0.0:
return 90.0
return -90.0


@_continueClass
class Region:

@classmethod
def from_ivoa_pos(cls, pos: str) -> Region:
"""Create a Region from an IVOA POS string.
Parameters
----------
pos : `str`
A string using the IVOA SIAv2 POS syntax.
Returns
-------
region : `Region`
A region equivalent to the POS string.
Notes
-----
See
https://ivoa.net/documents/SIA/20151223/REC-SIA-2.0-20151223.html#toc12
for a description of the POS parameter but in summary the options are:
* ``CIRCLE <longitude> <latitude> <radius>``
* ``RANGE <longitude1> <longitude2> <latitude1> <latitude2>``
* ``POLYGON <longitude1> <latitude1> ... (at least 3 pairs)``
Units are degrees in all coordinates.
"""
shape, *coordinates = pos.split()
coordinates = tuple(float(c) for c in coordinates)
n_floats = len(coordinates)
if shape == "CIRCLE":
if n_floats != 3:
raise ValueError(f"CIRCLE requires 3 numbers but got {n_floats} in '{pos}'.")
center = LonLat.fromDegrees(coordinates[0], coordinates[1])
radius = Angle.fromDegrees(coordinates[2])
return Circle(UnitVector3d(center), radius)
elif shape == "RANGE":
import math

if n_floats != 4:
raise ValueError(f"RANGE requires 4 numbers but got {n_floats} in '{pos}'.")
# POS allows +Inf and -Inf in ranges. These are not allowed by
# Box and so must be converted.
# If either of the longitude values are infinite, all longitudes
# should be included.
if math.isinf(coordinates[0]) or math.isinf(coordinates[1]):
return Box(
NormalizedAngleInterval.fromDegrees(0., 360.),
AngleInterval.fromDegrees(
_inf_to_lat(coordinates[2]), _inf_to_lat(coordinates[3])
)
)
else:
return Box(
LonLat.fromDegrees(coordinates[0], _inf_to_lat(coordinates[2])),
LonLat.fromDegrees(coordinates[1], _inf_to_lat(coordinates[3]))
)
elif shape == "POLYGON":
if n_floats % 2 != 0:
raise ValueError(f"POLYGON requires even number of floats but got {n_floats} in '{pos}'.")
if n_floats < 6:
raise ValueError(
f"POLYGON specification requires at least 3 coordinates, got {n_floats // 2} in '{pos}'"
)
# Coordinates are x1, y1, x2, y2, x3, y3...
# Get pairs by skipping every other value.
pairs = list(zip(coordinates[0::2], coordinates[1::2]))
vertices = [LonLat.fromDegrees(lon, lat) for lon, lat in pairs]
return ConvexPolygon([UnitVector3d(c) for c in vertices])

raise ValueError(f"Unrecognized shape in POS string '{pos}'")

del typing
68 changes: 68 additions & 0 deletions tests/test_ivoa.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
# This file is part of sphgeom.
#
# Developed for the LSST Data Management System.
# This product includes software developed by the LSST Project
# (http://www.lsst.org).
# See the COPYRIGHT file at the top-level directory of this distribution
# for details of code ownership.
#
# This software is dual licensed under the GNU General Public License and also
# under a 3-clause BSD license. Recipients may choose which of these licenses
# to use; please see the files gpl-3.0.txt and/or bsd_license.txt,
# respectively. If you choose the GPL option then the following text applies
# (but note that there is still no warranty even if you opt for BSD instead):
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.

import unittest

from lsst.sphgeom import (
Region,
)


class IvoaTestCase(unittest.TestCase):
"""Test Box."""

def test_construction(self):
# Example POS strings found in IVOA documentation.
example_pos = (
"CIRCLE 12.0 34.0 0.5",
"RANGE 12.0 12.5 34.0 36.0",
"POLYGON 12.0 34.0 14.0 35.0 14. 36.0 12.0 35.0",
"RANGE 0 360.0 -2.0 2.0",
"RANGE 0 360.0 89.0 +Inf",
"RANGE -Inf +Inf -Inf +Inf",
"POLYGON 12 34 14 34 14 36 12 36",
"RANGE 0 360 89 90",
)
for pos in example_pos:
region = Region.from_ivoa_pos(pos)
self.assertIsInstance(region, Region)

# Badly formed strings raising ValueError.
bad_pos = (
"circle 12 34 0.5",
"CIRCLE 12 34 1 1",
"RANGE 0 360",
"POLYGON 0 1 2 3",
"POLYGON 0 1 2 3 4 5 6",
)
for pos in bad_pos:
with self.assertRaises(ValueError):
Region.from_ivoa_pos(pos)


if __name__ == "__main__":
unittest.main()

0 comments on commit f36e789

Please sign in to comment.