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

Refactor for boundary lat/lon extraction #546

Open
wants to merge 40 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
e203899
Add refactor
ghiggi Oct 10, 2023
95decd3
Fix wrong import
ghiggi Oct 10, 2023
ae730f7
Lint
ghiggi Oct 10, 2023
edcfcb4
Fix typo
ghiggi Oct 11, 2023
eae1873
Fix issues
ghiggi Oct 11, 2023
1662d2d
Fix issues
ghiggi Oct 11, 2023
655fb72
Fix issues
ghiggi Oct 11, 2023
b7df9be
Improve doc
ghiggi Oct 11, 2023
a5322a6
Improve doc
ghiggi Oct 11, 2023
88065d7
Remove old use of frequency argument
ghiggi Oct 11, 2023
0e992a2
Merge branch 'main' into refactor-get_bbox-lonlat
djhoese Nov 22, 2023
8eff9b4
Address refactor boundary sides
ghiggi Nov 22, 2023
36795df
Fix tests related to gradient resampling
ghiggi Nov 22, 2023
1dca712
Deprecate get_edge_lonlat and remove duplicated coordinates
ghiggi Nov 22, 2023
3c60e74
Ensure shapely polygon is closed !
ghiggi Nov 22, 2023
bba3bde
Refactor get_polygon and get_border_lonlat in gradient
ghiggi Nov 22, 2023
34edc5c
Initial refactor of AreaBoundary
ghiggi Nov 22, 2023
8fbf02f
Deprecate get_bbox_lonlats
ghiggi Nov 23, 2023
d9f4165
Private get_polygon and get_border_lonlat in gradient.__init__
ghiggi Nov 23, 2023
83ec6df
Add test area fixtures
ghiggi Nov 23, 2023
9de0a44
Treat geo area inside earth disk as classical AreaDef
ghiggi Nov 23, 2023
efd0773
Deprecate get_edge_bbox_in_projection_coordinates
ghiggi Nov 23, 2023
0b669f8
Refactor boundary classes and ensure backward compatibilities
ghiggi Nov 23, 2023
cea363f
Add BoundarySides test units
ghiggi Nov 23, 2023
b9cda35
Add test units for visualizion utilities
ghiggi Nov 23, 2023
d41973a
Consistent naming across repo of lon/lat sides with sides_lons and si…
ghiggi Nov 23, 2023
c6a9378
Add geographic_boundary and projection_boundary methods
ghiggi Nov 23, 2023
6323814
Deprecate boundary() for geographic_boundary()
ghiggi Nov 24, 2023
6210bea
Pass AreaDef crs to ProjectionBoundary crs
ghiggi Nov 24, 2023
28b5527
Solve wasted polygon computations in gradient for GEO FD
ghiggi Nov 24, 2023
0eb7a6b
Deprecate get_boundary_lonlats and SimpleBoundary
ghiggi Nov 24, 2023
0f1b578
Remove use of __file__ and relative paths in test units
ghiggi Nov 24, 2023
059d496
Improve clarity of sides extraction for geographic and projection coo…
ghiggi Nov 24, 2023
d35f1b5
Refactor Geographic and Projection Boundary using composition and dep…
ghiggi Nov 24, 2023
b3273b8
Deal with AreaDefinition with unvalid sides: polar projections, globa…
ghiggi Nov 25, 2023
e3f4b24
Add code structure for boundary extraction for polar and global plana…
ghiggi Nov 25, 2023
915e34f
Fix boundary ordering logic for swath and projections !
ghiggi Nov 27, 2023
61240eb
Fix formatting warnings
ghiggi Nov 27, 2023
7069197
Renaming to SphericalBoundary and PlanarBoundary
ghiggi Nov 27, 2023
e638557
Some cleanout and notes
ghiggi Nov 27, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion pyresample/future/geometry/area.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,9 @@
from pyresample.geometry import AreaDefinition as LegacyAreaDefinition # noqa
from pyresample.geometry import ( # noqa
DynamicAreaDefinition,
_get_geostationary_bounding_box_in_lonlats,
get_full_geostationary_bounding_box_in_proj_coords,
get_geostationary_angle_extent,
get_geostationary_bounding_box_in_lonlats,
get_geostationary_bounding_box_in_proj_coords,
ignore_pyproj_proj_warnings,
)
Expand Down
153 changes: 74 additions & 79 deletions pyresample/geometry.py
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,7 @@
HashType = hashlib._hashlib.HASH



class DimensionError(ValueError):
"""Wrap ValueError."""

Expand Down Expand Up @@ -279,14 +280,19 @@ def get_boundary_lonlats(self):
return (SimpleBoundary(s1_lon.squeeze(), s2_lon.squeeze(), s3_lon.squeeze(), s4_lon.squeeze()),
SimpleBoundary(s1_lat.squeeze(), s2_lat.squeeze(), s3_lat.squeeze(), s4_lat.squeeze()))

@property
def is_geostationary(self):
return False

def get_bbox_lonlats(self, vertices_per_side: Optional[int] = None, force_clockwise: bool = True,
frequency: Optional[int] = None) -> tuple:
"""Return the bounding box lons and lats.
"""Return the bounding box lons and lats sides.

Args:
vertices_per_side:
The number of points to provide for each side. By default (None)
the full width and height will be provided.
the full width and height will be provided, except for geostationary
projections where by default only 50 points are selected.
frequency:
Deprecated, use vertices_per_side
force_clockwise:
Expand Down Expand Up @@ -318,14 +324,49 @@ def get_bbox_lonlats(self, vertices_per_side: Optional[int] = None, force_clockw
warnings.warn("The `frequency` argument is pending deprecation, use `vertices_per_side` instead",
PendingDeprecationWarning, stacklevel=2)
vertices_per_side = vertices_per_side or frequency
lons, lats = self._get_bbox_elements(self.get_lonlats, vertices_per_side)
if self.is_geostationary:
lon_sides, lat_sides = self._get_geo_boundary_sides(vertices_per_side=vertices_per_side)
else:
lon_sides, lat_sides = self._get_boundary_sides(self.get_lonlats, vertices_per_side)
if force_clockwise and not self._corner_is_clockwise(
lons[0][-2], lats[0][-2], lons[0][-1], lats[0][-1], lons[1][1], lats[1][1]):
lon_sides[0][-2], lat_sides[0][-2],
lon_sides[0][-1], lat_sides[0][-1],
lon_sides[1][1], lat_sides[1][1]):
# going counter-clockwise
lons, lats = self._reverse_boundaries(lons, lats)
return lons, lats
lon_sides, lat_sides = self._reverse_boundaries(lon_sides, lat_sides)
return lon_sides, lat_sides


def _get_bbox_elements(self, coord_fun, vertices_per_side: Optional[int] = None) -> tuple:
def _get_geo_boundary_sides(self, vertices_per_side=None):
ghiggi marked this conversation as resolved.
Show resolved Hide resolved
"""Retrieve the boundary sides list for geostationary projections."""
# Define default frequency
if vertices_per_side is None:
vertices_per_side = 50
# Ensure at least 4 points are used
if vertices_per_side < 4:
vertices_per_side = 4
# Ensure an even number of vertices for side creation
if (vertices_per_side % 2) != 0:
vertices_per_side = vertices_per_side + 1
lons, lats = _get_geostationary_bounding_box_in_lonlats(self, nb_points=vertices_per_side)

# Retrieve dummy sides for GEO (side1 and side3 always of length 2)
# - TODO: _get_geostationary_bounding_box_in_lonlats now does not return nb_points !
side02_step = int(lons.shape[0] / 2) - 1 #
ghiggi marked this conversation as resolved.
Show resolved Hide resolved
lon_sides = [lons[slice(0, side02_step + 1)],
lons[slice(side02_step, side02_step + 1 + 1)],
lons[slice(side02_step + 1, side02_step * 2 + 1 + 1)],
np.append(lons[side02_step * 2 + 1], lons[0])
]
lat_sides = [lats[slice(0, side02_step + 1)],
lats[slice(side02_step, side02_step + 1 + 1)],
lats[slice(side02_step + 1, side02_step * 2 + 1 + 1)],
np.append(lats[side02_step * 2 + 1], lats[0])
]
return lon_sides, lat_sides


def _get_boundary_sides(self, coord_fun, vertices_per_side: Optional[int] = None) -> tuple:
s1_slice, s2_slice, s3_slice, s4_slice = self._get_bbox_slices(vertices_per_side)
s1_dim1, s1_dim2 = coord_fun(data_slice=s1_slice)
s2_dim1, s2_dim2 = coord_fun(data_slice=s2_slice)
Expand All @@ -337,10 +378,10 @@ def _get_bbox_elements(self, coord_fun, vertices_per_side: Optional[int] = None)
(s4_dim1.squeeze(), s4_dim2.squeeze())])
if hasattr(dim1[0], 'compute') and da is not None:
dim1, dim2 = da.compute(dim1, dim2)
clean_dim1, clean_dim2 = self._filter_bbox_nans(dim1, dim2)
clean_dim1, clean_dim2 = self._filter_sides_nans(dim1, dim2)
return clean_dim1, clean_dim2

def _filter_bbox_nans(
def _filter_sides_nans(
self,
dim1_sides: list[np.ndarray],
dim2_sides: list[np.ndarray],
Expand Down Expand Up @@ -423,17 +464,18 @@ def get_edge_bbox_in_projection_coordinates(self, vertices_per_side: Optional[in
warnings.warn("The `frequency` argument is pending deprecation, use `vertices_per_side` instead",
PendingDeprecationWarning, stacklevel=2)
vertices_per_side = vertices_per_side or frequency
x, y = self._get_bbox_elements(self.get_proj_coords, vertices_per_side)
x, y = self._get_boundary_sides(self.get_proj_coords, vertices_per_side)
return np.hstack(x), np.hstack(y)

def boundary(self, vertices_per_side=None, force_clockwise=False, frequency=None):
def boundary(self, *, vertices_per_side=None, force_clockwise=False, frequency=None):
"""Retrieve the AreaBoundary object.

Parameters
----------
vertices_per_side:
(formerly `frequency`) The number of points to provide for each side. By default (None)
the full width and height will be provided.
the full width and height will be provided, except for geostationary
projection where by default only 50 points are selected.
force_clockwise:
Perform minimal checks and reordering of coordinates to ensure
that the returned coordinates follow a clockwise direction.
Expand Down Expand Up @@ -1570,65 +1612,6 @@ def is_geostationary(self):
return False
return 'geostationary' in coord_operation.method_name.lower()

def _get_geo_boundary_sides(self, vertices_per_side=None):
"""Retrieve the boundary sides list for geostationary projections."""
# Define default frequency
if vertices_per_side is None:
vertices_per_side = 50
# Ensure at least 4 points are used
if vertices_per_side < 4:
vertices_per_side = 4
# Ensure an even number of vertices for side creation
if (vertices_per_side % 2) != 0:
vertices_per_side = vertices_per_side + 1
lons, lats = get_geostationary_bounding_box_in_lonlats(self, nb_points=vertices_per_side)
# Retrieve dummy sides for GEO (side1 and side3 always of length 2)
side02_step = int(vertices_per_side / 2) - 1
lon_sides = [lons[slice(0, side02_step + 1)],
lons[slice(side02_step, side02_step + 1 + 1)],
lons[slice(side02_step + 1, side02_step * 2 + 1 + 1)],
np.append(lons[side02_step * 2 + 1], lons[0])
]
lat_sides = [lats[slice(0, side02_step + 1)],
lats[slice(side02_step, side02_step + 1 + 1)],
lats[slice(side02_step + 1, side02_step * 2 + 1 + 1)],
np.append(lats[side02_step * 2 + 1], lats[0])
]
return lon_sides, lat_sides

def boundary(self, *, vertices_per_side=None, force_clockwise=False, frequency=None):
"""Retrieve the AreaBoundary object.

Parameters
----------
vertices_per_side:
The number of points to provide for each side. By default (None)
the full width and height will be provided, except for geostationary
projection where by default only 50 points are selected.
frequency:
Deprecated, use vertices_per_side
force_clockwise:
Perform minimal checks and reordering of coordinates to ensure
that the returned coordinates follow a clockwise direction.
This is important for compatibility with
:class:`pyresample.spherical.SphPolygon` where operations depend
on knowing the inside versus the outside of a polygon. These
operations assume that coordinates are clockwise.
Default is False.
"""
from pyresample.boundary import AreaBoundary
if frequency is not None:
warnings.warn("The `frequency` argument is pending deprecation, use `vertices_per_side` instead",
PendingDeprecationWarning, stacklevel=2)
vertices_per_side = vertices_per_side or frequency
if self.is_geostationary:
lon_sides, lat_sides = self._get_geo_boundary_sides(vertices_per_side=vertices_per_side)
else:
lon_sides, lat_sides = self.get_bbox_lonlats(vertices_per_side=vertices_per_side,
force_clockwise=force_clockwise)
boundary = AreaBoundary.from_lonlat_sides(lon_sides, lat_sides)
return boundary

@property
def area_extent(self):
"""Tuple of this area's extent (xmin, ymin, xmax, ymax)."""
Expand Down Expand Up @@ -2743,9 +2726,10 @@ def geocentric_resolution(self, ellps='WGS84', radius=None):
def _get_area_boundary(area_to_cover: AreaDefinition) -> Boundary:
try:
if area_to_cover.is_geostationary:
return Boundary(*get_geostationary_bounding_box_in_lonlats(area_to_cover))
boundary_shape = max(max(*area_to_cover.shape) // 100 + 1, 3)
return area_to_cover.boundary(frequency=boundary_shape, force_clockwise=True)
vertices_per_side = None
else:
vertices_per_side = max(max(*area_to_cover.shape) // 100 + 1, 3)
return area_to_cover.boundary(vertices_per_side=vertices_per_side, force_clockwise=True)
ghiggi marked this conversation as resolved.
Show resolved Hide resolved
except ValueError:
raise NotImplementedError("Can't determine boundary of area to cover")

Expand Down Expand Up @@ -2832,11 +2816,10 @@ def get_full_geostationary_bounding_box_in_proj_coords(geos_area, nb_points=50):
y = -np.sin(points_around) * (y_max_angle - 0.0001)
x *= h
y *= h

return x, y


def get_geostationary_bounding_box_in_lonlats(geos_area, nb_points=50):
def _get_geostationary_bounding_box_in_lonlats(geos_area, nb_points=50):
"""Get the bbox in lon/lats of the valid pixels inside `geos_area`.

Args:
Expand All @@ -2847,16 +2830,28 @@ def get_geostationary_bounding_box_in_lonlats(geos_area, nb_points=50):
return lons, lats


def get_geostationary_bounding_box_in_lonlats(geos_area, nb_points=50):
"""Get the bbox in lon/lats of the valid pixels inside `geos_area`.

Args:
nb_points: Number of points on the polygon
"""
warnings.warn("'get_geostationary_bounding_box_in_lonlats' is deprecated. Please call "
"'area.boundary().contour()' instead.",
DeprecationWarning, stacklevel=2)
return _get_geostationary_bounding_box_in_lonlats(geos_area, nb_points)


def get_geostationary_bounding_box(geos_area, nb_points=50):
"""Get the bbox in lon/lats of the valid pixels inside `geos_area`.

Args:
nb_points: Number of points on the polygon
"""
warnings.warn("'get_geostationary_bounding_box' is deprecated. Please use "
"'get_geostationary_bounding_box_in_lonlats' instead.",
warnings.warn("'get_geostationary_bounding_box' is deprecated. Please call "
"'area.boundary().contour()' instead.",
DeprecationWarning, stacklevel=2)
return get_geostationary_bounding_box_in_lonlats(geos_area, nb_points)
return _get_geostationary_bounding_box_in_lonlats(geos_area, nb_points)


def combine_area_extents_vertical(area1, area2):
Expand Down
6 changes: 3 additions & 3 deletions pyresample/gradient/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@
from pyresample.geometry import (
AreaDefinition,
SwathDefinition,
get_geostationary_bounding_box_in_lonlats,
_get_geostationary_bounding_box_in_lonlats,
)
from pyresample.gradient._gradient_search import (
one_step_gradient_indices,
Expand Down Expand Up @@ -375,13 +375,13 @@ def _check_input_coordinates(dst_x, dst_y,

def get_border_lonlats(geo_def: AreaDefinition):
"""Get the border x- and y-coordinates."""
# TODO: we could use geo_def.boundary().contour() here
if geo_def.is_geostationary:
lon_b, lat_b = get_geostationary_bounding_box_in_lonlats(geo_def, 3600)
lon_b, lat_b = _get_geostationary_bounding_box_in_lonlats(geo_def, 3600)
else:
lons, lats = geo_def.get_boundary_lonlats()
lon_b = np.concatenate((lons.side1, lons.side2, lons.side3, lons.side4))
lat_b = np.concatenate((lats.side1, lats.side2, lats.side3, lats.side4))

return lon_b, lat_b


Expand Down
13 changes: 6 additions & 7 deletions pyresample/test/test_geometry/test_area.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,9 +27,9 @@
from pyresample import geo_filter, parse_area_file
from pyresample.future.geometry import AreaDefinition, SwathDefinition
from pyresample.future.geometry.area import (
_get_geostationary_bounding_box_in_lonlats,
get_full_geostationary_bounding_box_in_proj_coords,
get_geostationary_angle_extent,
get_geostationary_bounding_box_in_lonlats,
get_geostationary_bounding_box_in_proj_coords,
ignore_pyproj_proj_warnings,
)
Expand Down Expand Up @@ -1493,8 +1493,7 @@ def test_get_full_geostationary_bbox(self, truncated_geos_area):

def test_get_geostationary_bbox_works_with_truncated_area(self, truncated_geos_area):
"""Ensure the geostationary bbox works when truncated."""
lon, lat = get_geostationary_bounding_box_in_lonlats(truncated_geos_area, 20)

lon, lat = _get_geostationary_bounding_box_in_lonlats(truncated_geos_area, 20)
expected_lon = np.array(
[-64.24072434653284, -68.69662326361153, -65.92516214783112, -60.726360278290336,
-47.39851775032484, 9.500000000000018, 66.39851775032487, 79.72636027829033,
Expand Down Expand Up @@ -1524,13 +1523,13 @@ def test_get_geostationary_bbox_works_with_truncated_area_proj_coords(self, trun

def test_get_geostationary_bbox_does_not_contain_inf(self, truncated_geos_area):
"""Ensure the geostationary bbox does not contain np.inf."""
lon, lat = get_geostationary_bounding_box_in_lonlats(truncated_geos_area, 20)
lon, lat = _get_geostationary_bounding_box_in_lonlats(truncated_geos_area, 20)
assert not any(np.isinf(lon))
assert not any(np.isinf(lat))

def test_get_geostationary_bbox_returns_empty_lonlats_in_space(self, truncated_geos_area_in_space):
"""Ensure the geostationary bbox is empty when in space."""
lon, lat = get_geostationary_bounding_box_in_lonlats(truncated_geos_area_in_space, 20)
lon, lat = _get_geostationary_bounding_box_in_lonlats(truncated_geos_area_in_space, 20)

assert len(lon) == 0
assert len(lat) == 0
Expand All @@ -1547,7 +1546,7 @@ def test_get_geostationary_bbox(self):
geos_area.crs = CRS(proj_dict)
geos_area.area_extent = [-5500000., -5500000., 5500000., 5500000.]

lon, lat = get_geostationary_bounding_box_in_lonlats(geos_area, 20)
lon, lat = _get_geostationary_bounding_box_in_lonlats(geos_area, 20)
expected_lon = np.array([-78.19662326, -75.42516215, -70.22636028,
-56.89851775, 0., 56.89851775, 70.22636028,
75.42516215, 78.19662326, 79.23372832, 78.19662326,
Expand All @@ -1572,7 +1571,7 @@ def test_get_geostationary_bbox(self):
geos_area.crs = CRS(proj_dict)
geos_area.area_extent = [-5500000., -5500000., 5500000., 5500000.]

lon, lat = get_geostationary_bounding_box_in_lonlats(geos_area, 20)
lon, lat = _get_geostationary_bounding_box_in_lonlats(geos_area, 20)
np.testing.assert_allclose(lon, expected_lon + lon_0)

def test_get_geostationary_angle_extent(self):
Expand Down
5 changes: 2 additions & 3 deletions pyresample/test/test_gradient.py
Original file line number Diff line number Diff line change
Expand Up @@ -601,13 +601,12 @@ def test_check_overlap():
poly2 = Polygon(((5, 5), (6, 5), (6, 6), (5, 6)))
assert check_overlap(poly1, poly2) is False


def test_get_border_lonlats_geos():
"""Test that correct methods are called in get_border_lonlats() with geos inputs."""
from pyresample.gradient import get_border_lonlats
geo_def = AreaDefinition("", "", "",
"+proj=geos +h=1234567", 2, 2, [1, 2, 3, 4])
with mock.patch("pyresample.gradient.get_geostationary_bounding_box_in_lonlats") as get_geostationary_bounding_box:
"+proj=geos +h=1234567", 2, 2, [1, 2, 3, 4])
with mock.patch("pyresample.gradient._get_geostationary_bounding_box_in_lonlats") as get_geostationary_bounding_box:
get_geostationary_bounding_box.return_value = 1, 2
res = get_border_lonlats(geo_def)
assert res == (1, 2)
Expand Down