EasyDetect / pipeline /mmocr /tests /test_utils /test_polygon_utils.py
sunnychenxiwang's picture
update all
24c4def
raw
history blame
16.3 kB
# Copyright (c) OpenMMLab. All rights reserved.
import unittest
from itertools import chain, permutations
import numpy as np
import torch
from shapely.geometry import MultiPolygon, Polygon
from mmocr.utils import (boundary_iou, crop_polygon, offset_polygon, poly2bbox,
poly2shapely, poly_intersection, poly_iou,
poly_make_valid, poly_union, polys2shapely,
rescale_polygon, rescale_polygons, shapely2poly,
sort_points, sort_vertex, sort_vertex8)
class TestPolygonUtils(unittest.TestCase):
def test_crop_polygon(self):
# polygon cross box
polygon = np.array([20., -10., 40., 10., 10., 40., -10., 20.])
crop_box = np.array([0., 0., 60., 60.])
target_poly_cropped = np.array(
[10, 40, 0, 30, 0, 10, 10, 0, 30, 0, 40, 10])
poly_cropped = crop_polygon(polygon, crop_box)
self.assertTrue(
poly2shapely(poly_cropped).equals(
poly2shapely(target_poly_cropped)))
# polygon inside box
polygon = np.array([0., 0., 30., 0., 30., 30., 0., 30.])
crop_box = np.array([0., 0., 60., 60.])
target_poly_cropped = polygon
poly_cropped = crop_polygon(polygon, crop_box)
self.assertTrue(
poly2shapely(poly_cropped).equals(
poly2shapely(target_poly_cropped)))
# polygon outside box
polygon = np.array([0., 0., 30., 0., 30., 30., 0., 30.])
crop_box = np.array([80., 80., 90., 90.])
poly_cropped = crop_polygon(polygon, crop_box)
self.assertEqual(poly_cropped, None)
# polygon and box are overlapped at a point
polygon = np.array([0., 0., 10., 0., 10., 10., 0., 10.])
crop_box = np.array([10., 10., 20., 20.])
poly_cropped = crop_polygon(polygon, crop_box)
self.assertEqual(poly_cropped, None)
def test_rescale_polygon(self):
scale_factor = (0.3, 0.4)
with self.assertRaises(AssertionError):
polygons = [0, 0, 1, 0, 1, 1, 0]
rescale_polygon(polygons, scale_factor)
polygons = [0, 0, 1, 0, 1, 1, 0, 1]
self.assertTrue(
np.allclose(
rescale_polygon(polygons, scale_factor, mode='div'),
np.array([0, 0, 1 / 0.3, 0, 1 / 0.3, 1 / 0.4, 0, 1 / 0.4])))
self.assertTrue(
np.allclose(
rescale_polygon(polygons, scale_factor, mode='mul'),
np.array([0, 0, 0.3, 0, 0.3, 0.4, 0, 0.4])))
def test_rescale_polygons(self):
polygons = [
np.array([0, 0, 1, 0, 1, 1, 0, 1]),
np.array([1, 1, 2, 1, 2, 2, 1, 2])
]
scale_factor = (0.5, 0.5)
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='div'), [
np.array([0, 0, 2, 0, 2, 2, 0, 2]),
np.array([2, 2, 4, 2, 4, 4, 2, 4])
]))
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='mul'), [
np.array([0, 0, 0.5, 0, 0.5, 0.5, 0, 0.5]),
np.array([0.5, 0.5, 1, 0.5, 1, 1, 0.5, 1])
]))
polygons = np.array([[0, 0, 1, 0, 1, 1, 0, 1],
[1, 1, 2, 1, 2, 2, 1, 2]])
scale_factor = (0.5, 0.5)
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='div'),
np.array([[0, 0, 2, 0, 2, 2, 0, 2], [2, 2, 4, 2, 4, 4, 2,
4]])))
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='mul'),
np.array([[0, 0, 0.5, 0, 0.5, 0.5, 0, 0.5],
[0.5, 0.5, 1, 0.5, 1, 1, 0.5, 1]])))
polygons = [torch.Tensor([0, 0, 1, 0, 1, 1, 0, 1])]
scale_factor = (0.3, 0.4)
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='div'),
[np.array([0, 0, 1 / 0.3, 0, 1 / 0.3, 1 / 0.4, 0, 1 / 0.4])]))
self.assertTrue(
np.allclose(
rescale_polygons(polygons, scale_factor, mode='mul'),
[np.array([0, 0, 0.3, 0, 0.3, 0.4, 0, 0.4])]))
def test_poly2bbox(self):
# test np.array
polygon = np.array([0, 0, 1, 0, 1, 1, 0, 1])
self.assertTrue(np.all(poly2bbox(polygon) == np.array([0, 0, 1, 1])))
# test list
polygon = [0, 0, 1, 0, 1, 1, 0, 1]
self.assertTrue(np.all(poly2bbox(polygon) == np.array([0, 0, 1, 1])))
# test tensor
polygon = torch.Tensor([0, 0, 1, 0, 1, 1, 0, 1])
self.assertTrue(np.all(poly2bbox(polygon) == np.array([0, 0, 1, 1])))
def test_poly2shapely(self):
polygon = Polygon([[0, 0], [1, 0], [1, 1], [0, 1]])
# test np.array
poly = np.array([0, 0, 1, 0, 1, 1, 0, 1])
self.assertEqual(poly2shapely(poly), polygon)
# test list
poly = [0, 0, 1, 0, 1, 1, 0, 1]
self.assertEqual(poly2shapely(poly), polygon)
# test tensor
poly = torch.Tensor([0, 0, 1, 0, 1, 1, 0, 1])
self.assertEqual(poly2shapely(poly), polygon)
# test invalid
poly = [0, 0, 1]
with self.assertRaises(AssertionError):
poly2shapely(poly)
poly = [0, 0, 1, 0, 1, 1, 0, 1, 1]
with self.assertRaises(AssertionError):
poly2shapely(poly)
def test_polys2shapely(self):
polygons = [
Polygon([[0, 0], [1, 0], [1, 1], [0, 1]]),
Polygon([[1, 0], [1, 1], [0, 1], [0, 0]])
]
# test np.array
polys = np.array([[0, 0, 1, 0, 1, 1, 0, 1], [1, 0, 1, 1, 0, 1, 0, 0]])
self.assertEqual(polys2shapely(polys), polygons)
# test list
polys = [[0, 0, 1, 0, 1, 1, 0, 1], [1, 0, 1, 1, 0, 1, 0, 0]]
self.assertEqual(polys2shapely(polys), polygons)
# test tensor
polys = torch.Tensor([[0, 0, 1, 0, 1, 1, 0, 1],
[1, 0, 1, 1, 0, 1, 0, 0]])
self.assertEqual(polys2shapely(polys), polygons)
# test invalid
polys = [0, 0, 1]
with self.assertRaises(AssertionError):
polys2shapely(polys)
polys = [0, 0, 1, 0, 1, 1, 0, 1, 1]
with self.assertRaises(AssertionError):
polys2shapely(polys)
def test_shapely2poly(self):
polygon = Polygon([[0., 0.], [1., 0.], [1., 1.], [0., 1.]])
poly = np.array([0., 0., 1., 0., 1., 1., 0., 1., 0., 0.])
self.assertTrue(poly2shapely(poly).equals(polygon))
self.assertTrue(isinstance(shapely2poly(polygon), np.ndarray))
def test_poly_make_valid(self):
poly = Polygon([[0, 0], [1, 1], [1, 0], [0, 1]])
self.assertFalse(poly.is_valid)
poly = poly_make_valid(poly)
self.assertTrue(poly.is_valid)
# invalid input
with self.assertRaises(AssertionError):
poly_make_valid([0, 0, 1, 1, 1, 0, 0, 1])
poly = Polygon([[337, 441], [326, 386], [334, 397], [342, 412],
[296, 382], [317, 366], [324, 427], [315, 413],
[308, 400], [349, 419], [337, 441]])
self.assertFalse(poly.is_valid)
poly = poly_make_valid(poly)
self.assertTrue(poly.is_valid)
def test_poly_intersection(self):
# test unsupported type
with self.assertRaises(AssertionError):
poly_intersection(0, 1)
# test non-overlapping polygons
points = [0, 0, 0, 1, 1, 1, 1, 0]
points1 = [10, 20, 30, 40, 50, 60, 70, 80]
points2 = [0, 0, 0, 0, 0, 0, 0, 0] # Invalid polygon
points3 = [0, 0, 0, 1, 1, 0, 1, 1] # Self-intersected polygon
points4 = [0.5, 0, 1.5, 0, 1.5, 1, 0.5, 1]
poly = poly2shapely(points)
poly1 = poly2shapely(points1)
poly2 = poly2shapely(points2)
poly3 = poly2shapely(points3)
poly4 = poly2shapely(points4)
area_inters = poly_intersection(poly, poly1)
self.assertEqual(area_inters, 0.)
# test overlapping polygons
area_inters = poly_intersection(poly, poly)
self.assertEqual(area_inters, 1)
area_inters = poly_intersection(poly, poly4)
self.assertEqual(area_inters, 0.5)
# test invalid polygons
self.assertEqual(poly_intersection(poly2, poly2), 0)
self.assertEqual(poly_intersection(poly3, poly3, invalid_ret=1), 1)
self.assertEqual(
poly_intersection(poly3, poly3, invalid_ret=None), 0.25)
# test poly return
_, poly = poly_intersection(poly, poly4, return_poly=True)
self.assertTrue(isinstance(poly, Polygon))
_, poly = poly_intersection(
poly3, poly3, invalid_ret=None, return_poly=True)
self.assertTrue(isinstance(poly, Polygon))
_, poly = poly_intersection(
poly2, poly3, invalid_ret=1, return_poly=True)
self.assertTrue(poly is None)
def test_poly_union(self):
# test unsupported type
with self.assertRaises(AssertionError):
poly_union(0, 1)
# test non-overlapping polygons
points = [0, 0, 0, 1, 1, 1, 1, 0]
points1 = [2, 2, 2, 3, 3, 3, 3, 2]
points2 = [0, 0, 0, 0, 0, 0, 0, 0] # Invalid polygon
points3 = [0, 0, 0, 1, 1, 0, 1, 1] # Self-intersected polygon
points4 = [0.5, 0.5, 1, 0, 1, 1, 0.5, 0.5]
poly = poly2shapely(points)
poly1 = poly2shapely(points1)
poly2 = poly2shapely(points2)
poly3 = poly2shapely(points3)
poly4 = poly2shapely(points4)
assert poly_union(poly, poly1) == 2
# test overlapping polygons
assert poly_union(poly, poly) == 1
# test invalid polygons
self.assertEqual(poly_union(poly2, poly2), 0)
self.assertEqual(poly_union(poly3, poly3, invalid_ret=1), 1)
# The return value depends on the implementation of the package
self.assertEqual(poly_union(poly3, poly3, invalid_ret=None), 0.25)
self.assertEqual(poly_union(poly2, poly3), 0.25)
self.assertEqual(poly_union(poly3, poly4), 0.5)
# test poly return
_, poly = poly_union(poly, poly1, return_poly=True)
self.assertTrue(isinstance(poly, MultiPolygon))
_, poly = poly_union(poly3, poly3, return_poly=True)
self.assertTrue(isinstance(poly, Polygon))
_, poly = poly_union(poly2, poly3, invalid_ret=0, return_poly=True)
self.assertTrue(poly is None)
def test_poly_iou(self):
# test unsupported type
with self.assertRaises(AssertionError):
poly_iou([1], [2])
points = [0, 0, 0, 1, 1, 1, 1, 0]
points1 = [10, 20, 30, 40, 50, 60, 70, 80]
points2 = [0, 0, 0, 0, 0, 0, 0, 0] # Invalid polygon
points3 = [0, 0, 0, 1, 1, 0, 1, 1] # Self-intersected polygon
poly = poly2shapely(points)
poly1 = poly2shapely(points1)
poly2 = poly2shapely(points2)
poly3 = poly2shapely(points3)
self.assertEqual(poly_iou(poly, poly1), 0)
# test overlapping polygons
self.assertEqual(poly_iou(poly, poly), 1)
# test invalid polygons
self.assertEqual(poly_iou(poly2, poly2), 0)
self.assertEqual(poly_iou(poly3, poly3, zero_division=1), 1)
self.assertEqual(poly_iou(poly2, poly3), 0)
def test_offset_polygon(self):
# usual case
polygons = np.array([0, 0, 0, 1, 1, 1, 1, 0], dtype=np.float32)
expanded_polygon = offset_polygon(polygons, 1)
self.assertTrue(
poly2shapely(expanded_polygon).equals(
poly2shapely(
np.array(
[2, 0, 2, 1, 1, 2, 0, 2, -1, 1, -1, 0, 0, -1, 1,
-1]))))
# Overshrunk polygon doesn't exist
shrunk_polygon = offset_polygon(polygons, -10)
self.assertEqual(len(shrunk_polygon), 0)
# When polygon is shrunk into two polygons, it is regarded as invalid
# and an empty array is returned.
polygons = np.array([0, 0, 0, 3, 1, 2, 2, 3, 2, 0, 1, 1],
dtype=np.float32)
shrunk = offset_polygon(polygons, -1)
self.assertEqual(len(shrunk), 0)
def test_boundary_iou(self):
points = [0, 0, 0, 1, 1, 1, 1, 0]
points1 = [10, 20, 30, 40, 50, 60, 70, 80]
points2 = [0, 0, 0, 0, 0, 0, 0, 0] # Invalid polygon
points3 = [0, 0, 0, 1, 1, 0, 1, 1] # Self-intersected polygon
self.assertEqual(boundary_iou(points, points1), 0)
# test overlapping boundaries
self.assertEqual(boundary_iou(points, points), 1)
# test invalid boundaries
self.assertEqual(boundary_iou(points2, points2), 0)
self.assertEqual(boundary_iou(points3, points3, zero_division=1), 1)
self.assertEqual(boundary_iou(points2, points3), 0)
def test_sort_points(self):
points = np.array([[1, 1], [0, 0], [1, -1], [2, -2], [0, 2], [1, 1],
[0, 1], [-1, 1], [-1, -1]])
target = np.array([[-1, -1], [0, 0], [-1, 1], [0, 1], [0, 2], [1, 1],
[1, 1], [2, -2], [1, -1]])
self.assertTrue(np.allclose(target, sort_points(points)))
points = np.array([[1, 1], [1, -1], [-1, 1], [-1, -1]])
target = np.array([[-1, -1], [-1, 1], [1, 1], [1, -1]])
self.assertTrue(np.allclose(target, sort_points(points)))
points = [[1, 1], [1, -1], [-1, 1], [-1, -1]]
self.assertTrue(np.allclose(target, sort_points(points)))
points = [[0.5, 0.3], [1, 0.5], [-0.5, 0.8], [-0.1, 1]]
target = [[-0.5, 0.8], [-0.1, 1], [1, 0.5], [0.5, 0.3]]
self.assertTrue(np.allclose(target, sort_points(points)))
points = [[0.5, 3], [0.1, -0.2], [-0.5, -0.3], [-0.7, 3.1]]
target = [[-0.5, -0.3], [-0.7, 3.1], [0.5, 3], [0.1, -0.2]]
self.assertTrue(np.allclose(target, sort_points(points)))
points = [[1, 0.8], [0.8, -1], [1.8, 0.5], [1.9, -0.6], [-0.5, 2],
[-1, 1.8], [-2, 0.7], [-1.6, -0.2], [-1, -0.5]]
target = [[-1, -0.5], [-1.6, -0.2], [-2, 0.7], [-1, 1.8], [-0.5, 2],
[1, 0.8], [1.8, 0.5], [1.9, -0.6], [0.8, -1]]
self.assertTrue(np.allclose(target, sort_points(points)))
# concave polygon may failed
points = [[1, 0], [-1, 0], [0, 0], [0, -1], [0.25, 1], [0.75, 1],
[-0.25, 1], [-0.75, 1]]
target = [[-1, 0], [-0.75, 1], [-0.25, 1], [0, 0], [0.25, 1],
[0.75, 1], [1, 0], [0, -1]]
self.assertFalse(np.allclose(target, sort_points(points)))
with self.assertRaises(AssertionError):
sort_points([1, 2])
def test_sort_vertex(self):
dummy_points_x = [20, 20, 120, 120]
dummy_points_y = [20, 40, 40, 20]
expect_points_x = [20, 120, 120, 20]
expect_points_y = [20, 20, 40, 40]
with self.assertRaises(AssertionError):
sort_vertex([], dummy_points_y)
with self.assertRaises(AssertionError):
sort_vertex(dummy_points_x, [])
for perm in set(permutations([0, 1, 2, 3])):
points_x = [dummy_points_x[i] for i in perm]
points_y = [dummy_points_y[i] for i in perm]
ordered_points_x, ordered_points_y = sort_vertex(
points_x, points_y)
self.assertTrue(np.allclose(ordered_points_x, expect_points_x))
self.assertTrue(np.allclose(ordered_points_y, expect_points_y))
def test_sort_vertex8(self):
dummy_points_x = [21, 21, 122, 122]
dummy_points_y = [21, 39, 39, 21]
expect_points = [21, 21, 122, 21, 122, 39, 21, 39]
for perm in set(permutations([0, 1, 2, 3])):
points_x = [dummy_points_x[i] for i in perm]
points_y = [dummy_points_y[i] for i in perm]
points = list(chain.from_iterable(zip(points_x, points_y)))
ordered_points = sort_vertex8(points)
self.assertTrue(np.allclose(ordered_points, expect_points))