Skip to content

Commit 143a7cf

Browse files
authored
Merge branch 'geo-master' into percent_cover
2 parents 644ddc3 + a0eef4e commit 143a7cf

File tree

9 files changed

+121
-15
lines changed

9 files changed

+121
-15
lines changed

CHANGELOG.txt

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
1-
0.12dev
1+
0.12.0
22
- zone_func argument to apply a function to the masked array before computing stats
3+
- support shapely 1.6 exceptions
34

45
0.11
56
- change `band_num` parameter to `band`

src/rasterstats/_version.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
__version__ = "0.12a1"
1+
__version__ = "0.12.0"

src/rasterstats/io.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,10 @@
1010
from rasterio.transform import guard_transform
1111
from affine import Affine
1212
import numpy as np
13-
from shapely.geos import ReadingError
13+
try:
14+
from shapely.errors import ReadingError
15+
except:
16+
from shapely.geos import ReadingError
1417
from shapely import wkt, wkb
1518
from collections import Iterable, Mapping
1619

src/rasterstats/main.py

+12-7
Original file line numberDiff line numberDiff line change
@@ -161,8 +161,8 @@ def gen_zonal_stats(
161161
warnings.warn("Use `geojson_out` to preserve feature properties",
162162
DeprecationWarning)
163163

164-
bn = kwargs.get('band_num')
165-
if bn:
164+
band_num = kwargs.get('band_num')
165+
if band_num:
166166
warnings.warn("Use `band` to specify band number", DeprecationWarning)
167167
band = band_num
168168

@@ -238,8 +238,9 @@ def gen_zonal_stats(
238238
isnodata = (fsrc.array == fsrc.nodata)
239239

240240
# add nan mask (if necessary)
241-
if np.issubdtype(fsrc.array.dtype, float) and \
242-
np.isnan(fsrc.array.min()):
241+
has_nan = (np.issubdtype(fsrc.array.dtype, float)
242+
and np.isnan(fsrc.array.min()))
243+
if has_nan:
243244
isnodata = (isnodata | np.isnan(fsrc.array))
244245

245246
# Mask the source data array
@@ -325,9 +326,13 @@ def gen_zonal_stats(
325326
pctarr = masked.compressed()
326327
feature_stats[pctile] = np.percentile(pctarr, q)
327328

328-
if 'nodata' in stats:
329-
featmasked = np.ma.MaskedArray(fsrc.array, mask=np.logical_not(rv_array))
330-
feature_stats['nodata'] = float((featmasked == fsrc.nodata).sum())
329+
if 'nodata' in stats or 'nan' in stats:
330+
featmasked = np.ma.MaskedArray(fsrc.array, mask=(~rv_array))
331+
332+
if 'nodata' in stats:
333+
feature_stats['nodata'] = float((featmasked == fsrc.nodata).sum())
334+
if 'nan' in stats:
335+
feature_stats['nan'] = float(np.isnan(featmasked).sum()) if has_nan else 0
331336

332337
if add_stats is not None:
333338
for stat_name, stat_func in add_stats.items():

src/rasterstats/point.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ def bilinear(arr, x, y):
5353
if hasattr(arr, 'count') and arr.count() != 4:
5454
# a masked array with at least one nodata
5555
# fall back to nearest neighbor
56-
val = arr[round(1 - y), round(x)]
56+
val = arr[int(round(1 - y)), int(round(x))]
5757
if val is masked:
5858
return None
5959
else:
@@ -172,7 +172,7 @@ def gen_point_query(
172172
for x, y in geom_xys(geom):
173173
if interpolate == 'nearest':
174174
r, c = rast.index(x, y)
175-
window = ((r, r+1), (c, c+1))
175+
window = ((int(r), int(r+1)), (int(c), int(c+1)))
176176
src_array = rast.read(window=window, masked=True).array
177177
val = src_array[0, 0]
178178
if val is masked:

src/rasterstats/utils.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111

1212
DEFAULT_STATS = ['count', 'min', 'max', 'mean']
1313
VALID_STATS = DEFAULT_STATS + \
14-
['sum', 'std', 'median', 'majority', 'minority', 'unique', 'range', 'nodata']
14+
['sum', 'std', 'median', 'majority', 'minority', 'unique', 'range', 'nodata', 'nan']
1515
# also percentile_{q} but that is handled as special case
1616

1717

tests/test_cli.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -34,14 +34,15 @@ def test_cli_feature_stdin():
3434
warnings.simplefilter('ignore')
3535
result = runner.invoke(zonalstats,
3636
['--raster', raster,
37-
'--stats', 'mean',
37+
'--stats', 'all',
3838
'--prefix', 'test_'],
3939
input=open(vector, 'r').read())
4040
assert result.exit_code == 0
4141
outdata = json.loads(result.output)
4242
assert len(outdata['features']) == 1
4343
feature = outdata['features'][0]
4444
assert 'test_mean' in feature['properties']
45+
assert 'test_std' in feature['properties']
4546

4647

4748
def test_cli_features_sequence():

tests/test_io.py

+5
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,11 @@ def test_Raster():
268268

269269
r2 = Raster(arr, affine, nodata, band=1).read(bounds)
270270

271+
with pytest.raises(ValueError):
272+
r3 = Raster(arr, affine, nodata, band=1).read()
273+
with pytest.raises(ValueError):
274+
r4 = Raster(arr, affine, nodata, band=1).read(bounds=1, window=1)
275+
271276
# If the abstraction is correct, the arrays are equal
272277
assert np.array_equal(r1.array, r2.array)
273278

tests/test_zonal.py

+92-1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,16 @@ def test_main():
2929
assert round(stats[0]['mean'], 2) == 14.66
3030

3131

32+
# remove after band_num alias is removed
33+
def test_band_alias():
34+
polygons = os.path.join(DATA, 'polygons.shp')
35+
stats_a = zonal_stats(polygons, raster)
36+
stats_b = zonal_stats(polygons, raster, band=1)
37+
with pytest.deprecated_call():
38+
stats_c = zonal_stats(polygons, raster, band_num=1)
39+
assert stats_a[0]['count'] == stats_b[0]['count'] == stats_c[0]['count']
40+
41+
3242
def test_zonal_global_extent():
3343
polygons = os.path.join(DATA, 'polygons.shp')
3444
stats = zonal_stats(polygons, raster)
@@ -376,6 +386,27 @@ def test_some_nodata():
376386
assert stats[1]['nodata'] == 19
377387
assert stats[1]['count'] == 31
378388

389+
390+
# update this if nan end up being incorporated into nodata
391+
def test_nan_nodata():
392+
polygon = Polygon([[0, 0], [2, 0], [2, 2], [0, 2]])
393+
arr = np.array([
394+
[np.nan, 12.25],
395+
[-999, 12.75]
396+
])
397+
affine = Affine(1, 0, 0,
398+
0, -1, 2)
399+
400+
stats = zonal_stats(polygon, arr, affine=affine, nodata=-999,
401+
stats='nodata count sum mean min max')
402+
403+
assert stats[0]['nodata'] == 1
404+
assert stats[0]['count'] == 2
405+
assert stats[0]['mean'] == 12.5
406+
assert stats[0]['min'] == 12.25
407+
assert stats[0]['max'] == 12.75
408+
409+
379410
def test_some_nodata_ndarray():
380411
polygons = os.path.join(DATA, 'polygons.shp')
381412
raster = os.path.join(DATA, 'slope_nodata.tif')
@@ -427,6 +458,67 @@ def test_geojson_out():
427458
assert 'count' in feature['properties'] # from zonal stats
428459

429460

461+
# do not think this is actually testing the line i wanted it to
462+
# since the read_features func for this data type is generating
463+
# the properties field
464+
def test_geojson_out_with_no_properties():
465+
polygon = Polygon([[0, 0], [0, 0,5], [1, 1.5], [1.5, 2], [2, 2], [2, 0]])
466+
arr = np.array([
467+
[100, 1],
468+
[100, 1]
469+
])
470+
affine = Affine(1, 0, 0,
471+
0, -1, 2)
472+
473+
stats = zonal_stats(polygon, arr, affine=affine, geojson_out=True)
474+
assert 'properties' in stats[0]
475+
for key in ['count', 'min', 'max', 'mean']:
476+
assert key in stats[0]['properties']
477+
478+
assert stats[0]['properties']['mean'] == 34
479+
480+
481+
# remove when copy_properties alias is removed
482+
def test_copy_properties_warn():
483+
polygons = os.path.join(DATA, 'polygons.shp')
484+
# run once to trigger any other unrelated deprecation warnings
485+
# so the test does not catch them instead
486+
stats_a = zonal_stats(polygons, raster)
487+
with pytest.deprecated_call():
488+
stats_b = zonal_stats(polygons, raster, copy_properties=True)
489+
assert stats_a == stats_b
490+
491+
492+
def test_nan_counts():
493+
from affine import Affine
494+
transform = Affine(1, 0, 1, 0, -1, 3)
495+
496+
data = np.array([
497+
[np.nan, np.nan, np.nan],
498+
[0, 0, 0],
499+
[1, 4, 5]
500+
])
501+
502+
# geom extends an additional row to left
503+
geom = 'POLYGON ((1 0, 4 0, 4 3, 1 3, 1 0))'
504+
505+
# nan stat is requested
506+
stats = zonal_stats(geom, data, affine=transform, nodata=0.0, stats="*")
507+
508+
for res in stats:
509+
assert res['count'] == 3 # 3 pixels of valid data
510+
assert res['nodata'] == 3 # 3 pixels of nodata
511+
assert res['nan'] == 3 # 3 pixels of nans
512+
513+
# nan are ignored if nan stat is not requested
514+
stats = zonal_stats(geom, data, affine=transform, nodata=0.0, stats="count nodata")
515+
516+
for res in stats:
517+
assert res['count'] == 3 # 3 pixels of valid data
518+
assert res['nodata'] == 3 # 3 pixels of nodata
519+
assert 'nan' not in res
520+
521+
430522
def test_percent_cover_zonal_stats():
431523
polygon = Polygon([[0, 0], [0, 0,5], [1, 1.5], [1.5, 2], [2, 2], [2, 0]])
432524
arr = np.array([
@@ -480,4 +572,3 @@ def test_geodataframe_zonal():
480572
expected = zonal_stats(polygons, raster)
481573
assert zonal_stats(df, raster) == expected
482574

483-

0 commit comments

Comments
 (0)