#include <config.h>
#ifdef HAVE_VERSION_H
#include <version.h>
#endif
#include <iostream>
#include <string>
#include <limits.h>
#include <ctime>
#include <memory>
#include <xercesc/sax/SAXException.hpp>
#include <xercesc/sax/SAXParseException.hpp>
#include <utils/common/StringUtils.h>
#include <utils/common/MsgHandler.h>
#include <utils/common/UtilExceptions.h>
#include <utils/common/SystemFrame.h>
#include <utils/common/RandHelper.h>
#include <utils/common/ToString.h>
#include <utils/vehicle/SUMORouteHandler.h>
#ifdef HAVE_FOX
#include <utils/foxtools/MsgHandlerSynchronized.h>
#endif
#include <utils/iodevices/OutputDevice.h>
#include <utils/options/Option.h>
#include <utils/options/OptionsCont.h>
#include <utils/options/OptionsIO.h>
#include <utils/router/DijkstraRouter.h>
#include <utils/router/AFRouter.h>
#include <utils/router/AStarRouter.h>
#include <utils/router/CHRouter.h>
#include <utils/router/CHRouterWrapper.h>
#include <utils/vehicle/SUMOVehicleParserHelper.h>
#include <utils/xml/XMLSubSys.h>
#include <router/ROFrame.h>
#include <router/ROLoader.h>
#include <router/RONet.h>
#include <router/ROEdge.h>
#include "RODUAEdgeBuilder.h"
#include "RODUAFrame.h"
void
initNet(RONet& net, ROLoader& loader, OptionsCont& oc) {
RODUAEdgeBuilder builder;
ROEdge::setGlobalOptions(oc.getBool("weights.interpolate"));
loader.loadNet(net, builder);
if (oc.isSet("weight-files")) {
loader.loadWeights(net, "weight-files", oc.getString("weight-attribute"), false, oc.getBool("weights.expand"));
}
if (oc.isSet("lane-weight-files")) {
loader.loadWeights(net, "lane-weight-files", oc.getString("weight-attribute"), true, oc.getBool("weights.expand"));
}
}
void
computeRoutes(RONet& net, ROLoader& loader, OptionsCont& oc) {
loader.openRoutes(net);
auto ttFunction = gWeightsRandomFactor > 1 ? &ROEdge::getTravelTimeStaticRandomized : &ROEdge::getTravelTimeStatic;
SUMOAbstractRouter<ROEdge, ROVehicle>* router;
const std::string measure = oc.getString("weight-attribute");
const std::string routingAlgorithm = oc.getString("routing-algorithm");
const double priorityFactor = oc.getFloat("weights.priority-factor");
const SUMOTime begin = string2time(oc.getString("begin"));
const SUMOTime end = oc.isDefault("end") ? SUMOTime_MAX : string2time(oc.getString("end"));
DijkstraRouter<ROEdge, ROVehicle>::Operation op = &ROEdge::getTravelTimeStatic;
if (oc.isSet("restriction-params") &&
(routingAlgorithm == "CH" || routingAlgorithm == "CHWrapper")) {
throw ProcessError(TLF("Routing algorithm '%' does not support restriction-params", routingAlgorithm));
}
if (measure == "traveltime" && priorityFactor == 0) {
if (routingAlgorithm == "dijkstra") {
router = new DijkstraRouter<ROEdge, ROVehicle>(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttFunction, nullptr, false, nullptr, net.hasPermissions(), oc.isSet("restriction-params"));
} else if (routingAlgorithm == "astar") {
typedef AStarRouter<ROEdge, ROVehicle, ROMapMatcher> AStar;
std::shared_ptr<const AStar::LookupTable> lookup;
if (oc.isSet("astar.all-distances")) {
lookup = std::make_shared<const AStar::FLT>(oc.getString("astar.all-distances"), (int)ROEdge::getAllEdges().size());
} else if (oc.isSet("astar.landmark-distances")) {
DijkstraRouter<ROEdge, ROVehicle> forward(ROEdge::getAllEdges(), true, &ROEdge::getTravelTimeStatic);
std::vector<ReversedEdge<ROEdge, ROVehicle>*> reversed;
for (ROEdge* edge : ROEdge::getAllEdges()) {
reversed.push_back(edge->getReversedRoutingEdge());
}
for (ReversedEdge<ROEdge, ROVehicle>* redge : reversed) {
redge->init();
}
DijkstraRouter<ReversedEdge<ROEdge, ROVehicle>, ROVehicle> backward(reversed, true, &ReversedEdge<ROEdge, ROVehicle>::getTravelTimeStatic);
ROVehicle defaultVehicle(SUMOVehicleParameter(), nullptr, net.getVehicleTypeSecure(DEFAULT_VTYPE_ID), &net);
ROMapMatcher* mapMatcher = dynamic_cast<ROMapMatcher*>(loader.getRouteHandler());
lookup = std::make_shared<const AStar::LMLT>(oc.getString("astar.landmark-distances"), ROEdge::getAllEdges(), &forward, &backward, &defaultVehicle,
oc.isSet("astar.save-landmark-distances") ? oc.getString("astar.save-landmark-distances") : "", oc.getInt("routing-threads"), mapMatcher);
}
router = new AStar(ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttFunction, lookup, net.hasPermissions(), oc.isSet("restriction-params"));
} else if (routingAlgorithm == "CH" && !net.hasPermissions()) {
const SUMOTime weightPeriod = (oc.isSet("weight-files") ?
string2time(oc.getString("weight-period")) :
SUMOTime_MAX);
router = new CHRouter<ROEdge, ROVehicle>(
ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttFunction, SVC_IGNORING, weightPeriod, net.hasPermissions(), oc.isSet("restriction-params"));
} else if (routingAlgorithm == "CHWrapper" || routingAlgorithm == "CH") {
const SUMOTime weightPeriod = (oc.isSet("weight-files") ?
string2time(oc.getString("weight-period")) :
SUMOTime_MAX);
router = new CHRouterWrapper<ROEdge, ROVehicle>(
ROEdge::getAllEdges(), oc.getBool("ignore-errors"), ttFunction,
begin, end, weightPeriod, net.hasPermissions(), oc.getInt("routing-threads"));
} else if (routingAlgorithm == "arcflag") {
constexpr auto NUMBER_OF_LEVELS = 5;
ROVehicle defaultVehicle(SUMOVehicleParameter(), nullptr, net.getVehicleTypeSecure(DEFAULT_VTYPE_ID), &net);
KDTreePartition<ROEdge, RONode, ROVehicle>* partition = new KDTreePartition<ROEdge, RONode, ROVehicle>(NUMBER_OF_LEVELS, ROEdge::getAllEdges(), net.hasPermissions(), oc.isSet("restriction-params"));
partition->init(&defaultVehicle);
auto reversedTtFunction = gWeightsRandomFactor > 1 ? &FlippedEdge<ROEdge, RONode, ROVehicle>::getTravelTimeStaticRandomized : &FlippedEdge<ROEdge, RONode, ROVehicle>::getTravelTimeStatic;
router = new AFRouter<ROEdge, RONode, ROVehicle, ROMapMatcher>(ROEdge::getAllEdges(),
partition, oc.getBool("ignore-errors"), ttFunction, reversedTtFunction, (oc.isSet("weight-files") ? string2time(oc.getString("weight-period")) : SUMOTime_MAX),
nullptr, nullptr, net.hasPermissions(), oc.isSet("restriction-params"));
} else {
throw ProcessError(TLF("Unknown routing Algorithm '%'!", routingAlgorithm));
}
} else {
if (measure == "traveltime") {
if (ROEdge::initPriorityFactor(priorityFactor)) {
op = &ROEdge::getTravelTimeStaticPriorityFactor;
}
} else if (measure == "CO") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::CO>;
} else if (measure == "CO2") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::CO2>;
} else if (measure == "PMx") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::PM_X>;
} else if (measure == "HC") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::HC>;
} else if (measure == "NOx") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::NO_X>;
} else if (measure == "fuel") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::FUEL>;
} else if (measure == "electricity") {
op = &ROEdge::getEmissionEffort<PollutantsInterface::ELEC>;
} else if (measure == "noise") {
op = &ROEdge::getNoiseEffort;
} else {
op = &ROEdge::getStoredEffort;
}
if (measure != "traveltime" && !net.hasLoadedEffort()) {
WRITE_WARNINGF(TL("No weight data was loaded for attribute '%'."), measure);
}
router = new DijkstraRouter<ROEdge, ROVehicle>(
ROEdge::getAllEdges(), oc.getBool("ignore-errors"), op, ttFunction, false, nullptr, net.hasPermissions(), oc.isSet("restriction-params"));
}
const int carWalk = SUMOVehicleParserHelper::parseCarWalkTransfer(oc);
double taxiWait = STEPS2TIME(string2time(OptionsCont::getOptions().getString("persontrip.taxi.waiting-time")));
RailwayRouter<ROEdge, ROVehicle>* railRouter = nullptr;
if (net.hasBidiEdges()) {
railRouter = new RailwayRouter<ROEdge, ROVehicle>(ROEdge::getAllEdges(), true, op, ttFunction, false, net.hasPermissions(),
oc.isSet("restriction-params"),
oc.getFloat("railway.max-train-length"));
}
RORouterProvider provider(router, new PedestrianRouter<ROEdge, ROLane, RONode, ROVehicle>(),
new ROIntermodalRouter(RONet::adaptIntermodalRouter, carWalk, taxiWait, routingAlgorithm),
railRouter);
try {
net.openOutput(oc);
loader.processRoutes(begin, end, string2time(oc.getString("route-steps")), net, provider);
net.writeIntermodal(oc, provider.getIntermodalRouter());
net.cleanup();
} catch (ProcessError&) {
net.cleanup();
throw;
}
}
int
main(int argc, char** argv) {
OptionsCont& oc = OptionsCont::getOptions();
oc.setApplicationDescription(TL("Shortest path router and DUE computer for the microscopic, multi-modal traffic simulation SUMO."));
oc.setApplicationName("duarouter", "Eclipse SUMO duarouter " VERSION_STRING);
int ret = 0;
RONet* net = nullptr;
try {
XMLSubSys::init();
RODUAFrame::fillOptions();
OptionsIO::setArgs(argc, argv);
OptionsIO::getOptions();
if (oc.processMetaOptions(argc < 2)) {
SystemFrame::close();
return 0;
}
SystemFrame::checkOptions(oc);
XMLSubSys::setValidation(oc.getString("xml-validation"), oc.getString("xml-validation.net"), oc.getString("xml-validation.routes"));
#ifdef HAVE_FOX
if (oc.getInt("routing-threads") > 1) {
MsgHandler::setFactory(&MsgHandlerSynchronized::create);
}
#endif
MsgHandler::initOutputOptions();
if (!RODUAFrame::checkOptions()) {
throw ProcessError();
}
RandHelper::initRandGlobal();
ROLoader loader(oc, false, !oc.getBool("no-step-log"));
net = new RONet();
initNet(*net, loader, oc);
try {
computeRoutes(*net, loader, oc);
} catch (XERCES_CPP_NAMESPACE::SAXParseException& e) {
WRITE_ERROR(toString(e.getLineNumber()));
ret = 1;
} catch (XERCES_CPP_NAMESPACE::SAXException& e) {
WRITE_ERROR(StringUtils::transcode(e.getMessage()));
ret = 1;
}
if (MsgHandler::getErrorInstance()->wasInformed() || ret != 0) {
throw ProcessError();
}
} catch (const ProcessError& e) {
if (std::string(e.what()) != std::string("Process Error") && std::string(e.what()) != std::string("")) {
WRITE_ERROR(e.what());
}
MsgHandler::getErrorInstance()->inform("Quitting (on error).", false);
ret = 1;
#ifndef _DEBUG
} catch (const std::exception& e) {
if (std::string(e.what()) != std::string("")) {
WRITE_ERROR(e.what());
}
MsgHandler::getErrorInstance()->inform("Quitting (on error).", false);
ret = 1;
} catch (...) {
MsgHandler::getErrorInstance()->inform("Quitting (on unknown error).", false);
ret = 1;
#endif
}
delete net;
SystemFrame::close();
if (ret == 0) {
std::cout << "Success." << std::endl;
}
return ret;
}