LCOV - code coverage report
Current view: top level - src/utils/vehicle - SUMORouteHandler.cpp (source / functions) Coverage Total Hit
Test: lcov.info Lines: 96.7 % 336 325
Test Date: 2026-03-02 16:00:03 Functions: 91.7 % 12 11

            Line data    Source code
       1              : /****************************************************************************/
       2              : // Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
       3              : // Copyright (C) 2001-2026 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              : /****************************************************************************/
      14              : /// @file    SUMORouteHandler.cpp
      15              : /// @author  Daniel Krajzewicz
      16              : /// @author  Jakob Erdmann
      17              : /// @author  Sascha Krieg
      18              : /// @author  Michael Behrisch
      19              : /// @date    Mon, 9 Jul 2001
      20              : ///
      21              : // Parser for routes during their loading
      22              : /****************************************************************************/
      23              : #include <config.h>
      24              : 
      25              : #include <utils/common/MsgHandler.h>
      26              : #include <utils/common/FileHelpers.h>
      27              : #include <utils/options/OptionsCont.h>
      28              : #include <utils/vehicle/SUMOVTypeParameter.h>
      29              : #include <utils/vehicle/SUMOVehicleParserHelper.h>
      30              : #include <utils/xml/XMLSubSys.h>
      31              : 
      32              : #include "SUMORouteHandler.h"
      33              : 
      34              : 
      35              : // ===========================================================================
      36              : // method definitions
      37              : // ===========================================================================
      38              : 
      39        80426 : SUMORouteHandler::SUMORouteHandler(const std::string& file, const std::string& expectedRoot, const bool hardFail) :
      40              :     SUMOSAXHandler(file, expectedRoot),
      41        80426 :     myHardFail(hardFail),
      42        80426 :     myVehicleParameter(nullptr),
      43        80426 :     myLastDepart(-1),
      44        80426 :     myActiveRouteColor(nullptr),
      45        80426 :     myCurrentCosts(0.),
      46        80426 :     myCurrentVType(nullptr),
      47       241276 :     myBeginDefault(OptionsCont::getOptions().exists("begin") ? string2time(OptionsCont::getOptions().getString("begin")) : 0),
      48       241276 :     myEndDefault(OptionsCont::getOptions().exists("end") ? string2time(OptionsCont::getOptions().getString("end")) : -1),
      49        80426 :     myFirstDepart(-1),
      50        80426 :     myInsertStopEdgesAt(-1),
      51       160852 :     myAllowInternalRoutes(false) {
      52        80426 : }
      53              : 
      54              : 
      55        79986 : SUMORouteHandler::~SUMORouteHandler() {
      56        79986 :     delete myVehicleParameter;
      57        79986 :     delete myCurrentVType;
      58        79986 : }
      59              : 
      60              : 
      61              : bool
      62       609396 : SUMORouteHandler::checkLastDepart() {
      63       609396 :     if (myVehicleParameter->departProcedure == DepartDefinition::GIVEN) {
      64       608456 :         if (myVehicleParameter->depart < myLastDepart) {
      65           16 :             WRITE_WARNINGF(TL("Route file should be sorted by departure time, ignoring '%'!"), myVehicleParameter->id);
      66            8 :             return false;
      67              :         }
      68              :     }
      69              :     return true;
      70              : }
      71              : 
      72              : 
      73              : void
      74       658131 : SUMORouteHandler::registerLastDepart() {
      75              :     // register only non public transport to parse all public transport lines in advance
      76       658131 :     if (myVehicleParameter && myVehicleParameter->line == "" && myVehicleParameter->departProcedure == DepartDefinition::GIVEN) {
      77       652477 :         myLastDepart = myVehicleParameter->depart;
      78       652477 :         if (myFirstDepart == -1) {
      79        40579 :             myFirstDepart = myLastDepart;
      80              :         }
      81              :     }
      82              :     // else: we don't know when this vehicle will depart. keep the previous known depart time
      83       658131 : }
      84              : 
      85              : 
      86              : void
      87     11275533 : SUMORouteHandler::myStartElement(int element, const SUMOSAXAttributes& attrs) {
      88     11275533 :     myElementStack.push_back(element);
      89     11275533 :     switch (element) {
      90       402181 :         case SUMO_TAG_VEHICLE:
      91              :         case SUMO_TAG_PERSON:
      92              :         case SUMO_TAG_CONTAINER:
      93              :             // if myVehicleParameter is nullptr this will do nothing
      94       402181 :             delete myVehicleParameter;
      95              :             // we set to nullptr to have a consistent state if the parsing fails
      96       402181 :             myVehicleParameter = nullptr;
      97       402181 :             myVehicleParameter = SUMOVehicleParserHelper::parseVehicleAttributes(element, attrs, myHardFail, false, false, myAllowInternalRoutes);
      98       401921 :             myParamStack.push_back(myVehicleParameter);
      99       401921 :             if (element != SUMO_TAG_VEHICLE) {
     100        46376 :                 addTransportable(attrs, element == SUMO_TAG_PERSON);
     101              :             }
     102              :             break;
     103        21203 :         case SUMO_TAG_FLOW:
     104              :             // delete if myVehicleParameter isn't null
     105        21203 :             if (myVehicleParameter) {
     106            0 :                 delete myVehicleParameter;
     107            0 :                 myVehicleParameter = nullptr;
     108              :             }
     109              :             // parse vehicle parameters
     110              :             // might be called to parse vehicles from additional file in the
     111              :             // context of quickReload. In this case, rerouter flows must be ignored
     112        21203 :             if (myElementStack.size() == 1 || myElementStack[myElementStack.size() - 2] != SUMO_TAG_CALIBRATOR) {
     113        21203 :                 myVehicleParameter = SUMOVehicleParserHelper::parseFlowAttributes(SUMO_TAG_FLOW, attrs, myHardFail, true, myBeginDefault, myEndDefault, myAllowInternalRoutes);
     114              :             }
     115              :             // check if myVehicleParameter was successfully created
     116        21085 :             if (myVehicleParameter) {
     117              :                 // check tag
     118        21085 :                 if (myVehicleParameter->routeid.empty()) {
     119              :                     // open a route flow (It could be a flow with embedded route)
     120        12194 :                     openFlow(attrs);
     121              :                 } else {
     122              :                     // open a route flow
     123         8891 :                     openRouteFlow(attrs);
     124              :                 }
     125        21085 :                 myParamStack.push_back(myVehicleParameter);
     126              :             }
     127              :             break;
     128         2266 :         case SUMO_TAG_PERSONFLOW:
     129              :         case SUMO_TAG_CONTAINERFLOW:
     130              :             // delete if myVehicleParameter isn't null
     131         2266 :             if (myVehicleParameter) {
     132            0 :                 delete myVehicleParameter;
     133            0 :                 myVehicleParameter = nullptr;
     134              :             }
     135              :             // create a new flow
     136         2266 :             myVehicleParameter = SUMOVehicleParserHelper::parseFlowAttributes((SumoXMLTag)element, attrs, myHardFail, true, myBeginDefault, myEndDefault, myAllowInternalRoutes);
     137         2266 :             myParamStack.push_back(myVehicleParameter);
     138         2266 :             break;
     139        63179 :         case SUMO_TAG_VTYPE:
     140              :             // delete if myCurrentVType isn't null
     141        63179 :             if (myCurrentVType != nullptr) {
     142            4 :                 delete myCurrentVType;
     143            4 :                 myCurrentVType = nullptr;
     144              :             }
     145              :             // create a new vType
     146        63179 :             myCurrentVType = SUMOVehicleParserHelper::beginVTypeParsing(attrs, myHardFail, getFileName());
     147        63149 :             myParamStack.push_back(myCurrentVType);
     148        63149 :             break;
     149          533 :         case SUMO_TAG_VTYPE_DISTRIBUTION:
     150          533 :             openVehicleTypeDistribution(attrs);
     151          533 :             break;
     152       385337 :         case SUMO_TAG_ROUTE:
     153       385337 :             openRoute(attrs);
     154       385285 :             break;
     155        92353 :         case SUMO_TAG_ROUTE_DISTRIBUTION:
     156        92353 :             openRouteDistribution(attrs);
     157        92353 :             break;
     158        39963 :         case SUMO_TAG_STOP:
     159        39963 :             myParamStack.push_back(addStop(attrs));
     160        39851 :             break;
     161       235970 :         case SUMO_TAG_TRIP: {
     162              :             // delete if myVehicleParameter isn't null
     163       235970 :             if (myVehicleParameter) {
     164            0 :                 delete myVehicleParameter;
     165            0 :                 myVehicleParameter = nullptr;
     166              :             }
     167              :             // parse vehicle parameters
     168       235970 :             myVehicleParameter = SUMOVehicleParserHelper::parseVehicleAttributes(element, attrs, myHardFail, false, false, myAllowInternalRoutes);
     169              :             // check if myVehicleParameter was successfully created
     170       235916 :             if (myVehicleParameter) {
     171       235916 :                 myVehicleParameter->parametersSet |= VEHPARS_FORCE_REROUTE;
     172       235916 :                 myActiveRouteID = "!" + myVehicleParameter->id;
     173              :                 // open trip
     174       235916 :                 openTrip(attrs);
     175       235916 :                 myParamStack.push_back(myVehicleParameter);
     176              :             }
     177              :             break;
     178              :         }
     179         4811 :         case SUMO_TAG_PERSONTRIP:
     180         4811 :             addPersonTrip(attrs);
     181         4796 :             break;
     182        40354 :         case SUMO_TAG_WALK:
     183        40354 :             addWalk(attrs);
     184        40332 :             break;
     185           23 :         case SUMO_TAG_INTERVAL: {
     186              :             bool ok;
     187           23 :             myBeginDefault = attrs.getSUMOTimeReporting(SUMO_ATTR_BEGIN, nullptr, ok);
     188           23 :             myEndDefault = attrs.getSUMOTimeReporting(SUMO_ATTR_END, nullptr, ok);
     189              :             break;
     190              :         }
     191         5153 :         case SUMO_TAG_RIDE:
     192         5153 :             addRide(attrs);
     193         5121 :             break;
     194         1875 :         case SUMO_TAG_TRANSPORT:
     195         1875 :             addTransport(attrs);
     196         1859 :             break;
     197         1584 :         case SUMO_TAG_TRANSHIP:
     198         1584 :             addTranship(attrs);
     199         1584 :             break;
     200        56792 :         case SUMO_TAG_PARAM:
     201        56792 :             addParam(attrs);
     202        56792 :             break;
     203      9921956 :         default:
     204              :             // parse embedded car following model information
     205      9921956 :             if (myCurrentVType != nullptr) {
     206           32 :                 WRITE_WARNINGF(TL("Defining car-following parameters in a nested element is deprecated in vType '%', use attributes instead!"), myCurrentVType->id);
     207           16 :                 if (!SUMOVehicleParserHelper::parseCFMParams(myCurrentVType, (SumoXMLTag)element, attrs, true)) {
     208            0 :                     if (myHardFail) {
     209            0 :                         throw ProcessError(TL("Invalid parsing embedded VType"));
     210              :                     } else {
     211            0 :                         WRITE_ERROR(TL("Invalid parsing embedded VType"));
     212              :                     }
     213              :                 }
     214              :             }
     215              :             break;
     216              :     }
     217     11274822 : }
     218              : 
     219              : 
     220              : void
     221     11267989 : SUMORouteHandler::myEndElement(int element) {
     222     11267989 :     switch (element) {
     223        39851 :         case SUMO_TAG_STOP:
     224              :             myParamStack.pop_back();
     225              :             break;
     226       385285 :         case SUMO_TAG_ROUTE:
     227       385285 :             closeRoute();
     228       385231 :             break;
     229        63145 :         case SUMO_TAG_VTYPE:
     230        63145 :             closeVType();
     231        63121 :             delete myCurrentVType;
     232        63121 :             myCurrentVType = nullptr;
     233              :             myParamStack.pop_back();
     234              :             break;
     235        44177 :         case SUMO_TAG_PERSON:
     236        44177 :             closePerson();
     237        44151 :             delete myVehicleParameter;
     238        44151 :             myVehicleParameter = nullptr;
     239              :             myParamStack.pop_back();
     240              :             break;
     241         1931 :         case SUMO_TAG_PERSONFLOW:
     242         1931 :             closePersonFlow();
     243         1919 :             delete myVehicleParameter;
     244         1919 :             myVehicleParameter = nullptr;
     245              :             myParamStack.pop_back();
     246              :             break;
     247         2095 :         case SUMO_TAG_CONTAINER:
     248         2095 :             closeContainer();
     249         2087 :             delete myVehicleParameter;
     250         2087 :             myVehicleParameter = nullptr;
     251              :             myParamStack.pop_back();
     252              :             break;
     253          331 :         case SUMO_TAG_CONTAINERFLOW:
     254          331 :             closeContainerFlow();
     255          331 :             delete myVehicleParameter;
     256          331 :             myVehicleParameter = nullptr;
     257              :             myParamStack.pop_back();
     258              :             break;
     259       355397 :         case SUMO_TAG_VEHICLE:
     260       355397 :             if (myVehicleParameter == nullptr) {
     261              :                 break;
     262              :             }
     263       355397 :             if (myVehicleParameter->repetitionNumber > 0) {
     264            0 :                 myVehicleParameter->repetitionNumber++; // for backwards compatibility
     265              :                 // it is a flow, thus no break here
     266              :                 FALLTHROUGH;
     267              :             } else {
     268       355397 :                 closeVehicle();
     269       355311 :                 delete myVehicleParameter;
     270       355311 :                 myVehicleParameter = nullptr;
     271              :                 myParamStack.pop_back();
     272              :                 break;
     273              :             }
     274              :         case SUMO_TAG_FLOW:
     275        21083 :             if (myVehicleParameter) {
     276        21083 :                 closeFlow();
     277        21076 :                 delete myVehicleParameter;
     278              :                 myParamStack.pop_back();
     279              :             }
     280        21076 :             myVehicleParameter = nullptr;
     281        21076 :             myInsertStopEdgesAt = -1;
     282        21076 :             break;
     283       235871 :         case SUMO_TAG_TRIP:
     284       235871 :             closeTrip();
     285       235858 :             delete myVehicleParameter;
     286       235858 :             myVehicleParameter = nullptr;
     287              :             myParamStack.pop_back();
     288       235858 :             myInsertStopEdgesAt = -1;
     289       235858 :             break;
     290          530 :         case SUMO_TAG_VTYPE_DISTRIBUTION:
     291          530 :             closeVehicleTypeDistribution();
     292          530 :             break;
     293        92323 :         case SUMO_TAG_ROUTE_DISTRIBUTION:
     294        92323 :             closeRouteDistribution();
     295        92321 :             break;
     296        53688 :         case SUMO_TAG_PERSONTRIP:
     297              :         case SUMO_TAG_RIDE:
     298              :         case SUMO_TAG_TRANSPORT:
     299              :         case SUMO_TAG_TRANSHIP:
     300              :         case SUMO_TAG_WALK:
     301        53688 :             if (myParamStack.size() == 2) {
     302              :                 myParamStack.pop_back();
     303              :             }
     304              :             break;
     305           23 :         case SUMO_TAG_INTERVAL:
     306           23 :             myBeginDefault = string2time(OptionsCont::getOptions().getString("begin"));
     307           23 :             myEndDefault = string2time(OptionsCont::getOptions().getString("end"));
     308           23 :             break;
     309              :         default:
     310              :             break;
     311              :     }
     312              :     myElementStack.pop_back();
     313     11267757 : }
     314              : 
     315              : 
     316              : SUMORouteHandler::StopPos
     317       100807 : SUMORouteHandler::checkStopPos(double& startPos, double& endPos, const double laneLength, const double minLength, const bool friendlyPos) {
     318       100807 :     if (minLength > laneLength) {
     319              :         return STOPPOS_INVALID_LANELENGTH;
     320              :     }
     321       100807 :     if (startPos < 0) {
     322          161 :         startPos += laneLength;
     323              :     }
     324       100807 :     if (endPos < 0) {
     325         1497 :         endPos += laneLength;
     326              :     }
     327       100807 :     if ((endPos < minLength) || (endPos > laneLength)) {
     328          113 :         if (!friendlyPos) {
     329              :             return STOPPOS_INVALID_ENDPOS;
     330              :         }
     331          108 :         if (endPos < minLength) {
     332            3 :             endPos = minLength;
     333              :         }
     334          108 :         if (endPos > laneLength) {
     335          105 :             endPos = laneLength;
     336              :         }
     337              :     }
     338       100802 :     if ((startPos < 0) || (startPos > (endPos - minLength))) {
     339           37 :         if (!friendlyPos) {
     340              :             return STOPPOS_INVALID_STARTPOS;
     341              :         }
     342           37 :         if (startPos < 0) {
     343           13 :             startPos = 0;
     344              :         }
     345           37 :         if (startPos > (endPos - minLength)) {
     346           24 :             startPos = endPos - minLength;
     347              :         }
     348              :     }
     349              :     return STOPPOS_VALID;
     350              : }
     351              : 
     352              : 
     353              : SUMOTime
     354        98638 : SUMORouteHandler::getFirstDepart() const {
     355        98638 :     return myFirstDepart;
     356              : }
     357              : 
     358              : 
     359              : SUMOTime
     360      3243709 : SUMORouteHandler::getLastDepart() const {
     361      3243709 :     return myLastDepart;
     362              : }
     363              : 
     364              : 
     365              : void
     366        56792 : SUMORouteHandler::addParam(const SUMOSAXAttributes& attrs) {
     367        56792 :     bool ok = true;
     368        56792 :     const std::string key = attrs.get<std::string>(SUMO_ATTR_KEY, nullptr, ok);
     369              :     // only continue if key isn't empty
     370        56792 :     if (ok && (key.size() > 0)) {
     371              :         // circumventing empty string test
     372        56792 :         std::string val = attrs.hasAttribute(SUMO_ATTR_VALUE) ? attrs.getString(SUMO_ATTR_VALUE) : "";
     373              :         // check special params that must be interpreted as relative paths
     374        47959 :         if ((myVehicleParameter != nullptr || myCurrentVType != nullptr)
     375        63042 :                 && (key == "device.toc.file" || key == "device.ssm.file")) {
     376          786 :             val = FileHelpers::checkForRelativity(val, getFileName());
     377              :         }
     378              :         // add parameter in current created element
     379        56792 :         if (!myParamStack.empty()) {
     380        15083 :             myParamStack.back()->setParameter(key, val);
     381              :         }
     382              :     }
     383        56792 : }
     384              : 
     385              : 
     386              : bool
     387        39947 : SUMORouteHandler::parseStop(SUMOVehicleParameter::Stop& stop, const SUMOSAXAttributes& attrs, std::string errorSuffix, MsgHandler* const errorOutput) {
     388        39947 :     stop.parametersSet = 0;
     389        39947 :     if (attrs.hasAttribute(SUMO_ATTR_ARRIVAL)) {
     390          402 :         stop.parametersSet |= STOP_ARRIVAL_SET;
     391              :     }
     392        39947 :     if (attrs.hasAttribute(SUMO_ATTR_DURATION)) {
     393        21953 :         stop.parametersSet |= STOP_DURATION_SET;
     394              :     }
     395        39947 :     if (attrs.hasAttribute(SUMO_ATTR_UNTIL)) {
     396        19856 :         stop.parametersSet |= STOP_UNTIL_SET;
     397              :     }
     398        39947 :     if (attrs.hasAttribute(SUMO_ATTR_STARTED)) {
     399         6560 :         stop.parametersSet |= STOP_STARTED_SET;
     400              :     }
     401        39947 :     if (attrs.hasAttribute(SUMO_ATTR_ENDED)) {
     402         6087 :         stop.parametersSet |= STOP_ENDED_SET;
     403              :     }
     404        39947 :     if (attrs.hasAttribute(SUMO_ATTR_EXTENSION)) {
     405          464 :         stop.parametersSet |= STOP_EXTENSION_SET;
     406              :     }
     407        39947 :     if (attrs.hasAttribute(SUMO_ATTR_ENDPOS)) {
     408         8726 :         stop.parametersSet |= STOP_END_SET;
     409              :     }
     410        39947 :     if (attrs.hasAttribute(SUMO_ATTR_STARTPOS)) {
     411         1583 :         stop.parametersSet |= STOP_START_SET;
     412              :     }
     413        39947 :     if (attrs.hasAttribute(SUMO_ATTR_POSITION_LAT)) {
     414          125 :         stop.parametersSet |= STOP_POSLAT_SET;
     415              :     }
     416        39947 :     if (attrs.hasAttribute(SUMO_ATTR_TRIGGERED)) {
     417         1475 :         stop.parametersSet |= STOP_TRIGGER_SET;
     418              :     }
     419              :     // legacy attribute
     420        39947 :     if (attrs.hasAttribute(SUMO_ATTR_CONTAINER_TRIGGERED)) {
     421           56 :         stop.parametersSet |= STOP_TRIGGER_SET;
     422              :     }
     423        39947 :     if (attrs.hasAttribute(SUMO_ATTR_PARKING)) {
     424          914 :         stop.parametersSet |= STOP_PARKING_SET;
     425              :     }
     426        39947 :     if (attrs.hasAttribute(SUMO_ATTR_EXPECTED)) {
     427          242 :         stop.parametersSet |= STOP_EXPECTED_SET;
     428              :     }
     429        39947 :     if (attrs.hasAttribute(SUMO_ATTR_PERMITTED)) {
     430          344 :         stop.parametersSet |= STOP_PERMITTED_SET;
     431              :     }
     432        39947 :     if (attrs.hasAttribute(SUMO_ATTR_EXPECTED_CONTAINERS)) {
     433           77 :         stop.parametersSet |= STOP_EXPECTED_CONTAINERS_SET;
     434              :     }
     435        39947 :     if (attrs.hasAttribute(SUMO_ATTR_TRIP_ID)) {
     436          432 :         stop.parametersSet |= STOP_TRIP_ID_SET;
     437              :     }
     438        39947 :     if (attrs.hasAttribute(SUMO_ATTR_SPLIT)) {
     439          356 :         stop.parametersSet |= STOP_SPLIT_SET;
     440              :     }
     441        39947 :     if (attrs.hasAttribute(SUMO_ATTR_JOIN)) {
     442          116 :         stop.parametersSet |= STOP_JOIN_SET;
     443              :     }
     444        39947 :     if (attrs.hasAttribute(SUMO_ATTR_LINE)) {
     445          387 :         stop.parametersSet |= STOP_LINE_SET;
     446              :     }
     447        39947 :     if (attrs.hasAttribute(SUMO_ATTR_SPEED)) {
     448          670 :         stop.parametersSet |= STOP_SPEED_SET;
     449              :     }
     450        39947 :     if (attrs.hasAttribute(SUMO_ATTR_ONDEMAND)) {
     451          178 :         stop.parametersSet |= STOP_ONDEMAND_SET;
     452              :     }
     453        39947 :     if (attrs.hasAttribute(SUMO_ATTR_PRIORITY)) {
     454          300 :         stop.parametersSet |= STOP_PRIORITY_SET;
     455              :     }
     456        39947 :     if (attrs.hasAttribute(SUMO_ATTR_JUMP)) {
     457          545 :         stop.parametersSet |= STOP_JUMP_SET;
     458              :     }
     459        39947 :     if (attrs.hasAttribute(SUMO_ATTR_JUMP_UNTIL)) {
     460            7 :         stop.parametersSet |= STOP_JUMP_UNTIL_SET;
     461              :     }
     462        39947 :     bool ok = true;
     463        79894 :     stop.busstop = attrs.getOpt<std::string>(SUMO_ATTR_BUS_STOP, nullptr, ok, "");
     464        39947 :     stop.busstop = attrs.getOpt<std::string>(SUMO_ATTR_TRAIN_STOP, nullptr, ok, stop.busstop);
     465        39947 :     stop.chargingStation = attrs.getOpt<std::string>(SUMO_ATTR_CHARGING_STATION, nullptr, ok, "");
     466        39947 :     stop.overheadWireSegment = attrs.getOpt<std::string>(SUMO_ATTR_OVERHEAD_WIRE_SEGMENT, nullptr, ok, "");
     467        39947 :     stop.containerstop = attrs.getOpt<std::string>(SUMO_ATTR_CONTAINER_STOP, nullptr, ok, "");
     468        79894 :     stop.parkingarea = attrs.getOpt<std::string>(SUMO_ATTR_PARKING_AREA, nullptr, ok, "");
     469        39947 :     if (stop.busstop != "") {
     470        34302 :         errorSuffix = " at '" + stop.busstop + "'" + errorSuffix;
     471        28513 :     } else if (stop.chargingStation != "") {
     472         3390 :         errorSuffix = " at '" + stop.chargingStation + "'" + errorSuffix;
     473        27383 :     } else if (stop.overheadWireSegment != "") {
     474            0 :         errorSuffix = " at '" + stop.overheadWireSegment + "'" + errorSuffix;
     475        27383 :     } else if (stop.containerstop != "") {
     476         1908 :         errorSuffix = " at '" + stop.containerstop + "'" + errorSuffix;
     477        26747 :     } else if (stop.parkingarea != "") {
     478        46407 :         errorSuffix = " at '" + stop.parkingarea + "'" + errorSuffix;
     479        11278 :     } else if (attrs.hasAttribute(SUMO_ATTR_LANE)) {
     480        28557 :         errorSuffix = " on lane '" + attrs.get<std::string>(SUMO_ATTR_LANE, nullptr, ok) + "'" + errorSuffix;
     481         1759 :     } else if (attrs.hasAttribute(SUMO_ATTR_EDGE)) {
     482         4977 :         errorSuffix = " on edge '" + attrs.get<std::string>(SUMO_ATTR_EDGE, nullptr, ok) + "'" + errorSuffix;
     483              :     } else {
     484          200 :         errorSuffix = " at undefined location" + errorSuffix;
     485              :     }
     486              :     // speed for counting as stopped
     487        39947 :     stop.speed = attrs.getOpt<double>(SUMO_ATTR_SPEED, nullptr, ok, 0);
     488        39947 :     if (stop.speed < 0) {
     489           12 :         errorOutput->inform(TLF("Speed cannot be negative for stop%.", errorSuffix));
     490            6 :         return false;
     491              :     }
     492              : 
     493              :     // get the standing duration
     494        39941 :     bool expectTrigger = !attrs.hasAttribute(SUMO_ATTR_DURATION) && !attrs.hasAttribute(SUMO_ATTR_UNTIL) && !attrs.hasAttribute(SUMO_ATTR_SPEED);
     495        39941 :     std::vector<std::string> triggers = attrs.getOpt<std::vector<std::string> >(SUMO_ATTR_TRIGGERED, nullptr, ok);
     496              :     // legacy
     497        39941 :     if (attrs.getOpt<bool>(SUMO_ATTR_CONTAINER_TRIGGERED, nullptr, ok, false)) {
     498          112 :         triggers.push_back(toString(SUMO_TAG_CONTAINER));
     499              :     }
     500        39941 :     SUMOVehicleParameter::parseStopTriggers(triggers, expectTrigger, stop);
     501        39941 :     stop.arrival = attrs.getOptSUMOTimeReporting(SUMO_ATTR_ARRIVAL, nullptr, ok, -1);
     502        39941 :     stop.duration = attrs.getOptSUMOTimeReporting(SUMO_ATTR_DURATION, nullptr, ok, -1);
     503        39941 :     stop.until = attrs.getOptSUMOTimeReporting(SUMO_ATTR_UNTIL, nullptr, ok, -1);
     504        39941 :     if (!expectTrigger && (!ok || (stop.duration < 0 && stop.until < 0 && stop.speed == 0))) {
     505           36 :         errorOutput->inform(TLF("Invalid duration or end time is given for a stop%.", errorSuffix));
     506           12 :         return false;
     507              :     }
     508        39929 :     if (triggers.size() > 0 && stop.speed > 0) {
     509            9 :         errorOutput->inform(TLF("Triggers and waypoints cannot be combined%.", errorSuffix));
     510            3 :         return false;
     511              :     }
     512        39926 :     stop.extension = attrs.getOptSUMOTimeReporting(SUMO_ATTR_EXTENSION, nullptr, ok, -1);
     513        39926 :     const bool defaultParking = (stop.triggered || stop.containerTriggered || stop.parkingarea != "");
     514        39926 :     stop.parking = attrs.getOpt<ParkingType>(SUMO_ATTR_PARKING, nullptr, ok, defaultParking ? ParkingType::OFFROAD : ParkingType::ONROAD);
     515        39926 :     if ((stop.parkingarea != "") && (stop.parking == ParkingType::ONROAD)) {
     516           21 :         WRITE_WARNINGF(TL("Stop at parkingArea overrides attribute 'parking' for stop%."), errorSuffix);
     517            7 :         stop.parking = ParkingType::OFFROAD;
     518              :     }
     519              : 
     520              :     // expected persons
     521        39926 :     const std::vector<std::string>& expected = attrs.getOpt<std::vector<std::string> >(SUMO_ATTR_EXPECTED, nullptr, ok);
     522              :     stop.awaitedPersons.insert(expected.begin(), expected.end());
     523        39926 :     if (stop.awaitedPersons.size() > 0) {
     524          242 :         stop.triggered = true;
     525          242 :         if ((stop.parametersSet & STOP_PARKING_SET) == 0) {
     526          123 :             stop.parking = ParkingType::OFFROAD;
     527              :         }
     528              :     }
     529              : 
     530              :     // permitted transportables
     531        39926 :     const std::vector<std::string>& permitted = attrs.getOpt<std::vector<std::string> >(SUMO_ATTR_PERMITTED, nullptr, ok);
     532              :     stop.permitted.insert(permitted.begin(), permitted.end());
     533              : 
     534              :     // expected containers
     535        39926 :     const std::vector<std::string>& expectedContainers = attrs.getOpt<std::vector<std::string> >(SUMO_ATTR_EXPECTED_CONTAINERS, nullptr, ok);
     536              :     stop.awaitedContainers.insert(expectedContainers.begin(), expectedContainers.end());
     537        39926 :     if (stop.awaitedContainers.size() > 0) {
     538           77 :         stop.containerTriggered = true;
     539           77 :         if ((stop.parametersSet & STOP_PARKING_SET) == 0) {
     540           67 :             stop.parking = ParkingType::OFFROAD;
     541              :         }
     542              :     }
     543              :     // public transport trip id
     544        39926 :     stop.tripId = attrs.getOpt<std::string>(SUMO_ATTR_TRIP_ID, nullptr, ok, "");
     545        39926 :     stop.split = attrs.getOpt<std::string>(SUMO_ATTR_SPLIT, nullptr, ok, "");
     546        39926 :     stop.join = attrs.getOpt<std::string>(SUMO_ATTR_JOIN, nullptr, ok, "");
     547        39926 :     stop.line = attrs.getOpt<std::string>(SUMO_ATTR_LINE, nullptr, ok, "");
     548              : 
     549        39926 :     const std::string idx = attrs.getOpt<std::string>(SUMO_ATTR_INDEX, nullptr, ok, "end");
     550        39926 :     if (idx == "end") {
     551        39708 :         stop.index = STOP_INDEX_END;
     552          218 :     } else if (idx == "fit") {
     553           14 :         stop.index = STOP_INDEX_FIT;
     554              :     } else {
     555          204 :         stop.index = attrs.get<int>(SUMO_ATTR_INDEX, nullptr, ok);
     556          204 :         if (!ok || stop.index < 0) {
     557           18 :             errorOutput->inform(TLF("Invalid 'index' for stop%.", errorSuffix));
     558            6 :             return false;
     559              :         }
     560              :     }
     561        39920 :     stop.started = attrs.getOptSUMOTimeReporting(SUMO_ATTR_STARTED, nullptr, ok, -1);
     562        39920 :     stop.ended = attrs.getOptSUMOTimeReporting(SUMO_ATTR_ENDED, nullptr, ok, -1);
     563        39920 :     stop.posLat = attrs.getOpt<double>(SUMO_ATTR_POSITION_LAT, nullptr, ok, INVALID_DOUBLE);
     564        39920 :     stop.actType = attrs.getOpt<std::string>(SUMO_ATTR_ACTTYPE, nullptr, ok, "");
     565        39920 :     stop.onDemand = attrs.getOpt<bool>(SUMO_ATTR_ONDEMAND, nullptr, ok, false);
     566        39920 :     stop.priority = attrs.getOpt<double>(SUMO_ATTR_PRIORITY, nullptr, ok, -1);
     567        39920 :     stop.jump = attrs.getOptSUMOTimeReporting(SUMO_ATTR_JUMP, nullptr, ok, -1);
     568        39920 :     stop.jumpUntil = attrs.getOptSUMOTimeReporting(SUMO_ATTR_JUMP_UNTIL, nullptr, ok, -1);
     569        39920 :     stop.collision = attrs.getOpt<bool>(SUMO_ATTR_COLLISION, nullptr, ok, false);
     570        39920 :     return true;
     571        39941 : }
     572              : 
     573              : 
     574              : /****************************************************************************/
        

Generated by: LCOV version 2.0-1