Eclipse SUMO - Simulation of Urban MObility
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
MSDispatch.cpp
Go to the documentation of this file.
1/****************************************************************************/
2// Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
3// Copyright (C) 2007-2025 German Aerospace Center (DLR) and others.
4// This program and the accompanying materials are made available under the
5// terms of the Eclipse Public License 2.0 which is available at
6// https://www.eclipse.org/legal/epl-2.0/
7// This Source Code may also be made available under the following Secondary
8// Licenses when the conditions for such availability set forth in the Eclipse
9// Public License 2.0 are satisfied: GNU General Public License, version 2
10// or later which is available at
11// https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
12// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
13/****************************************************************************/
18// An algorithm that performs dispatch for the taxi device
19/****************************************************************************/
20#include <config.h>
21
22#include <limits>
23#include <microsim/MSNet.h>
24#include <microsim/MSEdge.h>
26#include "MSRoutingEngine.h"
27#include "MSDispatch.h"
28
29//#define DEBUG_RESERVATION
30//#define DEBUG_DETOUR
31//#define DEBUG_COND2(obj) (obj->getID() == "p0")
32#define DEBUG_COND2(obj) (true)
33
34
35// ===========================================================================
36// Reservation methods
37// ===========================================================================
38
39std::string
41 return toString(persons);
42}
43
44// ===========================================================================
45// MSDispatch methods
46// ===========================================================================
47
49 Parameterised(params),
50 myOutput(nullptr),
51 myReservationCount(0) {
52 const std::string opt = "device.taxi.dispatch-algorithm.output";
53 if (OptionsCont::getOptions().isSet(opt)) {
54 OutputDevice::createDeviceByOption(opt, "DispatchInfo");
56 }
57}
58
60 for (auto item : myGroupReservations) {
61 for (Reservation* res : item.second) {
62 delete res;
63 }
64 }
65 myGroupReservations.clear();
66}
67
68
71 SUMOTime reservationTime,
72 SUMOTime pickupTime,
73 SUMOTime earliestPickupTime,
74 const MSEdge* from, double fromPos,
75 const MSStoppingPlace* fromStop,
76 const MSEdge* to, double toPos,
77 const MSStoppingPlace* toStop,
78 std::string group,
79 const std::string& line,
80 int maxCapacity,
81 int maxContainerCapacity) {
82 // no new reservation nedded if the person can be added to an existing group
83 if (group == "") {
84 // the default empty group implies, no grouping is wanted (and
85 // transportable ids are unique)
86 group = person->getID();
87 }
88 Reservation* result = nullptr;
89 bool added = false;
90 auto it = myGroupReservations.find(group);
91 if (it != myGroupReservations.end()) {
92 // try to add to existing reservation
93 for (Reservation* res : it->second) {
94 if (res->persons.count(person) == 0
95 && res->from == from
96 && res->to == to
97 && res->fromPos == fromPos
98 && res->toPos == toPos) {
99 if (res->persons.size() > 0 && (*res->persons.begin())->isPerson() != person->isPerson()) {
100 WRITE_WARNINGF(TL("Mixing reservations of persons and containers with the same group is not supported for % and %"),
101 (*res->persons.begin())->getID(), person->getID());
102 }
103 if ((person->isPerson() && (int)res->persons.size() >= maxCapacity) ||
104 (!person->isPerson() && (int)res->persons.size() >= maxContainerCapacity)) {
105 // split group to ensure that at least one taxi is capable of delivering group size.
106 continue;
107 }
108 res->persons.insert(person);
109 result = res;
110 added = true;
111 break;
112 }
113 }
114 }
115 if (!added) {
116 Reservation* newRes = new Reservation(toString(myReservationCount++), {person}, reservationTime, pickupTime, earliestPickupTime, from, fromPos, fromStop, to, toPos, toStop, group, line);
117 myGroupReservations[group].push_back(newRes);
118 result = newRes;
119 }
121#ifdef DEBUG_RESERVATION
122 if (DEBUG_COND2(person)) std::cout << SIMTIME
123 << " addReservation p=" << person->getID()
124 << " rT=" << time2string(reservationTime)
125 << " pT=" << time2string(pickupTime)
126 << " from=" << from->getID() << " fromPos=" << fromPos
127 << " to=" << to->getID() << " toPos=" << toPos
128 << " group=" << group
129 << " added=" << added
130 << "\n";
131#endif
132 return result;
133}
134
135
136std::string
138 const MSEdge* from, double fromPos,
139 const MSEdge* to, double toPos,
140 std::string group) {
141 if (group == "") {
142 // the default empty group implies, no grouping is wanted (and
143 // transportable ids are unique)
144 group = person->getID();
145 }
146 std::string removedID = "";
147 auto it = myGroupReservations.find(group);
148 if (it != myGroupReservations.end()) {
149 for (auto itRes = it->second.begin(); itRes != it->second.end(); itRes++) {
150 Reservation* res = *itRes;
151 if (res->persons.count(person) != 0
152 && res->from == from
153 && res->to == to
154 && res->fromPos == fromPos
155 && res->toPos == toPos) {
156 res->persons.erase(person);
157 if (res->persons.empty()) {
158 removedID = res->id;
159 it->second.erase(itRes);
160 // cleans up MSDispatch_Greedy
162 if (it->second.empty()) {
163 myGroupReservations.erase(it);
164 }
165 }
166 break;
167 }
168 }
169 } else {
170 auto it2 = myRunningReservations.find(group);
171 if (it2 != myRunningReservations.end()) {
172 for (auto item : it2->second) {
173 const Reservation* res = item.first;
174 if (res->persons.count(person) != 0
175 && res->from == from
176 && res->to == to
177 && res->fromPos == fromPos
178 && res->toPos == toPos) {
179 MSDevice_Taxi* taxi = item.second;
180 taxi->cancelCustomer(person);
181 if (res->persons.empty()) {
182 removedID = res->id;
183 }
184 break;
185 }
186 }
187 }
188 }
190#ifdef DEBUG_RESERVATION
191 if (DEBUG_COND2(person)) std::cout << SIMTIME
192 << " removeReservation p=" << person->getID()
193 << " from=" << from->getID() << " fromPos=" << fromPos
194 << " to=" << to->getID() << " toPos=" << toPos
195 << " group=" << group
196 << " removedID=" << removedID
197 << " hasServable=" << myHasServableReservations
198 << "\n";
199#endif
200 return removedID;
201}
202
203
206 const MSEdge* from, double fromPos,
207 const MSEdge* to, double toPos,
208 std::string group, double newFromPos) {
209 if (group == "") {
210 // the default empty group implies, no grouping is wanted (and
211 // transportable ids are unique)
212 group = person->getID();
213 }
214 Reservation* result = nullptr;
215 std::string updatedID = "";
216 auto it = myGroupReservations.find(group);
217 if (it != myGroupReservations.end()) {
218 for (auto itRes = it->second.begin(); itRes != it->second.end(); itRes++) {
219 Reservation* res = *itRes;
220 // TODO: if there is already a reservation with the newFromPos, add to this reservation
221 // TODO: if there are other persons in this reservation, create a new reservation for the updated one
222 if (res->persons.count(person) != 0
223 && res->from == from
224 && res->to == to
225 && res->fromPos == fromPos
226 && res->toPos == toPos) {
227 // update fromPos
228 res->fromPos = newFromPos;
229 result = res;
230 updatedID = res->id;
231 break;
232 }
233 }
234 }
235#ifdef DEBUG_RESERVATION
236 if (DEBUG_COND2(person)) std::cout << SIMTIME
237 << " updateReservationFromPos p=" << person->getID()
238 << " from=" << from->getID() << " fromPos=" << fromPos
239 << " to=" << to->getID() << " toPos=" << toPos
240 << " group=" << group
241 << " newFromPos=" << newFromPos
242 << " updatedID=" << updatedID
243 << "\n";
244#endif
245 return result;
246}
247
248
249std::vector<Reservation*>
251 std::vector<Reservation*> reservations;
252 for (const auto& it : myGroupReservations) {
253 reservations.insert(reservations.end(), it.second.begin(), it.second.end());
254 }
255 return reservations;
256}
257
258
259std::vector<const Reservation*>
261 std::vector<const Reservation*> result;
262 for (auto item : myRunningReservations) {
263 for (auto item2 : item.second) {
264 result.push_back(item2.first);
265 }
266 }
267 return result;
268}
269
270
271void
273 auto itR = myRunningReservations.find(res->group);
274 if (itR != myRunningReservations.end() && itR->second.count(res) != 0) {
275 return; // was redispatch
276 }
277 auto it = myGroupReservations.find(res->group);
278 if (it == myGroupReservations.end()) {
279 throw ProcessError(TL("Inconsistent group reservations."));
280 }
281 auto it2 = std::find(it->second.begin(), it->second.end(), res);
282 if (it2 == it->second.end()) {
283 throw ProcessError(TL("Inconsistent group reservations (2)."));
284 }
285 myRunningReservations[res->group][res] = taxi;
286 const_cast<Reservation*>(*it2)->state = Reservation::ASSIGNED;
287 it->second.erase(it2);
288 if (it->second.empty()) {
289 myGroupReservations.erase(it);
290 }
291}
292
293
294void
296 myRunningReservations[res->group].erase(res);
297 if (myRunningReservations[res->group].empty()) {
298 myRunningReservations.erase(res->group);
299 }
300 delete res;
301}
302
303
306 ConstMSEdgeVector edges;
307 router.compute(taxi->getHolder().getEdge(), taxi->getHolder().getPositionOnLane() - NUMERICAL_EPS,
308 res.from, res.fromPos, &taxi->getHolder(), t, edges, true);
309 return TIME2STEPS(router.recomputeCosts(edges, &taxi->getHolder(), t));
310}
311
312
313double
315 const MSEdge* from, double fromPos,
316 const MSEdge* via, double viaPos,
317 const MSEdge* to, double toPos,
319 double& timeDirect) {
320 ConstMSEdgeVector edges;
321 if (timeDirect < 0) {
322 router.compute(from, fromPos, to, toPos, &taxi->getHolder(), t, edges, true);
323 timeDirect = router.recomputeCostsPos(edges, &taxi->getHolder(), fromPos, toPos, t);
324 edges.clear();
325 }
326
327 router.compute(from, fromPos, via, viaPos, &taxi->getHolder(), t, edges, true);
328 const double start = STEPS2TIME(t);
329 const double leg1 = router.recomputeCostsPos(edges, &taxi->getHolder(), fromPos, viaPos, t);
330#ifdef DEBUG_DETOUR
331 std::cout << " leg1=" << toString(edges) << " startPos=" << fromPos << " toPos=" << viaPos << " time=" << leg1 << "\n";
332#endif
333 const double wait = MAX2(0.0, STEPS2TIME(viaTime) - (start + leg1));
334 edges.clear();
335 const SUMOTime timeContinue = TIME2STEPS(start + leg1 + wait);
336 router.compute(via, viaPos, to, toPos, &taxi->getHolder(), timeContinue, edges, true);
337 const double leg2 = router.recomputeCostsPos(edges, &taxi->getHolder(), viaPos, toPos, timeContinue);
338 const double timeDetour = leg1 + wait + leg2;
339#ifdef DEBUG_DETOUR
340 std::cout << " leg2=" << toString(edges) << " startPos=" << viaPos << " toPos=" << toPos << " time=" << leg2 << "\n";
341 std::cout << " t=" << STEPS2TIME(t) << " vt=" << STEPS2TIME(viaTime)
342 << " from=" << from->getID() << " to=" << to->getID() << " via=" << via->getID()
343 << " direct=" << timeDirect << " detour=" << timeDetour << " wait=" << wait << "\n";
344#endif
345 return timeDetour;
346}
347
348
349int
351 assert(res->persons.size() > 0);
352 return ((*res->persons.begin())->isPerson()
354 : taxi->getHolder().getVehicleType().getContainerCapacity()) - (int)res->persons.size();
355}
356
357
358/****************************************************************************/
long long int SUMOTime
Definition GUI.h:36
#define DEBUG_COND2(obj)
Definition MESegment.cpp:52
std::vector< const MSEdge * > ConstMSEdgeVector
Definition MSEdge.h:74
#define WRITE_WARNINGF(...)
Definition MsgHandler.h:284
#define TL(string)
Definition MsgHandler.h:301
std::string time2string(SUMOTime t, bool humanReadable)
convert SUMOTime to string (independently of global format setting)
Definition SUMOTime.cpp:91
#define STEPS2TIME(x)
Definition SUMOTime.h:55
#define SIMTIME
Definition SUMOTime.h:62
#define TIME2STEPS(x)
Definition SUMOTime.h:57
T MAX2(T a, T b)
Definition StdDefs.h:82
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:46
A device which collects info on the vehicle trip (mainly on departure and arrival)
bool cancelCustomer(const MSTransportable *t)
remove person from reservations
OutputDevice * myOutput
optional file output for dispatch information
Definition MSDispatch.h:203
int remainingCapacity(const MSDevice_Taxi *taxi, const Reservation *res)
whether the given taxi has sufficient capacity to serve the reservation
static SUMOTime computePickupTime(SUMOTime t, const MSDevice_Taxi *taxi, const Reservation &res, SUMOAbstractRouter< MSEdge, SUMOVehicle > &router)
compute time to pick up the given reservation
virtual std::string removeReservation(MSTransportable *person, const MSEdge *from, double fromPos, const MSEdge *to, double toPos, std::string group)
remove person from reservation. If the whole reservation is removed, return its id
bool myHasServableReservations
whether the last call to computeDispatch has left servable reservations
Definition MSDispatch.h:191
virtual Reservation * updateReservationFromPos(MSTransportable *person, const MSEdge *from, double fromPos, const MSEdge *to, double toPos, std::string group, double newFromPos)
update fromPos of the person's reservation. TODO: if there is already a reservation with the newFromP...
std::map< std::string, std::vector< Reservation * > > myGroupReservations
Definition MSDispatch.h:206
std::vector< Reservation * > getReservations()
retrieve all reservations
virtual std::vector< const Reservation * > getRunningReservations()
retrieve all reservations that were already dispatched and are still active
static double computeDetourTime(SUMOTime t, SUMOTime viaTime, const MSDevice_Taxi *taxi, const MSEdge *from, double fromPos, const MSEdge *via, double viaPos, const MSEdge *to, double toPos, SUMOAbstractRouter< MSEdge, SUMOVehicle > &router, double &timeDirect)
compute directTime and detourTime
virtual Reservation * addReservation(MSTransportable *person, SUMOTime reservationTime, SUMOTime pickupTime, SUMOTime earliestPickupTime, const MSEdge *from, double fromPos, const MSStoppingPlace *fromStop, const MSEdge *to, double toPos, const MSStoppingPlace *tostop, std::string group, const std::string &line, int maxCapacity, int maxContainerCapacity)
add a new reservation
int myReservationCount
Definition MSDispatch.h:205
virtual void fulfilledReservation(const Reservation *res)
erase reservation from storage
MSDispatch(const Parameterised::Map &params)
Constructor;.
std::map< std::string, std::map< const Reservation *, MSDevice_Taxi * > > myRunningReservations
Definition MSDispatch.h:200
virtual ~MSDispatch()
Destructor.
void servedReservation(const Reservation *res, MSDevice_Taxi *taxi)
A road/street connecting two junctions.
Definition MSEdge.h:77
A lane area vehicles can halt at.
bool isPerson() const
Whether it is a person.
SUMOVehicle & getHolder() const
Returns the vehicle that holds this device.
int getPersonCapacity() const
Get this vehicle type's person capacity.
int getContainerCapacity() const
Get this vehicle type's container capacity.
const std::string & getID() const
Returns the id.
Definition Named.h:74
static OptionsCont & getOptions()
Retrieves the options.
static bool createDeviceByOption(const std::string &optionName, const std::string &rootElement="", const std::string &schemaFile="")
Creates the device using the output definition stored in the named option.
static OutputDevice & getDeviceByOption(const std::string &name)
Returns the device described by the option.
An upper class for objects with additional parameters.
std::map< std::string, std::string > Map
parameters map
virtual bool compute(const E *from, const E *to, const V *const vehicle, SUMOTime msTime, std::vector< const E * > &into, bool silent=false)=0
Builds the route between the given edges using the minimum effort at the given time The definition of...
virtual double recomputeCosts(const std::vector< const E * > &edges, const V *const v, SUMOTime msTime, double *lengthp=nullptr) const
double recomputeCostsPos(const std::vector< const E * > &edges, const V *const v, double fromPos, double toPos, SUMOTime msTime, double *lengthp=nullptr) const
virtual const MSVehicleType & getVehicleType() const =0
Returns the object's "vehicle" type.
virtual const MSEdge * getEdge() const =0
Returns the edge the object is currently at.
virtual double getPositionOnLane() const =0
Get the object's position along the lane.
std::string id
Definition MSDispatch.h:76
const MSEdge * to
Definition MSDispatch.h:84
std::string getID() const
debug identification
double fromPos
Definition MSDispatch.h:82
const MSEdge * from
Definition MSDispatch.h:81
std::string group
Definition MSDispatch.h:87
ReservationState state
Definition MSDispatch.h:90
std::set< const MSTransportable * > persons
Definition MSDispatch.h:77
double toPos
Definition MSDispatch.h:85