5.2. Modifying the Network
object: Public transport schedules¶
GeNet has a number of methods to change the Schedule objects. Available as a jupyter notebook or wiki page. Make sure you validate the end result.
# read example network
import os
import shutil
from genet import Route, Service, Stop, read_matsim
path_to_matsim_network = "example_data/pt2matsim_network"
network = os.path.join(path_to_matsim_network, "network.xml")
schedule = os.path.join(path_to_matsim_network, "schedule.xml")
vehicles = os.path.join(path_to_matsim_network, "vehicles.xml")
n = read_matsim(
path_to_network=network, epsg="epsg:27700", path_to_schedule=schedule, path_to_vehicles=vehicles
)
# you don't need to read the vehicles file, but doing so ensures all vehicles
# in the schedule are of the expected type and the definition of the vehicle
# is preserved
n.print()
Graph info: MultiDiGraph with 1662 nodes and 3166 edges Schedule info: Schedule: Number of services: 9 Number of routes: 68 Number of stops: 118
Adding Routes, Services¶
You can add Route
s and Service
s. To add a Route
, you need to identify which existing Service
it should come under.
You can make Routes by specifying a headway parameter, that will generate explicit trips for you:
route = Route(
route_short_name="N55",
mode="bus",
headway_spec={("07:00:00", "08:00:00"): 15},
arrival_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
departure_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
id="new_route",
# route= ['834', '1573', '3139', '3141', '574', '3154', '979', '980', '981'],
await_departure=[True, True, True, True],
stops=[
n.schedule.stop("490000235X"),
Stop(id="new_stop", x=529500, y=181300, name="New Stop", epsg="epsg:27700"),
Stop(id="other_new_stop", x=529502, y=181302, name="Other New Stop", epsg="epsg:27700"),
n.schedule.stop("490010689KB"),
],
)
route.trips
{'trip_id': ['new_route_07:00:00', 'new_route_07:15:00', 'new_route_07:30:00', 'new_route_07:45:00', 'new_route_08:00:00'], 'trip_departure_time': ['07:00:00', '07:15:00', '07:30:00', '07:45:00', '08:00:00'], 'vehicle_id': ['veh_bus_new_route_07:00:00', 'veh_bus_new_route_07:15:00', 'veh_bus_new_route_07:30:00', 'veh_bus_new_route_07:45:00', 'veh_bus_new_route_08:00:00']}
Or you can use provide the exact trips and vehicles on this route:
route = Route(
route_short_name="N55",
mode="bus",
trips={
"trip_id": ["fun_trip_1", "fun_trip_2"],
"trip_departure_time": ["03:53:00", "16:23:00"],
"vehicle_id": ["fun_bus_1", "fun_bus_2"],
},
arrival_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
departure_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
id="new_route",
# route= ['834', '1573', '3139', '3141', '574', '3154', '979', '980', '981'],
await_departure=[True, True, True, True],
stops=[
n.schedule.stop("490000235X"),
Stop(id="new_stop", x=529500, y=181300, name="New Stop", epsg="epsg:27700"),
Stop(id="other_new_stop", x=529502, y=181302, name="Other New Stop", epsg="epsg:27700"),
n.schedule.stop("490010689KB"),
],
)
You can create and add a new Service
, or add the route to an existing Service
new_service = Service(id="new_service", routes=[route])
new_service.print()
Service ID: new_service Name: N55 Number of routes: 1 Number of stops: 4
n.schedule.add_service(Service(id="new_service", routes=[route]))
2023-12-08 13:44:17,452 - Added Services with IDs `['new_service']` and Routes: [['new_route']]
n.schedule.add_route("20274", route)
2023-12-08 13:44:17,570 - Route with ID `new_route` for Service 20274 within already exists in the Schedule. This Route will be reindexed to `20274_4` 2023-12-08 13:44:17,573 - Reindexed Route from new_route to 20274_4 2023-12-08 13:44:17,575 - Added Routes with IDs ['20274_4'], to Services `['20274']` within the Schedule /Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles}
n.schedule["new_service"].print()
Service ID: new_service Name: N55 Number of routes: 1 Number of stops: 4
n.schedule.route("20274_4").ordered_stops
['490000235X', 'new_stop', 'other_new_stop', '490010689KB']
You can also refer to existing stops in the Schedule
when creating aRoute
to be added. You can either just pass Stop IDs as strings or use a stop method on the schedule to take and use that stop object. Note that in the case of the former (passing ID strings), the route will not have the spatial information for those stops until it is added to the Schedule.
r = Route(
route_short_name="N55",
mode="bus",
trips={
"trip_id": ["some_trip_1"],
"trip_departure_time": ["16:23:00"],
"vehicle_id": ["some_bus_2"],
},
arrival_offsets=["00:00:00", "00:06:00"],
departure_offsets=["00:00:00", "00:06:00"],
id="another_new_route",
# route= ['834', '1573', '3139', '3141', '574', '3154', '979', '980', '981'],
await_departure=[True, True],
stops=["490000235X.link:834", "490010689KB.link:981"],
)
r.graph().nodes["490000235X.link:834"]
{'routes': {'another_new_route'}}
n.schedule.add_route("20274", r)
2023-12-08 13:44:17,731 - Added Routes with IDs ['another_new_route'], to Services `['20274']` within the Schedule /Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles}
r.graph().nodes["490000235X.link:834"]
{'services': {'14134', '18853', '20274'}, 'routes': {'VJ12ba6089dfb2733e29c415a1a0015fef30fd5305', 'VJ256e98df611ff48afe737ddc81cbcde82e4e81c8', 'VJ2aba67e3ed98f2ed5f5966c1ac394cbf6d1943d7', 'VJ375a660d47a2aa570aa20a8568012da8497ffecf', 'VJ4e2b897edf0e7b8a8e3b5516ab43ce56f72c5cff', 'VJa7f37392e276aeac26c7e73bbc05e6a71af38dba', 'VJd78967364a302cf232c5139d40622dcb6c238c9e', 'VJdf3936da1a51eb33db594ef99738802c14b19995', 'VJf3e316e5e605bb512147dee2a989be5a82ef1b5f', 'VJf9a22035ae6f25bb420df833474943ad76065c89', 'another_new_route'}, 'id': '490000235X.link:834', 'x': 529981.7958802709, 'y': 181412.0975758662, 'epsg': 'epsg:27700', 'name': 'Tottenham Court Road Station (Stop X)', 'lon': -0.12809598708996447, 'lat': 51.51668503324075, 's2_id': 5221390722025467597, 'linkRefId': '834', 'isBlocking': 'false'}
r = Route(
route_short_name="N55",
mode="bus",
trips={
"trip_id": ["some_trip_1"],
"trip_departure_time": ["16:23:00"],
"vehicle_id": ["some_bus_2"],
},
arrival_offsets=["00:00:00", "00:06:00"],
departure_offsets=["00:00:00", "00:06:00"],
id="another_new_route_2",
# route= ['834', '1573', '3139', '3141', '574', '3154', '979', '980', '981'],
await_departure=[True, True],
stops=[n.schedule.stop("490000235X.link:834"), n.schedule.stop("490010689KB.link:981")],
)
n.schedule.add_route("20274", r)
2023-12-08 13:44:17,959 - Added Routes with IDs ['another_new_route_2'], to Services `['20274']` within the Schedule /Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles}
Note that for a Schedule to be a valid MATSim network, each stop referred to by a route needs a linkRefId
attribute which links the stop to the Network
.
Trying to add Stops with IDs already in the Schedule will result in an error, unless the force=True
is set. The added route/service will inherit the data stored under those stops. The idea is that you can either specify the Stops in the route/service to be added correctly, or if they are to be changed, you use a dedicated method - check section 'Modifying data' below.
route = Route(
route_short_name="N55",
mode="bus",
trips={
"trip_id": ["fun_trip_1", "fun_trip_2"],
"trip_departure_time": ["03:53:00", "16:23:00"],
"vehicle_id": ["fun_bus_1", "fun_bus_2"],
},
arrival_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
departure_offsets=["00:00:00", "00:02:00", "00:04:00", "00:06:00"],
id="another_new_route_3",
# route= ['834', '1573', '3139', '3141', '574', '3154', '979', '980', '981'],
await_departure=[True, True, True, True],
stops=[
Stop(id="490000235X.link:834", x=529981, y=181412, epsg="epsg:27700"),
Stop(id="new_stop", x=529500, y=181300, epsg="epsg:27700", name="New Stop"),
Stop(id="other_new_stop", x=529502, y=181302, epsg="epsg:27700", name="Other New Stop"),
Stop(id="490010689KB.link:981", x=529166, y=181256, epsg="epsg:27700"),
],
)
n.schedule.add_route("20274", route, force=True)
2023-12-08 13:44:18,133 - The following stops will inherit the data currently stored under those Stop IDs in the Schedule: ['490000235X.link:834', '490010689KB.link:981']. 2023-12-08 13:44:18,135 - Added Routes with IDs ['another_new_route_3'], to Services `['20274']` within the Schedule /Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles}
Note the message above: The following stops will inherit the data currently stored under those Stop IDs in the Schedule: ['490000235X.link:834', '490010689KB.link:981'].
NOTE: adding routes and services results in new vehicles (unless you reuse the ones already in the Schedule---beware that the same vehicle cannot service multiple trips at the same time, genet does not currently have checks for this, the user needs to be mindful of the physics of shared vehicles). New vehicles need definitions, you can add them yourself to schedule.vehicles['vehicle_id'] = {'type': 'bus'}
ensuring this vehicle type is defined in schedule.vehicle_types['bus']
, or you can use a genet method to generate those vehicles, the type will be derived from the mode of the route. Then you can check if all of the types that vehicles are referring to have definitions.
len(n.schedule.vehicles)
1566
n.schedule.generate_vehicles()
/Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles}
n.schedule.validate_vehicle_definitions()
2023-12-08 13:44:18,335 - The following vehicle types are missing from the `vehicle_types` attribute: {'bus'} 2023-12-08 13:44:18,337 - Vehicles affected by missing vehicle types: {'fun_bus_1': {'type': 'bus'}, 'fun_bus_2': {'type': 'bus'}, 'some_bus_2': {'type': 'bus'}}
False
len(n.schedule.vehicles)
1566
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
0 | 2023-12-08 13:44:17 | add | service | None | new_service | None | {'id': 'new_service', 'name': 'N55'} | [(add, , [('id', 'new_service'), ('name', 'N55... |
1 | 2023-12-08 13:44:17 | add | route | None | 20274_4 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
2 | 2023-12-08 13:44:17 | add | route | None | another_new_route | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
3 | 2023-12-08 13:44:17 | add | route | None | another_new_route_2 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
4 | 2023-12-08 13:44:18 | add | route | None | another_new_route_3 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
There are no methods to add Stop
objects on their own. They are added to the Schedule with Route
and Service
objects.
Routing added Services/Routes¶
You can use methods in GeNet to relate the Stops of PT services and find network routes between them. First you need to know the ID of the Service you want to snap (you can also snap the entire schedule, but depending on the size and complexity of your network and schedule it might take a long time). GeNet will then relate all stops in that service to a link in the Network graph and route all of the Route
s of the Service
object. It will do this for directed subsets (subgraphs) of the Service (e.g. if you have a Northbound and Southboud service, the stops may have to find different links in the graph)
There are a lot of different parameters you can add to this method, that give you a bit more control.
solver
: You can specify different mathematical solvers. For example GLPK, an open source solver which can be found here: https://www.gnu.org/software/glpk/. Another good open source choice is CBC: https://projects.coin-or.org/Cbc. You specify it as a string e.g. 'glpk', 'cbc', 'gurobi'. The solver needs to support MILP - mixed integer linear programmingallow_partial
: Sometimes there isn't a link available for snapping within threshold for all stops. If allowed (default) an artificial self-loop link will be created as well as any connecting links to that unsnapped stop, under modal conditions. If set to False and the problem is partial, it will raise PartialMaxStableSetProblem error instead.distance_threshold
,step_size
: You can vary the threshold for snapping stops to links. There are two parameters, the overall threshold and a step size; when snapping, the search area for links increases in steps until some links are found, we don't use the threshold right away because we don't want to have too many choices for snapping (makes it a little less heavy computationally)).additional_modes
: You can specify additional modes (when snapping, genet will consider a modal subset of the network that matches the pt service, e.g. bus, but you might want to use links that allow cars too).allow_directional_split
: You can opt for splitting the problem of snapping by direction. GeNet will then solve a series of problems for subgraphs (disjoint w.r.t. edges) of the Service. This is useful for rail type services that might share the same stop regardless of direction (in comparison to buses which have distinct bus stops, depending on which way they're travelling).
Check the methods' doc strings for up to date details.
n.schedule["new_service"].route("new_route").network_links
[]
n.schedule["new_service"].route("new_route").ordered_stops
['490000235X', 'new_stop', 'other_new_stop', '490010689KB']
if shutil.which("cbc"):
n.route_service("new_service")
else:
print("Cannot route service without a solver installed")
2023-12-08 13:44:19,064 - Routing Service new_service with modes = {'bus'} 2023-12-08 13:44:19,077 - Building Maximum Stable Set for PT graph with 4 stops and 3 edges 2023-12-08 13:44:19,121 - This Maximum Stable Set Problem is partially viable. 2023-12-08 13:44:19,124 - Maximum Stable Set problem to snap the PT graph to the network is partially viable, meaning not all stops have found a link to snap to within the distance_threshold.Partial snapping is ON, this problem will proceed to the solver. 2023-12-08 13:44:19,125 - Passing problem to solver 2023-12-08 13:44:19,128 - Initializing ordered Set vertices with a fundamentally unordered data source (type: set). This WILL potentially lead to nondeterministic behavior in Pyomo 2023-12-08 13:44:19,132 - Passing problem to solver 2023-12-08 13:44:19,208 - Successfully snapped 3 stops to network links. 2023-12-08 13:44:19,217 - Stop ID changes detected for Routes: {'new_route'} 2023-12-08 13:44:19,218 - Changed Route attributes for 1 routes 2023-12-08 13:44:19,222 - Added 1 nodes 2023-12-08 13:44:19,266 - Generated 0 link ids. 2023-12-08 13:44:19,269 - Added 2 links 2023-12-08 13:44:19,270 - Changed Link attributes for 4 links
n.schedule["new_service"].route("new_route").network_links
['artificial_link===from:490000235X===to:490000235X', 'artificial_link===from:490000235X===to:9521035', '3154', '979', '980', '981']
n.schedule["new_service"].route("new_route").ordered_stops
['490000235X.link:artificial_link===from:490000235X===to:490000235X', 'new_stop.link:3154', 'other_new_stop.link:3154', '490010689KB.link:981']
Reindexing¶
n.schedule["new_service"].reindex(new_id="more_appropriate_id")
2023-12-08 13:44:19,290 - Reindexed Service from new_service to more_appropriate_id
n.schedule.route("new_route").reindex(new_id="more_appropriate_route_id")
2023-12-08 13:44:19,297 - Reindexed Route from new_route to more_appropriate_route_id
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
3 | 2023-12-08 13:44:17 | add | route | None | another_new_route_2 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
4 | 2023-12-08 13:44:18 | add | route | None | another_new_route_3 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, , [('route_short_name', 'N55'), ('mode'... |
5 | 2023-12-08 13:44:19 | modify | route | new_route | new_route | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | [(add, route, [(0, 'artificial_link===from:490... |
6 | 2023-12-08 13:44:19 | modify | service | new_service | more_appropriate_id | {'id': 'new_service'} | {'id': 'more_appropriate_id'} | [(change, id, (new_service, more_appropriate_i... |
7 | 2023-12-08 13:44:19 | modify | route | new_route | more_appropriate_route_id | {'id': 'new_route'} | {'id': 'more_appropriate_route_id'} | [(change, id, (new_route, more_appropriate_rou... |
Removing Stops, Routes, Services¶
n.schedule.remove_service("more_appropriate_id")
/Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles} 2023-12-08 13:44:19,404 - Removed Services with IDs `more_appropriate_id`, and Routes: {'more_appropriate_route_id'}
for route_id in {"another_new_route", "another_new_route_2", "another_new_route_3"}:
n.schedule.remove_route(route_id)
/Users/bryn.pickering/Repos/arup-group/genet/genet/schedule_elements.py:1803: UserWarning: DataFrame columns are not unique, some columns will be omitted. self.vehicles = {**df.T.to_dict(), **self.vehicles} 2023-12-08 13:44:19,507 - Removed Routes with IDs ['another_new_route_2'], to Services `20274`. 2023-12-08 13:44:19,603 - Removed Routes with IDs ['another_new_route_3'], to Services `20274`.
2023-12-08 13:44:19,700 - Removed Routes with IDs ['another_new_route'], to Services `20274`.
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
7 | 2023-12-08 13:44:19 | modify | route | new_route | more_appropriate_route_id | {'id': 'new_route'} | {'id': 'more_appropriate_route_id'} | [(change, id, (new_route, more_appropriate_rou... |
8 | 2023-12-08 13:44:19 | remove | service | more_appropriate_id | None | {'id': 'more_appropriate_id', 'name': 'N55'} | None | [(remove, , [('id', 'more_appropriate_id'), ('... |
9 | 2023-12-08 13:44:19 | remove | route | another_new_route_2 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | None | [(remove, , [('route_short_name', 'N55'), ('mo... |
10 | 2023-12-08 13:44:19 | remove | route | another_new_route_3 | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | None | [(remove, , [('route_short_name', 'N55'), ('mo... |
11 | 2023-12-08 13:44:19 | remove | route | another_new_route | None | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | None | [(remove, , [('route_short_name', 'N55'), ('mo... |
You can also remove Stop
s. This will disconnect Route
s and Service
s using that Stop
s and likely render them invalid. The method will warn you which Route
s and Servce
s are affected.
n.schedule.remove_stop("new_stop")
2023-12-08 13:44:19,715 - Removed Stops with indices `['new_stop']`.Routes affected: {'20274_4'}. Services affected: {'20274'}.
You can also remove Stop
s. This will disconnect Route
s and Service
s using that Stop
s and likely render them invalid. The method will warn you which Route
s and Servce
s are affected.
n.schedule.remove_unused_stops()
2023-12-08 13:44:19,721 - Removed Stops with indices `['490000252R', '9400ZZLUTCR3', 'new_stop.link:3154', '490015196N', '9400ZZLUOXC6', '9400ZZLUGDG1', '9400ZZLUOXC1', '490000235W1', '490000235X.link:artificial_link===from:490000235X===to:490000235X', '490010198W', '490000252S', '9400ZZLUESQ2', '490000091F', '9400ZZLURGP1', '9400ZZLUWRR3', '9400ZZLUWRR1', '490000173RF', '9400ZZLUTCR4', '490000235N', '490015196R', '9400ZZLUOXC3', '9400ZZLURGP2', '9400ZZLUWRR2', '490000356NE', '9400ZZLUOXC5', '490011126K', '9400ZZLUTCR1', 'other_new_stop.link:3154', '9400ZZLUOXC4', '490000173RD', '9400ZZLUGPS2', '490013600C', '9400ZZLUWRR4', '490019675D', '490000091E', '9400ZZLUESQ1', '9400ZZLUOXC2']`.Routes affected: set(). Services affected: set().
Modifying data stored for Stops, Routes, Services¶
Applying known or pre-computed changes¶
Applying changes or new attributes to Services, Routes and Stops can be done via Schedule level methods. They all work with a dictionary where the keys are the object IDs and the values are dictionaries holding attribute names and values. The method to extract a DataFrame on attributes comes in handy here. E.g.
df = n.schedule.service_attribute_data(keys="name")
df.head()
name | |
---|---|
20274 | N55 |
18915 | N5 |
14134 | 98 |
15660 | 113 |
18853 | N8 |
DataFrames are easy to work with. Youcould for exmaple manipulate the names or use other data to change these. For demonstration here, let's just set the names to something easy.
df["name"] = df["name"].apply(lambda x: f"Service_{x}")
df.head()
name | |
---|---|
20274 | Service_N55 |
18915 | Service_N5 |
14134 | Service_98 |
15660 | Service_113 |
18853 | Service_N8 |
You can then convert this to a dictionary and pass it to the apply_attributes_to_services
method.
n.schedule.apply_attributes_to_services(df.T.to_dict())
2023-12-08 13:44:19,740 - Changed Service attributes for 9 services
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
54 | 2023-12-08 13:44:19 | modify | service | 18853 | 18853 | {'id': '18853', 'name': 'N8'} | {'id': '18853', 'name': 'Service_N8'} | [(change, name, (N8, Service_N8))] |
55 | 2023-12-08 13:44:19 | modify | service | 12430 | 12430 | {'id': '12430', 'name': '205'} | {'id': '12430', 'name': 'Service_205'} | [(change, name, (205, Service_205))] |
56 | 2023-12-08 13:44:19 | modify | service | 15234 | 15234 | {'id': '15234', 'name': '134'} | {'id': '15234', 'name': 'Service_134'} | [(change, name, (134, Service_134))] |
57 | 2023-12-08 13:44:19 | modify | service | 17732 | 17732 | {'id': '17732', 'name': 'N20'} | {'id': '17732', 'name': 'Service_N20'} | [(change, name, (N20, Service_N20))] |
58 | 2023-12-08 13:44:19 | modify | service | 14073 | 14073 | {'id': '14073', 'name': '94'} | {'id': '14073', 'name': 'Service_94'} | [(change, name, (94, Service_94))] |
You can do the same for Routes
and Stops
. Your dictionaries cannot however hold changes to indices. You will encounter an error and should use reindex
methods for such operations.
n.schedule.apply_attributes_to_routes(
{
"VJ375a660d47a2aa570aa20a8568012da8497ffecf": {
"name": "my_favourite_route",
"mode": "piggyback",
}
}
)
2023-12-08 13:44:19,754 - Changed Route attributes for 1 routes
n.schedule.apply_attributes_to_stops({"490000235YB.link:574": {"new_attribute": "hello!"}})
2023-12-08 13:44:19,759 - Changed Stop attributes for 1 stops
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
56 | 2023-12-08 13:44:19 | modify | service | 15234 | 15234 | {'id': '15234', 'name': '134'} | {'id': '15234', 'name': 'Service_134'} | [(change, name, (134, Service_134))] |
57 | 2023-12-08 13:44:19 | modify | service | 17732 | 17732 | {'id': '17732', 'name': 'N20'} | {'id': '17732', 'name': 'Service_N20'} | [(change, name, (N20, Service_N20))] |
58 | 2023-12-08 13:44:19 | modify | service | 14073 | 14073 | {'id': '14073', 'name': '94'} | {'id': '14073', 'name': 'Service_94'} | [(change, name, (94, Service_94))] |
59 | 2023-12-08 13:44:19 | modify | route | VJ375a660d47a2aa570aa20a8568012da8497ffecf | VJ375a660d47a2aa570aa20a8568012da8497ffecf | {'route_short_name': 'N55', 'mode': 'bus', 'ar... | {'route_short_name': 'N55', 'mode': 'piggyback... | [(change, mode, (bus, piggyback)), (add, , [('... |
60 | 2023-12-08 13:44:19 | modify | stop | 490000235YB.link:574 | 490000235YB.link:574 | {'services': {'18853', '14134', '20274'}, 'rou... | {'services': {'18853', '14134', '20274'}, 'rou... | [(add, , [('new_attribute', 'hello!')])] |
Trip and vehicle changes¶
You can use trips_to_dataframe
to extract all of the trips, their departures and vehicle IDs associated with the trips in the schedule. Trip ids need not be unique, route IDs provide a secondary index. Associated service IDs are also given for convenience.
trips = n.schedule.trips_to_dataframe(gtfs_day="20210101")
trips.head()
mode | route_id | service_id | trip_id | trip_departure_time | vehicle_id | |
---|---|---|---|---|---|---|
0 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | VJ0b0180c7b6bcef5834ec857e9b5a94254803694f_03:... | 2021-01-01 03:48:00 | veh_2160_bus |
1 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a_03:... | 2021-01-01 03:18:00 | veh_2161_bus |
2 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | VJ5e32459fcb7ab3481a1bab1b2c106f592a67d8ff_04:... | 2021-01-01 04:43:00 | veh_2162_bus |
3 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | VJ691d8b8a2b60e4f943babbea813c047824d60e6e_02:... | 2021-01-01 02:28:00 | veh_2163_bus |
4 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | VJ9b62613eaaadfb63206602708def459f48c9d7e5_01:... | 2021-01-01 01:58:00 | veh_2164_bus |
Let's change all of the trip ids to something shorter
trips["trip_id"] = "trip_" + trips.index.to_series().astype(str)
trips.head()
mode | route_id | service_id | trip_id | trip_departure_time | vehicle_id | |
---|---|---|---|---|---|---|
0 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | trip_0 | 2021-01-01 03:48:00 | veh_2160_bus |
1 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | trip_1 | 2021-01-01 03:18:00 | veh_2161_bus |
2 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | trip_2 | 2021-01-01 04:43:00 | veh_2162_bus |
3 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | trip_3 | 2021-01-01 02:28:00 | veh_2163_bus |
4 | bus | VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | 17732 | trip_4 | 2021-01-01 01:58:00 | veh_2164_bus |
You can set_trips_dataframe
which takes this dataframe and applies changes to all route trips based on the data in the dataframe. This means you can generate this DataFrame as shown below, manipulate trips (delete them, add new ones), change their departure times or change their vehicle ids to be shared for differnt trips, perhaps on some temporal logic and as long as the dataframe has the same schema, you can use it to set new trips in the schedule. This will appear in the changelog as a route level modify event.
Nb removing all trips of the same route from the dataframe will have no effect when being applied. If there is data in the dataframe for a route, all of its trips will be replaced by the data in the dataframe, and if there is no data for a route in the frame, no changes will be applied to that route (i.e. the trips attribute for routes missing from the dataframe will not be set as empty).
n.schedule.set_trips_dataframe(trips)
n.schedule.route_attribute_data(keys=[{"trips": "trip_id"}]).head()
2023-12-08 13:44:19,838 - Changed Route attributes for 69 routes
trips::trip_id | |
---|---|
VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a | [trip_0, trip_1, trip_2, trip_3, trip_4, trip_... |
VJ5b511605b1e07428c2e0a7d676d301c6c40dcca6 | [trip_12, trip_13, trip_14, trip_15, trip_16, ... |
VJ4e2b897edf0e7b8a8e3b5516ab43ce56f72c5cff | [trip_26, trip_27, trip_28, trip_29, trip_30, ... |
VJd78967364a302cf232c5139d40622dcb6c238c9e | [trip_45, trip_46] |
VJ256e98df611ff48afe737ddc81cbcde82e4e81c8 | [trip_47] |
Generating new trips using headway information¶
You can replace old trips using headway information. This is useful when creating scenario networks. You can do it to a Route or Service (by specifying the route ID to be changed within) objects.
route_id = "VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13"
n.schedule.route(route_id).trips
{'trip_id': ['trip_203', 'trip_204'], 'trip_departure_time': ['07:51:00', '22:50:00'], 'vehicle_id': ['veh_887_bus', 'veh_888_bus']}
n.schedule.generate_trips_from_headway(
route_id=route_id, headway_spec={("07:51:00", "22:50:00"): 120}
) # headway in minutes
2023-12-08 13:44:19,863 - Changed Route attributes for 1 routes
n.schedule.route(route_id).trips
{'trip_id': ['VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_07:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_09:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_11:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_13:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_15:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_17:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_19:51:00', 'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_21:51:00'], 'trip_departure_time': ['07:51:00', '09:51:00', '11:51:00', '13:51:00', '15:51:00', '17:51:00', '19:51:00', '21:51:00'], 'vehicle_id': ['veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_07:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_09:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_11:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_13:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_15:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_17:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_19:51:00', 'veh_Bus_VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13_21:51:00']}
Applying changes using functions or dictionary mappings¶
If you have some logic that can be written into a function of object's attributes, you can pass this to apply_function_to_x
methods. You need to select location
, which refers to the name of the attribute the result should be stored under. It can already exist and be overwritten. The function passed, is not expected to work with all objects. It will fail silently, only evaluating and generating outputs where possible.
from shapely.geometry import Point
def add_shapely_geometry_points(stop_attribs):
return Point(stop_attribs["x"], stop_attribs["y"])
n.schedule.apply_function_to_stops(add_shapely_geometry_points, location="geometry")
2023-12-08 13:44:19,894 - Changed Stop attributes for 85 stops
n.schedule.change_log().tail(2)
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
214 | 2023-12-08 13:44:19 | modify | stop | 490000235YB | 490000235YB | {'id': '490000235YB', 'x': 529570.7813227688, ... | {'id': '490000235YB', 'x': 529570.7813227688, ... | [(add, , [('geometry', <POINT (529570.781 1813... |
215 | 2023-12-08 13:44:19 | modify | stop | other_new_stop | other_new_stop | {'services': {'20274'}, 'routes': {'20274_4'},... | {'services': {'20274'}, 'routes': {'20274_4'},... | [(add, , [('geometry', <POINT (529502 181302)>... |
n.schedule.stop_attribute_data(keys=["name", "x", "y", "geometry"]).head()
name | x | y | geometry | |
---|---|---|---|---|
490000235X.link:834 | Tottenham Court Road Station (Stop X) | 529981.795880 | 181412.097576 | POINT (529981.7958802709 181412.0975758662) |
490000235YB.link:574 | Oxford Street Soho Street (Stop YB) | 529570.781323 | 181336.281593 | POINT (529570.7813227688 181336.2815925331) |
490014214HE.link:3154 | Wardour Street (Stop OM) | 529477.750156 | 181314.437043 | POINT (529477.7501560802 181314.43704307207) |
490010689KB.link:981 | Great Titchfield Street Oxford Circus Station... | 529166.734973 | 181256.336723 | POINT (529166.7349732723 181256.33672284335) |
490000235V.link:3140 | Tottenham Court Road Station (Stop V) | 529716.723832 | 181371.384818 | POINT (529716.7238324936 181371.3848181052) |
from geopandas import GeoDataFrame
GeoDataFrame(n.schedule.stop_attribute_data(keys="geometry")).plot()
<Axes: >
n.schedule.stop("490000235YB.link:574").__dict__
{'id': '490000235YB.link:574', 'x': 529570.7813227688, 'y': 181336.2815925331, 'epsg': 'epsg:27700', 'name': 'Oxford Street Soho Street (Stop YB)', 'lat': 51.51609803324078, 'lon': -0.13404398709291904, 's2_id': 5221390696959560815, 'linkRefId': '574', 'isBlocking': 'false', 'new_attribute': 'hello!', 'geometry': <POINT (529570.781 181336.282)>}
For this let's say we want to reduce the number of trips. For simplicity of demonstration we don't have about which trips we delete, but logic around timings of trips can be added in this function, as the trips are saved as one of routes attributes (check out the summary methods)
# before
len(n.schedule.route_trips_to_dataframe())
2023-12-08 13:44:20,294 - `route_trips_to_dataframe` method is deprecated and will be replaced by `trips_to_dataframe`in later versions.
1570
def reduce_trips(attribs):
# just delete any other trip
attribs["trips"]["trip_id"] = attribs["trips"]["trip_id"][::2]
attribs["trips"]["trip_departure_time"] = attribs["trips"]["trip_departure_time"][::2]
attribs["trips"]["vehicle_id"] = attribs["trips"]["vehicle_id"][::2]
return attribs["trips"]
n.schedule.apply_function_to_routes(reduce_trips, "trips")
2023-12-08 13:44:20,387 - Changed Route attributes for 69 routes
# after
len(n.schedule.route_trips_to_dataframe())
2023-12-08 13:44:20,395 - `route_trips_to_dataframe` method is deprecated and will be replaced by `trips_to_dataframe`in later versions.
796
Note, this could also be done using the route_trips_to_dataframe
and set_route_trips_dataframe
mentioned above.
Let's give an example of using a mapping. We can re-use the service name DataFrame we generated above.
df["new_name"] = "Brand_new_name" + df["name"]
df.head()
name | new_name | |
---|---|---|
20274 | Service_N55 | Brand_new_nameService_N55 |
18915 | Service_N5 | Brand_new_nameService_N5 |
14134 | Service_98 | Brand_new_nameService_98 |
15660 | Service_113 | Brand_new_nameService_113 |
18853 | Service_N8 | Brand_new_nameService_N8 |
name_map = dict(zip(df["name"], df["new_name"]))
name_map
{'Service_N55': 'Brand_new_nameService_N55', 'Service_N5': 'Brand_new_nameService_N5', 'Service_98': 'Brand_new_nameService_98', 'Service_113': 'Brand_new_nameService_113', 'Service_N8': 'Brand_new_nameService_N8', 'Service_205': 'Brand_new_nameService_205', 'Service_134': 'Brand_new_nameService_134', 'Service_N20': 'Brand_new_nameService_N20', 'Service_94': 'Brand_new_nameService_94'}
In this case, location
refers to the attribute to be mapped.
n.schedule.apply_function_to_services(name_map, location="name")
2023-12-08 13:44:20,431 - Changed Service attributes for 9 services
n.schedule.change_log().tail()
timestamp | change_event | object_type | old_id | new_id | old_attributes | new_attributes | diff | |
---|---|---|---|---|---|---|---|---|
289 | 2023-12-08 13:44:20 | modify | service | 18853 | 18853 | {'id': '18853', 'name': 'Service_N8'} | {'id': '18853', 'name': 'Brand_new_nameService... | [(change, name, (Service_N8, Brand_new_nameSer... |
290 | 2023-12-08 13:44:20 | modify | service | 12430 | 12430 | {'id': '12430', 'name': 'Service_205'} | {'id': '12430', 'name': 'Brand_new_nameService... | [(change, name, (Service_205, Brand_new_nameSe... |
291 | 2023-12-08 13:44:20 | modify | service | 15234 | 15234 | {'id': '15234', 'name': 'Service_134'} | {'id': '15234', 'name': 'Brand_new_nameService... | [(change, name, (Service_134, Brand_new_nameSe... |
292 | 2023-12-08 13:44:20 | modify | service | 17732 | 17732 | {'id': '17732', 'name': 'Service_N20'} | {'id': '17732', 'name': 'Brand_new_nameService... | [(change, name, (Service_N20, Brand_new_nameSe... |
293 | 2023-12-08 13:44:20 | modify | service | 14073 | 14073 | {'id': '14073', 'name': 'Service_94'} | {'id': '14073', 'name': 'Brand_new_nameService... | [(change, name, (Service_94, Brand_new_nameSer... |
{s.name for s in n.schedule.services()}
{'Brand_new_nameService_113', 'Brand_new_nameService_134', 'Brand_new_nameService_205', 'Brand_new_nameService_94', 'Brand_new_nameService_98', 'Brand_new_nameService_N20', 'Brand_new_nameService_N5', 'Brand_new_nameService_N55', 'Brand_new_nameService_N8'}
Subsetting¶
You can subset the Schedule object using Service IDs.
services = n.schedule.extract_service_ids_on_attributes(
{"name": ["Brand_new_nameService_134", "Brand_new_nameService_98"]}
)
len(services)
2
n.schedule.print()
Schedule: Number of services: 9 Number of routes: 69 Number of stops: 85
sub_s = n.schedule.subschedule(service_ids=services)
2023-12-08 13:44:20,655 - The following vehicle types are missing from the `vehicle_types` attribute: {'piggyback'} 2023-12-08 13:44:20,658 - Vehicles affected by missing vehicle types: {'veh_2331_bus': {'type': 'piggyback'}, 'veh_2333_bus': {'type': 'piggyback'}, 'veh_2335_bus': {'type': 'piggyback'}, 'veh_2337_bus': {'type': 'piggyback'}, 'veh_2339_bus': {'type': 'piggyback'}} 2023-12-08 13:44:20,713 - Removed Services with IDs `20274`, and Routes: {'VJ6c64ab7b477e201cae950efde5bd0cb4e2e8888e', '20274_4', 'VJ812fad65e7fa418645b57b446f00cba573f2cdaf', 'VJ375a660d47a2aa570aa20a8568012da8497ffecf'} 2023-12-08 13:44:20,774 - Removed Services with IDs `18915`, and Routes: {'VJ887921c00645929c5402ac46592e57c368ea63a1', 'VJ8a4b1ca7dfd0a130abd1de9f55f3b756617dd4ca', 'VJ520ec0c0ca58a849349fa614b5cf9270ac5c93da', 'VJ0d304b95d39f4bce48e6ff26ddd73a9c06f17f4f', 'VJb08f8a2de01a4ef99d3b7fefd9022117ac307531'} 2023-12-08 13:44:20,831 - Removed Services with IDs `15660`, and Routes: {'VJ3716910ec59c370d9f5c69137df7276b68cf0a08', 'VJf2e0de4f5dad68cb03064e6064e372dde52cc678', 'VJ1cf651142378958b52229bfe1fa552e49136e60e'} 2023-12-08 13:44:20,895 - Removed Services with IDs `18853`, and Routes: {'VJ8cacca9a6722c497c413005568182ecf4d50b160', 'VJfc4917783c2ca3227789fa7c532c9adf47702095', 'VJf3e316e5e605bb512147dee2a989be5a82ef1b5f'} 2023-12-08 13:44:20,935 - Removed Services with IDs `12430`, and Routes: {'VJf8e38a73359b6cf743d8e35ee64ef1f7b7914daa', 'VJ15419796737689e742962a625abcf3fd5b3d58b1', 'VJ0f3c08222de16c2e278be0a1bf0f9ea47370774e', 'VJ8f9aea7491080b0137d3092706f53dc11f7dba45', 'VJef7f20c3a9bf1419f6401e1e9131fe2c634bcb9a', 'VJ95b4c534d7c903d76ec0340025aa88b81dba3ce4', 'VJ06420fdab0dfe5c8e7f2f9504df05cf6289cd7d3', 'VJeb72539d69ddf8e29f1adf74d43953def196ae41', 'VJ948e8caa0f08b9c6bf6330927893942c474b5100', 'VJeae6e634f8479e0b6712780d5728f0afca964e64', 'VJ06cd41dcd58d947097df4a8f33234ef423210154', 'VJ235c8fca539cf931b3c673f9b056606384aff950'} 2023-12-08 13:44:20,977 - Removed Services with IDs `17732`, and Routes: {'VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a', 'VJ85c23573d670bab5485618b0c5fddff3314efc89'} 2023-12-08 13:44:21,006 - Removed Services with IDs `14073`, and Routes: {'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13', 'VJaa5ee0daec7529d7668c81fe7fac0c4ff545daea', 'VJe18efadf172576fea7989ec1f233f26854c0f66a', 'VJfc35884fc4f11dc408a209c19f56f3b60f634daf', 'VJe8cffad09738ff7b9698b333e3247918d5c45358', 'VJb4309b7a9598539ab9942ea1bcadc60a91b978ba', 'VJ24fe211d801738b556a39f815256d7f6bc544ec5', 'VJdbc280077e505b4f8d66586ca51751a125cb4ef0', 'VJc8cdbd902dadeebeeb4dbd7332b564ee2e4b00ce', 'VJd9dbeefeca6d74ef2594a17514ebc08ee2d503b2', 'VJea6046f64f85febf1854290fb8f76e921e3ac96b', 'VJd132b905afc6c0e8e8a994142e301ca5c0f70e22', 'VJe6ba07ef9f19ae40517261ad626bf34dd656491a', 'VJ93d8207ae8540b4ff59d47c9ee1ec5689084522d', 'VJf6055fdf9ef0dd6d0500b6c11adcfdd4d10655dc'} 2023-12-08 13:44:21,007 - Removed Stops with indices `['490000173JB.link:1663', 'other_new_stop']`.Routes affected: set(). Services affected: set().
sub_s.print()
Schedule: Number of services: 2 Number of routes: 25 Number of stops: 83
Spatial Subsetting¶
There is a convenience method to extract a subset schedule using spatial conditions - refer to Using Schedule notebook or docs to learn more about what spatial inputs are supported.
# pop it in https://s2.sidewalklabs.com/ to see it
region = "48761ad0b14,48761ad0b3,48761ad0b5,48761ad0b7,48761ad0b84,48761ad0d,48761ad0e04,48761ad11f4,48761ad11fc,48761ad13,48761ad145fc,48761ad147,48761ad14c,48761ad153,48761ad41df,48761ad41e4,48761ad41fc,48761ad421,48761ad427,48761ad429,48761ad42b,48761ad5d5,48761ad5d7,48761ad5d9,48761ad5df,48761ad5e1,48761ad5e3,48761ad64,48761ad6c,48761ad71,48761ad73,48761ad744,48761ad74c,48761ad751,48761ad753,48761ad757554,48761ad75c,48761ad77,48761ad79,48761ad7b,48761ad7d,48761ad7e4,48761ad7ec,48761ad7f4,48761ad7f9,48761ad7fb,48761ad7fd,48761ad827,48761ad829,48761ad82b,48761ad9d5,48761ad9d7,48761ad9d84,48761b2802c,48761b2817,48761b281c,48761b283,48761b2847fc,48761b2849,48761b284b,48761b29b5,48761b29b7,48761b29b9,48761b29d,48761b29e4,48761b29e9,48761b29ea4,48761b29ef,48761b29fb4,48761b29fd"
n.subnetwork_on_spatial_condition(region, how="intersect")
2023-12-08 13:44:21,462 - Subsetting a Network will likely result in a disconnected network graph. A cleaner will be ran that will remove links to make the resulting Network strongly connected for modes: car, walk, bike. 2023-12-08 13:44:21,512 - Schedule will be subsetted using given services: ['12430']. Links pertaining to their network routes will also be retained. 2023-12-08 13:44:21,599 - The following vehicle types are missing from the `vehicle_types` attribute: {'piggyback'} 2023-12-08 13:44:21,599 - Vehicles affected by missing vehicle types: {'veh_2331_bus': {'type': 'piggyback'}, 'veh_2333_bus': {'type': 'piggyback'}, 'veh_2335_bus': {'type': 'piggyback'}, 'veh_2337_bus': {'type': 'piggyback'}, 'veh_2339_bus': {'type': 'piggyback'}} 2023-12-08 13:44:21,658 - Removed Services with IDs `20274`, and Routes: {'VJ6c64ab7b477e201cae950efde5bd0cb4e2e8888e', '20274_4', 'VJ812fad65e7fa418645b57b446f00cba573f2cdaf', 'VJ375a660d47a2aa570aa20a8568012da8497ffecf'} 2023-12-08 13:44:21,789 - Removed Services with IDs `18915`, and Routes: {'VJ887921c00645929c5402ac46592e57c368ea63a1', 'VJ8a4b1ca7dfd0a130abd1de9f55f3b756617dd4ca', 'VJ520ec0c0ca58a849349fa614b5cf9270ac5c93da', 'VJ0d304b95d39f4bce48e6ff26ddd73a9c06f17f4f', 'VJb08f8a2de01a4ef99d3b7fefd9022117ac307531'} 2023-12-08 13:44:21,834 - Removed Services with IDs `14134`, and Routes: {'VJ4e2b897edf0e7b8a8e3b5516ab43ce56f72c5cff', 'VJd78967364a302cf232c5139d40622dcb6c238c9e', 'VJ256e98df611ff48afe737ddc81cbcde82e4e81c8', 'VJ4e311a625836374adf4cfaa841224840dbeb7619', 'VJ2c87b2a59184888f3175b55bde7b02d024ea8607', 'VJ26095b8f9f9db92ca2e53d4c086a7dcd82a13be9', 'VJdb0c128567fcbcc063d554ae1c95851cee41b909', 'VJ5909ba51575a9459eb0013fbd31c8205455ca2fd', 'VJa7f37392e276aeac26c7e73bbc05e6a71af38dba', 'VJb93a17a405fe502c5b3a2d6544105b0311da9fe2', 'VJ2aba67e3ed98f2ed5f5966c1ac394cbf6d1943d7', 'VJ12ba6089dfb2733e29c415a1a0015fef30fd5305', 'VJ4c6fa387b0d4be94a6c3679b94790b183e2558ca', 'VJdf3936da1a51eb33db594ef99738802c14b19995', 'VJf9a22035ae6f25bb420df833474943ad76065c89', 'VJ323d02e117552af1565f2ff1273a612655c829c4'} 2023-12-08 13:44:21,875 - Removed Services with IDs `15660`, and Routes: {'VJ3716910ec59c370d9f5c69137df7276b68cf0a08', 'VJf2e0de4f5dad68cb03064e6064e372dde52cc678', 'VJ1cf651142378958b52229bfe1fa552e49136e60e'} 2023-12-08 13:44:21,912 - Removed Services with IDs `18853`, and Routes: {'VJ8cacca9a6722c497c413005568182ecf4d50b160', 'VJfc4917783c2ca3227789fa7c532c9adf47702095', 'VJf3e316e5e605bb512147dee2a989be5a82ef1b5f'} 2023-12-08 13:44:21,938 - Removed Services with IDs `15234`, and Routes: {'VJ5b511605b1e07428c2e0a7d676d301c6c40dcca6', 'VJ28a8a6a4ab02807a4fdfd199e5c2ca0622d34d0c', 'VJ3d50b96792ae8495dbe5a5e372849a60c48b2279', 'VJd4cbfb092a104ac6a3164a86e9765f68734fdfcf', 'VJ652c769bc42361cc0308dff59a1fdcf0949bdade', 'VJ8ccf92aa0f351b2e31f1a078b968dff4c2505c02', 'VJ1a8cc306354fdc322d739ae644eb73444341d08d', 'VJ9b58a59e3d74941586a5bca7726a8aa624da67fc', 'VJbf9d4fdb976223e6a026c0c669ed290418abefee'} 2023-12-08 13:44:21,963 - Removed Services with IDs `17732`, and Routes: {'VJ0cb60de3ed229c1413abac506e770b6ab8a7c49a', 'VJ85c23573d670bab5485618b0c5fddff3314efc89'} 2023-12-08 13:44:21,979 - Removed Services with IDs `14073`, and Routes: {'VJ6cf76a4c03cca468cb6954db7f7aad5ae189df13', 'VJaa5ee0daec7529d7668c81fe7fac0c4ff545daea', 'VJe18efadf172576fea7989ec1f233f26854c0f66a', 'VJfc35884fc4f11dc408a209c19f56f3b60f634daf', 'VJe8cffad09738ff7b9698b333e3247918d5c45358', 'VJb4309b7a9598539ab9942ea1bcadc60a91b978ba', 'VJ24fe211d801738b556a39f815256d7f6bc544ec5', 'VJdbc280077e505b4f8d66586ca51751a125cb4ef0', 'VJc8cdbd902dadeebeeb4dbd7332b564ee2e4b00ce', 'VJd9dbeefeca6d74ef2594a17514ebc08ee2d503b2', 'VJea6046f64f85febf1854290fb8f76e921e3ac96b', 'VJd132b905afc6c0e8e8a994142e301ca5c0f70e22', 'VJe6ba07ef9f19ae40517261ad626bf34dd656491a', 'VJ93d8207ae8540b4ff59d47c9ee1ec5689084522d', 'VJf6055fdf9ef0dd6d0500b6c11adcfdd4d10655dc'} 2023-12-08 13:44:21,981 - Removed Stops with indices `['490000173JB.link:1663', 'other_new_stop']`.Routes affected: set(). Services affected: set(). 2023-12-08 13:44:21,990 - Param: strongly_connected_modes is defaulting to `{'car', 'walk', 'bike'}` You can change this behaviour by passing the parameter. 2023-12-08 13:44:21,995 - The graph for mode car is not strongly connected. The largest 1 connected components will be extracted. 2023-12-08 13:44:22,000 - Extracting largest connected components resulted in mode: car being deleted from 82 edges /Users/bryn.pickering/Repos/arup-group/genet/genet/core.py:674: UserWarning: Boolean Series key will be reindexed to match DataFrame index. df = df.loc[df.index.intersection(links)][df["modes"].apply(lambda x: bool(mode & x))] 2023-12-08 13:44:22,033 - Changed Link attributes for 82 links 2023-12-08 13:44:22,036 - Removed 56 links 2023-12-08 13:44:22,037 - The graph for modes: walk does not have any connected components. This method returns True because if the graph is empty for this mode there is no reason to fail this check. 2023-12-08 13:44:22,038 - The graph for modes: bike does not have any connected components. This method returns True because if the graph is empty for this mode there is no reason to fail this check. 2023-12-08 13:44:22,039 - Subsetted Network is ready - do not forget to validate and visualise your subset!
<Network instance at 11001368784: with graph: MultiDiGraph with 156 nodes and 203 edges and schedule Schedule: Number of services: 1 Number of routes: 12 Number of stops: 83