Refactoring: move a nested function to the module level; add tests for it

This commit is contained in:
Alexey Zakharenkov 2022-10-13 13:28:49 +03:00 committed by Alexey Zakharenkov
parent a9c32e3c12
commit e53683655d
2 changed files with 171 additions and 12 deletions

View file

@ -130,19 +130,21 @@ def is_near(p1, p2):
)
def project_on_line(p, line):
def project_on_segment(p, p1, p2):
dp = (p2[0] - p1[0], p2[1] - p1[1])
d2 = dp[0] * dp[0] + dp[1] * dp[1]
if d2 < 1e-14:
return None
# u is the position of projection of p point on line p1p2
# regarding point p1 and (p2-p1) direction vector
u = ((p[0] - p1[0]) * dp[0] + (p[1] - p1[1]) * dp[1]) / d2
if not 0 <= u <= 1:
return None
return u
def project_on_segment(p, p1, p2):
"""Given three points, return u - the position of projection of
point p onto segment p1p2 regarding point p1 and (p2-p1) direction vector
"""
dp = (p2[0] - p1[0], p2[1] - p1[1])
d2 = dp[0] * dp[0] + dp[1] * dp[1]
if d2 < 1e-14:
return None
u = ((p[0] - p1[0]) * dp[0] + (p[1] - p1[1]) * dp[1]) / d2
if not 0 <= u <= 1:
return None
return u
def project_on_line(p, line):
result = {
# In the first approximation, position on rails is the index of the
# closest vertex of line to the point p. Fractional value means that

157
tests/test_projection.py Normal file
View file

@ -0,0 +1,157 @@
import collections
import itertools
import unittest
from subway_structure import project_on_segment
class TestProjection(unittest.TestCase):
"""Test subway_structure.project_on_segment function"""
PRECISION = 10 # decimal places in assertAlmostEqual
SHIFT = 1e-6 # Small distance between projected point and segment endpoint
def _test_projection_in_bulk(self, points, segments, answers):
"""Test 'project_on_segment' function for array of points and array
of parallel segments projections on which are equal.
"""
for point, ans in zip(points, answers):
for seg in segments:
for segment, answer in zip(
(seg, seg[::-1]), # What if invert the segment?
(ans, None if ans is None else 1 - ans),
):
u = project_on_segment(point, segment[0], segment[1])
if answer is None:
self.assertIsNone(
u,
f"Project of point {point} onto segment {segment} "
f"should be None, but {u} returned",
)
else:
self.assertAlmostEqual(
u,
answer,
self.PRECISION,
f"Wrong projection of point {point} onto segment "
f"{segment}: {u} returned, {answer} expected",
)
def test_projection_on_horizontal_segments(self):
points = [
(-2, 0),
(-1 - self.SHIFT, 0),
(-1, 0),
(-1 + self.SHIFT, 0),
(-0.5, 0),
(0, 0),
(0.5, 0),
(1 - self.SHIFT, 0),
(1, 0),
(1 + self.SHIFT, 0),
(2, 0),
]
horizontal_segments = [
((-1, -1), (1, -1)),
((-1, 0), (1, 0)),
((-1, 1), (1, 1)),
]
answers = [
None,
None,
0,
self.SHIFT / 2,
0.25,
0.5,
0.75,
1 - self.SHIFT / 2,
1,
None,
None,
]
self._test_projection_in_bulk(points, horizontal_segments, answers)
def test_projection_on_vertical_segments(self):
points = [
(0, -2),
(0, -1 - self.SHIFT),
(0, -1),
(0, -1 + self.SHIFT),
(0, -0.5),
(0, 0),
(0, 0.5),
(0, 1 - self.SHIFT),
(0, 1),
(0, 1 + self.SHIFT),
(0, 2),
]
vertical_segments = [
((-1, -1), (-1, 1)),
((0, -1), (0, 1)),
((1, -1), (1, 1)),
]
answers = [
None,
None,
0,
self.SHIFT / 2,
0.25,
0.5,
0.75,
1 - self.SHIFT / 2,
1,
None,
None,
]
self._test_projection_in_bulk(points, vertical_segments, answers)
def test_projection_on_inclined_segment(self):
points = [
(-2, -2),
(-1, -1),
(-0.5, -0.5),
(0, 0),
(0.5, 0.5),
(1, 1),
(2, 2),
]
segments = [
((-2, 0), (0, 2)),
((-1, -1), (1, 1)),
((0, -2), (2, 0)),
]
answers = [None, 0, 0.25, 0.5, 0.75, 1, None]
self._test_projection_in_bulk(points, segments, answers)
def test_projection_with_different_collections(self):
"""The tested function should accept points as any consecutive
container with index operator.
"""
types = (tuple, list, collections.deque,)
point = (0, 0.5)
segment_end1 = (0, 0)
segment_end2 = (1, 0)
for p_type, s1_type, s2_type in itertools.product(types, types, types):
p = p_type(point)
s1 = s1_type(segment_end1)
s2 = s2_type(segment_end2)
project_on_segment(p, s1, s2)
def test_projection_on_degenerate_segment(self):
coords = [-1, 0, 1]
points = [(x, y) for x, y in itertools.product(coords, coords)]
segments = [
((0, 0), (0, 0)),
((0, 0), (0, 1e-8)),
]
answers = [None] * len(points)
self._test_projection_in_bulk(points, segments, answers)