Separate functions for normalizing and fitting a bbox

`normalize_bbox()` just parses incoming coordinates into a normalized dict
`fit_bbox()`  will translate any outlying coordinates to ensure they fit
the -180, -90, 180, 90 extent
This commit is contained in:
amercader 2022-08-31 13:19:14 +02:00
parent 87b723e9de
commit eebd3954a1
2 changed files with 97 additions and 23 deletions

View File

@ -24,21 +24,27 @@ def get_srid(crs):
return int(srid) return int(srid)
def validate_bbox(bbox_values): def normalize_bbox(bbox_values):
""" """
Ensures a bbox is expressed in a standard dict. Ensures a bbox is expressed in a standard dict
bbox_values may be: bbox_values may be:
a string: "-4.96,55.70,-3.78,56.43" a string: "-4.96,55.70,-3.78,56.43"
or a list [-4.96, 55.70, -3.78, 56.43] or a list [-4.96, 55.70, -3.78, 56.43]
or a list of strings ["-4.96", "55.70", "-3.78", "56.43"] or a list of strings ["-4.96", "55.70", "-3.78", "56.43"]
and returns a dict:
{'minx': -4.96,
'miny': 55.70,
'maxx': -3.78,
'maxy': 56.43}
Any problems and it returns None. ordered as MinX, MinY, MaxX, MaxY.
Returns a dict with the keys:
{
"minx": -4.96,
"miny": 55.70,
"maxx": -3.78,
"maxy": 56.43
}
If there are any problems parsing the input it returns None.
""" """
if isinstance(bbox_values, six.string_types): if isinstance(bbox_values, six.string_types):
@ -56,4 +62,48 @@ def validate_bbox(bbox_values):
except ValueError: except ValueError:
return None return None
bbox = fit_bbox(bbox)
return bbox return bbox
def fit_bbox(bbox_dict):
"""
Ensures that all coordinates in a bounding box
fall within -180, -90, 180, 90 degrees
Accepts a dict with the following keys:
{
"minx": -4.96,
"miny": 55.70,
"maxx": -3.78,
"maxy": 56.43
}
"""
def _adjust_longitude(value):
if value < -180 or value > 180:
value = value % 360
if value < -180:
value = 360 + value
elif value > 180:
value = -360 + value
return value
def _adjust_latitude(value):
if value < -90 or value > 90:
value = value % 180
if value < -90:
value = 180 + value
elif value > 90:
value = -180 + value
return value
return {
"minx": _adjust_longitude(bbox_dict["minx"]),
"maxx": _adjust_longitude(bbox_dict["maxx"]),
"miny": _adjust_latitude(bbox_dict["miny"]),
"maxy": _adjust_latitude(bbox_dict["maxy"]),
}

View File

@ -1,21 +1,45 @@
from ckanext.spatial.lib import validate_bbox from ckanext.spatial.lib import normalize_bbox, fit_bbox
class TestValidateBbox(object):
bbox_dict = {"minx": -4.96, "miny": 55.70, "maxx": -3.78, "maxy": 56.43} bbox_dict = {"minx": -4.96, "miny": 55.70, "maxx": -3.78, "maxy": 56.43}
def test_string(self):
res = validate_bbox("-4.96,55.70,-3.78,56.43")
assert res == self.bbox_dict
def test_list(self): def test_string():
res = validate_bbox([-4.96, 55.70, -3.78, 56.43]) res = normalize_bbox("-4.96,55.70,-3.78,56.43")
assert res == self.bbox_dict assert res == bbox_dict
def test_bad(self):
res = validate_bbox([-4.96, 55.70, -3.78]) def test_list():
res = normalize_bbox([-4.96, 55.70, -3.78, 56.43])
assert res == bbox_dict
def test_list_of_strings():
res = normalize_bbox(["-4.96", "55.70", "-3.78", "56.43"])
assert res == bbox_dict
def test_bad():
res = normalize_bbox([-4.96, 55.70, -3.78])
assert res is None assert res is None
def test_bad_2(self):
res = validate_bbox("random") def test_bad_2():
res = normalize_bbox("random")
assert res is None assert res is None
def test_fit_within_bounds():
assert fit_bbox(bbox_dict) == bbox_dict
def test_fit_out_of_bounds():
bbox_dict = {"minx": -185, "miny": -95, "maxx": 195, "maxy": 95}
assert fit_bbox(bbox_dict) == {
"minx": 175,
"miny": 85,
"maxx": -165,
"maxy": -85,
}