Skip to content

Commit 36c8762

Browse files
committed
update code
1 parent 01857b9 commit 36c8762

File tree

2 files changed

+124
-55
lines changed

2 files changed

+124
-55
lines changed

GeoAnalyze/stream.py

+62-55
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,6 @@ def is_flw_path_us_to_ds(
5555

5656
return output
5757

58-
# pytest pending
5958
def flw_path_reverse(
6059
self,
6160
input_file: str,
@@ -270,7 +269,6 @@ def point_junctions(
270269

271270
return output_gdf
272271

273-
# pytest pending
274272
def point_segment_subbasin_drainage(
275273
self,
276274
input_file: str,
@@ -391,8 +389,7 @@ def point_main_outlets(
391389

392390
return outlet_gdf
393391

394-
# pytest pending
395-
def create_box_touching_selected_segment(
392+
def box_touch_selected_segment(
396393
self,
397394
input_file: str,
398395
column_name: str,
@@ -402,8 +399,8 @@ def create_box_touching_selected_segment(
402399
) -> geopandas.GeoDataFrame:
403400

404401
'''
405-
Creates a square box polygon that touches a specified stream segment
406-
in the input flow path at a randomly chosen point along the segment.
402+
Creates a square box polygon that touches a specified segment
403+
in the stream path at a randomly chosen point along the segment.
407404
408405
Parameters
409406
----------
@@ -429,17 +426,19 @@ def create_box_touching_selected_segment(
429426
touches the specified stream segment at a random point.
430427
'''
431428

429+
# check validity of output file path
430+
check_file = Core().is_valid_ogr_driver(output_file)
431+
if check_file is True:
432+
pass
433+
else:
434+
raise Exception('Could not retrieve driver from the file path.')
435+
432436
# input line segment
433437
gdf = geopandas.read_file(input_file)
434438
line = gdf[gdf[column_name].isin([column_value])].geometry.iloc[0]
435439

436440
# line coords
437-
if isinstance(line, shapely.LineString):
438-
line_coords = line.coords[:]
439-
else:
440-
line_coords = []
441-
for line_split in line.geoms:
442-
line_coords.extend(line_split.coords[:])
441+
line_coords = line.coords[:] if isinstance(line, shapely.LineString) else [c for ls in line.geoms for c in ls.coords[:]]
443442

444443
while True:
445444
# choose points
@@ -464,8 +463,6 @@ def create_box_touching_selected_segment(
464463
check_touch = line.touches(rotate_box) and not line.crosses(rotate_box)
465464
if check_touch is True:
466465
break
467-
else:
468-
pass
469466

470467
# saving box geodataframe
471468
box_gdf = geopandas.GeoDataFrame(
@@ -476,20 +473,19 @@ def create_box_touching_selected_segment(
476473

477474
return box_gdf
478475

479-
# pytest pending
480-
def create_box_crosses_segment_at_endpoint(
476+
def box_touch_selected_segment_at_endpoint(
481477
self,
482478
input_file: str,
483479
column_name: str,
484480
column_value: typing.Any,
485481
box_length: float,
486482
output_file: str,
487-
downstream_point: bool = True
483+
upstream_point: bool = True
488484
) -> geopandas.GeoDataFrame:
489485

490486
'''
491-
Creates a square box polygon that crosses a specified stream segment
492-
in the input flow path and passes through an endpoint of the segment.
487+
Creates a square box polygon that touches an endpoint
488+
of a specified segment in the input stream path.
493489
494490
Parameters
495491
----------
@@ -508,43 +504,50 @@ def create_box_crosses_segment_at_endpoint(
508504
output_file : str
509505
Path to save the output box polygon shapefile.
510506
511-
downstream_point : bool, optional
512-
If True, the box is positioned to pass through the downstream endpoint
513-
of the segment; if False, it passes through the upstream endpoint. Default is True.
507+
upstream_point : bool, optional
508+
If True, the box is positioned to pass through the upstream endpoint
509+
of the segment; if False, it passes through the downstream endpoint. Default is True.
514510
515511
Returns
516512
-------
517513
GeoDataFrame
518-
A GeoDataFrame containing the box polygon, which crosses the
519-
specified stream segment and passes through an endpoint of the segment.
514+
A GeoDataFrame containing the box polygon, which touches an endpoint of
515+
the specified segment in the input stream path.
520516
'''
521517

518+
# check validity of output file path
519+
check_file = Core().is_valid_ogr_driver(output_file)
520+
if check_file is True:
521+
pass
522+
else:
523+
raise Exception('Could not retrieve driver from the file path.')
524+
522525
# input line segement
523526
gdf = geopandas.read_file(input_file)
524527
line = gdf[gdf[column_name].isin([column_value])].geometry.iloc[0]
525528

526529
# get point
527-
point_coords = line.coords[-1] if downstream_point is True else line.coords[0]
530+
point_coords = line.coords[0] if upstream_point is True else line.coords[-1]
528531
point = shapely.Point(*point_coords)
529532

530533
# create box
531534
box = shapely.box(
532535
xmin=point.x,
533536
ymin=point.y,
534537
xmax=point.x + box_length,
535-
ymax=point.y + box_length,
538+
ymax=point.y + box_length
536539
)
537540

538-
# check whether the box crosses the line; otherwise rotate
541+
# check whether the box touches the line; otherwise rotate
539542
while True:
540-
if line.crosses(box):
543+
box = shapely.affinity.rotate(
544+
geom=box,
545+
angle=random.randint(0, 360),
546+
origin=point
547+
)
548+
check_touch = line.touches(box) and not line.crosses(box)
549+
if check_touch:
541550
break
542-
else:
543-
box = shapely.affinity.rotate(
544-
geom=box,
545-
angle=random.randint(0, 360),
546-
origin=point
547-
)
548551

549552
# saving box geodataframe
550553
box_gdf = geopandas.GeoDataFrame(
@@ -555,20 +558,19 @@ def create_box_crosses_segment_at_endpoint(
555558

556559
return box_gdf
557560

558-
# pytest pending
559-
def create_box_touch_segment_at_endpoint(
561+
def box_cross_selected_segment_at_endpoint(
560562
self,
561563
input_file: str,
562564
column_name: str,
563565
column_value: typing.Any,
564566
box_length: float,
565567
output_file: str,
566-
upstream_point: bool = True
568+
downstream_point: bool = True
567569
) -> geopandas.GeoDataFrame:
568570

569571
'''
570-
Creates a square box polygon that touches an endpoint
571-
of a specified segment in the input stream path.
572+
Creates a square box polygon that crosses a specified segment
573+
in the stream path and passes through an endpoint of the segment.
572574
573575
Parameters
574576
----------
@@ -587,44 +589,49 @@ def create_box_touch_segment_at_endpoint(
587589
output_file : str
588590
Path to save the output box polygon shapefile.
589591
590-
upstream_point : bool, optional
591-
If True, the box is positioned to pass through the upstream endpoint
592-
of the segment; if False, it passes through the downstream endpoint. Default is True.
592+
downstream_point : bool, optional
593+
If True, the box is positioned to pass through the downstream endpoint
594+
of the segment; if False, it passes through the upstream endpoint. Default is True.
593595
594596
Returns
595597
-------
596598
GeoDataFrame
597-
A GeoDataFrame containing the box polygon, which touches an endpoint of
598-
the specified segment in the input stream path.
599+
A GeoDataFrame containing the box polygon, which crosses the
600+
specified stream segment and passes through an endpoint of the segment.
599601
'''
600602

603+
# check validity of output file path
604+
check_file = Core().is_valid_ogr_driver(output_file)
605+
if check_file is True:
606+
pass
607+
else:
608+
raise Exception('Could not retrieve driver from the file path.')
609+
601610
# input line segement
602611
gdf = geopandas.read_file(input_file)
603612
line = gdf[gdf[column_name].isin([column_value])].geometry.iloc[0]
604613

605614
# get point
606-
point_coords = line.coords[0] if upstream_point is True else line.coords[-1]
615+
point_coords = line.coords[-1] if downstream_point is True else line.coords[0]
607616
point = shapely.Point(*point_coords)
608617

609618
# create box
610619
box = shapely.box(
611620
xmin=point.x,
612621
ymin=point.y,
613622
xmax=point.x + box_length,
614-
ymax=point.y + box_length,
623+
ymax=point.y + box_length
615624
)
616625

617-
# check whether the box touches the line; otherwise rotate
626+
# check whether the box crosses the line; otherwise rotate
618627
while True:
619-
check_touch = line.touches(box) and not line.crosses(box)
620-
if check_touch:
628+
box = shapely.affinity.rotate(
629+
geom=box,
630+
angle=random.randint(0, 360),
631+
origin=point
632+
)
633+
if line.crosses(box):
621634
break
622-
else:
623-
box = shapely.affinity.rotate(
624-
geom=box,
625-
angle=random.randint(0, 360),
626-
origin=point
627-
)
628635

629636
# saving box geodataframe
630637
box_gdf = geopandas.GeoDataFrame(

tests/test_stream.py

+62
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,38 @@ def test_functions(
9191
output_file=os.path.join(tmp_dir, 'main_outlet_points.shp')
9292
)
9393
assert stream_gdf['geometry'].iloc[-1].coords[-1] == outlet_gdf['geometry'].iloc[-1].coords[0]
94+
# pass test of box touching the selected segment in a stream path
95+
selected_line = stream_gdf[stream_gdf['flw_id'] == 3]['geometry'].iloc[0]
96+
box_gdf = stream.box_touch_selected_segment(
97+
input_file=stream_file,
98+
column_name='flw_id',
99+
column_value=3,
100+
box_length=500,
101+
output_file=os.path.join(tmp_dir, 'box.shp')
102+
)
103+
polygon = box_gdf.geometry.iloc[0]
104+
assert selected_line.touches(polygon)
105+
# pass test of box touching the selected segment at endpoint in a stream path
106+
box_gdf = stream.box_touch_selected_segment_at_endpoint(
107+
input_file=stream_file,
108+
column_name='flw_id',
109+
column_value=3,
110+
box_length=500,
111+
output_file=os.path.join(tmp_dir, 'box.shp')
112+
)
113+
polygon = box_gdf.geometry.iloc[0]
114+
assert selected_line.touches(polygon)
115+
# pass test of box crossing the selected segment at endpoint in a stream path
116+
box_gdf = stream.box_cross_selected_segment_at_endpoint(
117+
input_file=stream_file,
118+
column_name='flw_id',
119+
column_value=3,
120+
box_length=500,
121+
output_file=os.path.join(tmp_dir, 'box.shp')
122+
)
123+
polygon = box_gdf.geometry.iloc[0]
124+
intersection = selected_line.intersection(polygon)
125+
assert isinstance(intersection, shapely.MultiLineString) or len(intersection.coords[:]) > 1
94126

95127

96128
def test_error_linestring_geometry(
@@ -190,3 +222,33 @@ def test_error_shapefile_driver(
190222
output_file='main_outlet_points.sh'
191223
)
192224
assert exc_info.value.args[0] == message['error_driver']
225+
# box touching the selected segment in a stream path
226+
with pytest.raises(Exception) as exc_info:
227+
stream.box_touch_selected_segment(
228+
input_file='stream.shp',
229+
column_name='flw_id',
230+
column_value=3,
231+
box_length=500,
232+
output_file='box.sh'
233+
)
234+
assert exc_info.value.args[0] == message['error_driver']
235+
# box touching the selected segment at endpoint in a stream path
236+
with pytest.raises(Exception) as exc_info:
237+
stream.box_touch_selected_segment_at_endpoint(
238+
input_file='stream.shp',
239+
column_name='flw_id',
240+
column_value=3,
241+
box_length=500,
242+
output_file='box.sh'
243+
)
244+
assert exc_info.value.args[0] == message['error_driver']
245+
# box crossing the selected segment at endpoint in a stream path
246+
with pytest.raises(Exception) as exc_info:
247+
stream.box_cross_selected_segment_at_endpoint(
248+
input_file='stream.shp',
249+
column_name='flw_id',
250+
column_value=3,
251+
box_length=500,
252+
output_file='box.sh'
253+
)
254+
assert exc_info.value.args[0] == message['error_driver']

0 commit comments

Comments
 (0)