#include <config.h>
#ifdef HAVE_VERSION_H
#include <version.h>
#endif
#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstdlib>
#include <string>
#include <utils/options/Option.h>
#include <utils/options/OptionsCont.h>
#include <utils/options/OptionsIO.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/xml/XMLSubSys.h>
#include <utils/common/StringUtils.h>
#include <od/ODDistrictCont.h>
#include <od/ODDistrictHandler.h>
#include <od/ODMatrix.h>
#include <utils/common/StringUtils.h>
#include <utils/common/SUMOTime.h>
#include <utils/common/StringTokenizer.h>
#include <utils/common/FileHelpers.h>
#include <utils/vehicle/SUMOVehicleParameter.h>
#include <utils/importio/LineReader.h>
#include <utils/iodevices/OutputDevice.h>
void
fillOptions() {
OptionsCont& oc = OptionsCont::getOptions();
oc.addCallExample("-c <CONFIGURATION>", "run with configuration file");
SystemFrame::addConfigurationOptions(oc);
oc.addOptionSubTopic("Input");
oc.addOptionSubTopic("Output");
oc.addOptionSubTopic("Time");
oc.addOptionSubTopic("Processing");
oc.addOptionSubTopic("Defaults");
SystemFrame::addReportOptions(oc);
oc.doRegister("taz-files", 'n', new Option_FileName());
oc.addSynonyme("taz-files", "taz");
oc.addSynonyme("taz-files", "net-file");
oc.addSynonyme("taz-files", "net");
oc.addDescription("taz-files", "Input", TL("Loads TAZ (districts; also from networks) from FILE(s)"));
oc.doRegister("od-matrix-files", 'd', new Option_FileName());
oc.addSynonyme("od-matrix-files", "od-files");
oc.addSynonyme("od-matrix-files", "od");
oc.addDescription("od-matrix-files", "Input", TL("Loads O/D-files from FILE(s)"));
oc.doRegister("od-amitran-files", new Option_FileName());
oc.addSynonyme("od-amitran-files", "amitran-files");
oc.addSynonyme("od-amitran-files", "amitran");
oc.addDescription("od-amitran-files", "Input", TL("Loads O/D-matrix in Amitran format from FILE(s)"));
oc.doRegister("tazrelation-files", 'z', new Option_FileName());
oc.addDescription("tazrelation-files", "Input", TL("Loads O/D-matrix in tazRelation format from FILE(s)"));
oc.doRegister("tazrelation-attribute", new Option_String("count"));
oc.addSynonyme("tazrelation-attribute", "attribute");
oc.addDescription("tazrelation-attribute", "Input", TL("Define data attribute for loading counts (default 'count')"));
oc.doRegister("output-file", 'o', new Option_FileName());
oc.addSynonyme("output-file", "output", true);
oc.addDescription("output-file", "Output", TL("Writes trip definitions into FILE"));
oc.doRegister("flow-output", new Option_FileName());
oc.addDescription("flow-output", "Output", TL("Writes flow definitions into FILE"));
oc.doRegister("flow-output.probability", new Option_Bool(false));
oc.addDescription("flow-output.probability", "Output", TL("Writes probabilistic flow instead of evenly spaced flow"));
oc.doRegister("pedestrians", new Option_Bool(false));
oc.addDescription("pedestrians", "Output", TL("Writes pedestrians instead of vehicles"));
oc.doRegister("persontrips", new Option_Bool(false));
oc.addDescription("persontrips", "Output", TL("Writes persontrips instead of vehicles"));
oc.doRegister("persontrips.modes", new Option_StringVector());
oc.addDescription("persontrips.modes", "Output", TL("Add modes attribute to personTrips"));
oc.doRegister("ignore-vehicle-type", new Option_Bool(false));
oc.addSynonyme("ignore-vehicle-type", "no-vtype", true);
oc.addDescription("ignore-vehicle-type", "Output", TL("Does not save vtype information"));
oc.doRegister("junctions", new Option_Bool(false));
oc.addDescription("junctions", "Output", TL("Writes trips between junctions"));
oc.doRegister("begin", 'b', new Option_String("0", "TIME"));
oc.addDescription("begin", "Time", TL("Defines the begin time; Previous trips will be discarded"));
oc.doRegister("end", 'e', new Option_String("-1", "TIME"));
oc.addDescription("end", "Time", TL("Defines the end time; Later trips will be discarded; Defaults to the maximum time that SUMO can represent"));
oc.doRegister("scale", 's', new Option_Float(1));
oc.addDescription("scale", "Processing", TL("Scales the loaded flows by FLOAT"));
oc.doRegister("spread.uniform", new Option_Bool(false));
oc.addDescription("spread.uniform", "Processing", TL("Spreads trips uniformly over each time period"));
oc.doRegister("different-source-sink", new Option_Bool(false));
oc.addDescription("different-source-sink", "Processing", TL("Always choose source and sink edge which are not identical"));
oc.doRegister("vtype", new Option_String(""));
oc.addDescription("vtype", "Processing", TL("Defines the name of the vehicle type to use"));
oc.doRegister("prefix", new Option_String(""));
oc.addDescription("prefix", "Processing", TL("Defines the prefix for vehicle names"));
oc.doRegister("timeline", new Option_StringVector());
oc.addDescription("timeline", "Processing", TL("Uses STR[] as a timeline definition"));
oc.doRegister("timeline.day-in-hours", new Option_Bool(false));
oc.addDescription("timeline.day-in-hours", "Processing", TL("Uses STR as a 24h-timeline definition"));
oc.doRegister("ignore-errors", new Option_Bool(false));
oc.addSynonyme("ignore-errors", "dismiss-loading-errors", true);
oc.addDescription("ignore-errors", "Report", TL("Continue on broken input"));
oc.doRegister("no-step-log", new Option_Bool(false));
oc.addDescription("no-step-log", "Processing", TL("Disable console output of current time step"));
oc.doRegister("departlane", new Option_String("free"));
oc.addDescription("departlane", "Defaults", TL("Assigns a default depart lane"));
oc.doRegister("departpos", new Option_String());
oc.addDescription("departpos", "Defaults", TL("Assigns a default depart position"));
oc.doRegister("departspeed", new Option_String("max"));
oc.addDescription("departspeed", "Defaults", TL("Assigns a default depart speed"));
oc.doRegister("arrivallane", new Option_String());
oc.addDescription("arrivallane", "Defaults", TL("Assigns a default arrival lane"));
oc.doRegister("arrivalpos", new Option_String());
oc.addDescription("arrivalpos", "Defaults", TL("Assigns a default arrival position"));
oc.doRegister("arrivalspeed", new Option_String());
oc.addDescription("arrivalspeed", "Defaults", TL("Assigns a default arrival speed"));
RandHelper::insertRandOptions(oc);
}
bool
checkOptions() {
OptionsCont& oc = OptionsCont::getOptions();
bool ok = true;
if (!oc.isSet("taz-files")) {
WRITE_ERROR(TL("No TAZ input file (-n) specified."));
ok = false;
}
if (!oc.isSet("od-matrix-files") && !oc.isSet("od-amitran-files") && !oc.isSet("tazrelation-files")) {
WRITE_ERROR(TL("No input specified."));
ok = false;
}
if (!oc.isSet("output-file") && !oc.isSet("flow-output")) {
WRITE_ERROR(TL("No trip table output file (-o) or flow-output is specified."));
ok = false;
}
if (oc.getBool("pedestrians") && oc.getBool("persontrips")) {
WRITE_ERROR(TL("Only one of the Options 'pedestrians' and 'persontrips' may be set."));
ok = false;
}
SUMOVehicleParameter p;
std::string error;
if (oc.isSet("departlane") && !SUMOVehicleParameter::parseDepartLane(oc.getString("departlane"), "option", "departlane", p.departLane, p.departLaneProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
if (oc.isSet("departpos") && !SUMOVehicleParameter::parseDepartPos(oc.getString("departpos"), "option", "departpos", p.departPos, p.departPosProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
if (oc.isSet("departspeed") && !SUMOVehicleParameter::parseDepartSpeed(oc.getString("departspeed"), "option", "departspeed", p.departSpeed, p.departSpeedProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
if (oc.isSet("arrivallane") && !SUMOVehicleParameter::parseArrivalLane(oc.getString("arrivallane"), "option", "arrivallane", p.arrivalLane, p.arrivalLaneProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
if (oc.isSet("arrivalpos") && !SUMOVehicleParameter::parseArrivalPos(oc.getString("arrivalpos"), "option", "arrivalpos", p.arrivalPos, p.arrivalPosProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
if (oc.isSet("arrivalspeed") && !SUMOVehicleParameter::parseArrivalSpeed(oc.getString("arrivalspeed"), "option", "arrivalspeed", p.arrivalSpeed, p.arrivalSpeedProcedure, error)) {
WRITE_ERROR(error);
ok = false;
}
ok &= SystemFrame::checkOptions(oc);
return ok;
}
int
main(int argc, char** argv) {
OptionsCont& oc = OptionsCont::getOptions();
oc.setApplicationDescription(TL("Importer of O/D-matrices for the microscopic, multi-modal traffic simulation SUMO."));
oc.setApplicationName("od2trips", "Eclipse SUMO od2trips " VERSION_STRING);
int ret = 0;
try {
XMLSubSys::init();
fillOptions();
OptionsIO::setArgs(argc, argv);
OptionsIO::getOptions();
if (oc.processMetaOptions(argc < 2)) {
SystemFrame::close();
return 0;
}
XMLSubSys::setValidation(oc.getString("xml-validation"), "never", "never");
MsgHandler::initOutputOptions();
if (!checkOptions()) {
throw ProcessError();
}
RandHelper::initRandGlobal();
if (!oc.isSet("taz-files")) {
throw ProcessError(TL("You must supply a TAZ, network or districts file ('-n')."));
}
ODDistrictCont districts;
districts.loadDistricts(oc.getStringVector("taz-files"));
if (districts.size() == 0) {
throw ProcessError(TL("No districts loaded."));
}
ODMatrix matrix(districts, oc.getFloat("scale"));
matrix.loadMatrix(oc);
if (matrix.getNumLoaded() == 0) {
throw ProcessError(TL("No vehicles loaded."));
}
if (MsgHandler::getErrorInstance()->wasInformed() && !oc.getBool("ignore-errors")) {
throw ProcessError(TL("Loading failed."));
}
WRITE_MESSAGE(toString(matrix.getNumLoaded()) + " vehicles loaded.");
if (oc.isSet("timeline")) {
matrix.applyCurve(matrix.parseTimeLine(oc.getStringVector("timeline"), oc.getBool("timeline.day-in-hours")));
}
const std::string modes = toString(oc.getStringVector("persontrips.modes"));
bool haveOutput = false;
const SUMOTime end = oc.isDefault("end") ? SUMOTime_MAX : string2time(oc.getString("end"));
if (OutputDevice::createDeviceByOption("output-file", "routes", "routes_file.xsd")) {
matrix.write(string2time(oc.getString("begin")), end,
OutputDevice::getDeviceByOption("output-file"),
oc.getBool("spread.uniform"), oc.getBool("different-source-sink"),
oc.getBool("ignore-vehicle-type"),
oc.getString("prefix"), !oc.getBool("no-step-log"),
oc.getBool("pedestrians"),
oc.getBool("persontrips"), modes);
haveOutput = true;
}
if (OutputDevice::createDeviceByOption("flow-output", "routes", "routes_file.xsd")) {
matrix.writeFlows(string2time(oc.getString("begin")), end,
OutputDevice::getDeviceByOption("flow-output"),
oc.getBool("ignore-vehicle-type"), oc.getString("prefix"),
oc.getBool("flow-output.probability"), oc.getBool("pedestrians"),
oc.getBool("persontrips"), modes);
haveOutput = true;
}
if (!haveOutput) {
throw ProcessError(TL("No output file given."));
}
WRITE_MESSAGE(toString(matrix.getNumDiscarded()) + " vehicles discarded.");
WRITE_MESSAGE(toString(matrix.getNumWritten()) + " vehicles written.");
} 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
}
SystemFrame::close();
if (ret == 0) {
std::cout << "Success." << std::endl;
}
return ret;
}