Merge pull request #6 from organicmaps/upstream

Upstream sync
This commit is contained in:
Alexander Borsuk 2023-05-30 12:28:57 +02:00 committed by GitHub
commit fd533c449d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 136 additions and 12 deletions

View file

@ -26,7 +26,7 @@ jobs:
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install flake8==6.0.0 black==23.1.0
pip install flake8==6.0.0 black==23.1.0 shapely==2.0.1
if [ -f requirements.txt ]; then pip install -r requirements.txt; fi
- name: Lint with flake8
run: |

View file

@ -1,18 +1,21 @@
import argparse
import shapely.geometry
import shapely.ops
from shapely import unary_union
from shapely.geometry import MultiPolygon, Polygon
from process_subways import DEFAULT_CITIES_INFO_URL, get_cities_info
def make_disjoint_metro_polygons(cities_info_url: str) -> None:
"""Make disjoint polygon from cities bboxes and write them
in *.poly format to stdout.
"""
cities_info = get_cities_info(cities_info_url)
polygons = []
for ci in cities_info:
bbox = tuple(map(float, ci["bbox"].split(",")))
polygon = shapely.geometry.Polygon(
polygon = Polygon(
[
(bbox[0], bbox[1]),
(bbox[0], bbox[3]),
@ -22,14 +25,17 @@ def make_disjoint_metro_polygons(cities_info_url: str) -> None:
)
polygons.append(polygon)
union = shapely.ops.unary_union(polygons)
union = unary_union(polygons)
if union.geom_type == "Polygon":
union = MultiPolygon([union])
print("all metro")
for i, polygon in enumerate(union, start=1):
for i, polygon in enumerate(union.geoms, start=1):
assert len(polygon.interiors) == 0
print(i)
for point in polygon.exterior.coords:
print(" {lon} {lat}".format(lon=point[0], lat=point[1]))
for lon, lat in polygon.exterior.coords:
print(f" {lon} {lat}")
print("END")
print("END")

View file

@ -263,6 +263,7 @@ def validate_cities(cities: list[City]) -> list[City]:
else:
c.validate()
if c.is_good:
c.calculate_distances()
good_cities.append(c)
return good_cities

View file

@ -91,7 +91,7 @@ function check_poly() {
if [ -z "${POLY-}" -o ! -f "${POLY-}" ]; then
POLY=${POLY:-$(mktemp "$TMPDIR/all-metro.XXXXXXXX.poly")}
if [ -n "$("$PYTHON" -c "import shapely" 2>&1)" ]; then
"$PYTHON" -m pip install shapely==1.7.1
"$PYTHON" -m pip install shapely==2.0.1
fi
"$PYTHON" "$SUBWAYS_PATH"/make_all_metro_poly.py \
${CITIES_INFO_URL:+--cities-info-url "$CITIES_INFO_URL"} > "$POLY"

View file

@ -1040,7 +1040,7 @@ class Route:
)
return stop_position_elements
def process_tracks(self, stop_position_elements):
def process_tracks(self, stop_position_elements: list[dict]) -> None:
tracks, line_nodes = self.build_longest_line()
for stop_el in stop_position_elements:
@ -1084,7 +1084,6 @@ class Route:
projected_stops_data = self.project_stops_on_line()
self.check_and_recover_stops_order(projected_stops_data)
self.apply_projected_stops_data(projected_stops_data)
self.calculate_distances()
def apply_projected_stops_data(self, projected_stops_data: dict) -> None:
"""Store better stop coordinates and indexes of first/last stops
@ -2074,6 +2073,11 @@ class City:
self.validate_called = True
def calculate_distances(self) -> None:
for route_master in self:
for route in route_master:
route.calculate_distances()
def find_transfers(elements, cities):
transfers = []

View file

@ -0,0 +1,2 @@
#,City,Country,Region,Stations,Subway Lines,Light Rail +Monorail,Interchanges,"BBox (lon, lat)",Networks (opt.),Approved,Comment,Source
291,Moscow,Russia,Europe,351,14,3,68,"37.1667,55.3869,38.2626,56.0136","subway,train:Московский метрополитен;МЦК;МЦД"
Can't render this file because it has a wrong number of fields in line 2.

View file

@ -0,0 +1,3 @@
#,City,Country,Region,Stations,Subway Lines,Light Rail +Monorail,Interchanges,"BBox (lon, lat)",Networks (opt.),Approved,Comment,Source
313,London,UK,Europe,750,11,23,54,"-0.9747,51.1186,0.3315,51.8459","subway,train,light_rail:London Underground;London Overground;Docklands Light Railway;London Trams;Crossrail",,,https://tfl.gov.uk/maps/track/tube
291,Moscow,Russia,Europe,351,14,3,68,"37.1667,55.3869,38.2626,56.0136","subway,train:Московский метрополитен;МЦК;МЦД"
Can't render this file because it has a wrong number of fields in line 3.

View file

Can't render this file because it has a wrong number of fields in line 2.

View file

@ -0,0 +1,108 @@
import contextlib
import io
import os
from unittest import TestCase
from make_all_metro_poly import make_disjoint_metro_polygons
cases = [
{
"csv_file": "cities_info_1city.csv",
"expected_stdout": """all metro
1
37.1667 55.3869
37.1667 56.0136
38.2626 56.0136
38.2626 55.3869
37.1667 55.3869
END
END
""",
"shape_line_ranges": [
{
"start": 2,
"end": 6,
},
],
},
{
"csv_file": "cities_info_2cities.csv",
"expected_stdout": """all metro
1
-0.9747 51.8459
0.3315 51.8459
0.3315 51.1186
-0.9747 51.1186
-0.9747 51.8459
END
2
37.1667 56.0136
38.2626 56.0136
38.2626 55.3869
37.1667 55.3869
37.1667 56.0136
END
END
""",
"shape_line_ranges": [
{
"start": 2,
"end": 6,
},
{
"start": 9,
"end": 13,
},
],
},
]
class TestMakeAllMetroPoly(TestCase):
def test_make_disjoint_metro_polygons(self) -> None:
for case in cases:
with self.subTest(msg=case["csv_file"]):
file_url = (
f"file://{os.getcwd()}/tests/assets/{case['csv_file']}"
)
stream = io.StringIO()
with contextlib.redirect_stdout(stream):
make_disjoint_metro_polygons(file_url)
generated_poly = stream.getvalue()
expected_poly = case["expected_stdout"]
# Since shapely may produce multipolygon with different order
# of polygons in it and different vertex order in a polygon,
# we should compare polygons/vertexes as sets.
generated_poly_lines = generated_poly.split("\n")
expected_poly_lines = expected_poly.split("\n")
self.assertSetEqual(
set(expected_poly_lines), set(generated_poly_lines)
)
line_ranges = case["shape_line_ranges"]
# Check that polygons are closed
for line_range in line_ranges:
self.assertEqual(
generated_poly_lines[line_range["start"]],
generated_poly_lines[line_range["end"]],
)
generated_points = [
sorted(
generated_poly_lines[r["start"] : r["end"]] # noqa 203
)
for r in line_ranges
]
expected_points = [
sorted(
expected_poly_lines[r["start"] : r["end"]] # noqa 203
)
for r in line_ranges
]
expected_points.sort()
generated_points.sort()
self.assertListEqual(expected_points, generated_points)

View file

@ -10,7 +10,7 @@ class TestPrepareCities(TestCase):
csv_path = (
Path(inspect.getfile(self.__class__)).parent
/ "assets"
/ "networks_with_bad_values.csv"
/ "cities_info_with_bad_values.csv"
)
cities = prepare_cities(cities_info_url=f"file://{csv_path}")