#include <config.h>
#ifdef HAVE_VERSION_H
#include <version.h>
#endif
#include <iostream>
#include <fstream>
#include <string>
#include <ctime>
#include <netgen/NGNet.h>
#include <netgen/NGRandomNetBuilder.h>
#include <netgen/NGFrame.h>
#include <netbuild/NBNetBuilder.h>
#include <netbuild/NBFrame.h>
#include <netwrite/NWFrame.h>
#include <netimport/NITypeLoader.h>
#include <netimport/NIXMLTypesHandler.h>
#include <utils/options/OptionsCont.h>
#include <utils/options/OptionsIO.h>
#include <utils/options/Option.h>
#include <utils/common/MsgHandler.h>
#include <utils/common/SystemFrame.h>
#include <utils/common/UtilExceptions.h>
#include <utils/common/RandHelper.h>
#include <utils/common/ToString.h>
#include <utils/distribution/RandomDistributor.h>
#include <utils/geom/GeoConvHelper.h>
#include <utils/xml/XMLSubSys.h>
#include <utils/iodevices/OutputDevice.h>
void
fillOptions() {
OptionsCont& oc = OptionsCont::getOptions();
oc.addCallExample("-c <CONFIGURATION>", "create net from given configuration");
oc.addCallExample("--grid [grid-network options] -o <OUTPUTFILE>", "create grid net");
oc.addCallExample("--spider [spider-network options] -o <OUTPUTFILE>", "create spider net");
oc.addCallExample("--rand [random-network options] -o <OUTPUTFILE>", "create random net");
oc.setAdditionalHelpMessage(" Either \"--grid\", \"--spider\" or \"--rand\" must be supplied.\n In dependence to these switches other options are used.");
SystemFrame::addConfigurationOptions(oc);
oc.addOptionSubTopic("Grid Network");
oc.addOptionSubTopic("Spider Network");
oc.addOptionSubTopic("Random Network");
oc.addOptionSubTopic("Input");
oc.addOptionSubTopic("Output");
oc.addOptionSubTopic("Processing");
oc.addOptionSubTopic("Building Defaults");
oc.addOptionSubTopic("TLS Building");
oc.addOptionSubTopic("Edge Removal");
oc.addOptionSubTopic("Unregulated Nodes");
oc.addOptionSubTopic("Junctions");
oc.addOptionSubTopic("Pedestrian");
oc.addOptionSubTopic("Bicycle");
SystemFrame::addReportOptions(oc);
NGFrame::fillOptions();
NBFrame::fillOptions(oc, true);
NWFrame::fillOptions(oc, true);
oc.doRegister("default-junction-type", 'j', new Option_String());
oc.addSynonyme("default-junction-type", "junctions");
oc.addDescription("default-junction-type", "Building Defaults", TL("[traffic_light|priority|right_before_left|left_before_right|traffic_light_right_on_red|priority_stop|allway_stop|...] Determines junction type (see wiki/Networks/PlainXML#Node_types)"));
RandHelper::insertRandOptions(oc);
oc.doRegister("tls.discard-simple", new Option_Bool(false));
oc.addDescription("tls.discard-simple", "TLS Building", "Does not instantiate traffic lights at geometry-like nodes");
oc.doRegister("railway.signal.permit-unsignalized", new Option_StringVector({"tram", "cable_car"}));
oc.addDescription("railway.signal.permit-unsignalized", "TLS Building", TL("List rail classes that may run without rail signals"));
}
bool
checkOptions() {
OptionsCont& oc = OptionsCont::getOptions();
bool ok = NGFrame::checkOptions();
ok &= NBFrame::checkOptions(oc);
ok &= NWFrame::checkOptions(oc);
ok &= SystemFrame::checkOptions(oc);
return ok;
}
NGNet*
buildNetwork(NBNetBuilder& nb) {
OptionsCont& oc = OptionsCont::getOptions();
const double laneWidth = oc.isDefault("default.lanewidth") ? SUMO_const_laneWidth : oc.getFloat("default.lanewidth");
double minLength = (oc.getInt("default.lanenumber") + oc.getInt("turn-lanes")) * 2 * laneWidth + oc.getFloat("default.junctions.radius") * 2 + POSITION_EPS;
if (oc.getBool("spider")) {
bool hadError = false;
if (oc.getInt("spider.arm-number") < 3) {
WRITE_ERROR(TL("Spider networks need at least 3 arms."));
hadError = true;
}
if (oc.getInt("spider.arm-number") > 4 && !oc.getBool("spider.omit-center")) {
WRITE_WARNING(TL("Spider networks with many arms should use option spider.omit-center"));
}
if (oc.getInt("spider.circle-number") < 1) {
WRITE_ERROR(TL("Spider networks need at least one circle."));
hadError = true;
}
minLength = MAX2(minLength, laneWidth * 2 * MAX2(oc.getInt("spider.arm-number"), 3) / (2 * M_PI));
if (oc.getFloat("spider.space-radius") < POSITION_EPS) {
WRITE_ERROR("The radius of spider networks must be at least " + toString(POSITION_EPS));
hadError = true;
} else if (oc.getFloat("spider.space-radius") < minLength) {
WRITE_WARNINGF(TL("The radius of spider networks should be at least % for the given lanenumber, lanewidth and junction radius"), toString(minLength));
}
if (hadError) {
throw ProcessError();
}
NGNet* net = new NGNet(nb);
net->createSpiderWeb(oc.getInt("spider.arm-number"), oc.getInt("spider.circle-number"),
oc.getFloat("spider.space-radius"), !oc.getBool("spider.omit-center"),
oc.getFloat("spider.attach-length"));
return net;
}
if (oc.getBool("grid")) {
int xNo = oc.getInt("grid.x-number");
int yNo = oc.getInt("grid.y-number");
double xLength = oc.getFloat("grid.x-length");
double yLength = oc.getFloat("grid.y-length");
double xAttachLength = oc.getFloat("grid.x-attach-length");
double yAttachLength = oc.getFloat("grid.y-attach-length");
if (oc.isDefault("grid.x-number") && !oc.isDefault("grid.number")) {
xNo = oc.getInt("grid.number");
}
if (oc.isDefault("grid.y-number") && !oc.isDefault("grid.number")) {
yNo = oc.getInt("grid.number");
}
if (oc.isDefault("grid.x-length") && !oc.isDefault("grid.length")) {
xLength = oc.getFloat("grid.length");
}
if (oc.isDefault("grid.y-length") && !oc.isDefault("grid.length")) {
yLength = oc.getFloat("grid.length");
}
if (oc.isDefault("grid.x-attach-length") && !oc.isDefault("grid.attach-length")) {
xAttachLength = oc.getFloat("grid.attach-length");
}
if (oc.isDefault("grid.y-attach-length") && !oc.isDefault("grid.attach-length")) {
yAttachLength = oc.getFloat("grid.attach-length");
}
bool hadError = false;
if (xNo < 1 || yNo < 1 || (xAttachLength == 0 && yAttachLength == 0 && (xNo < 2 && yNo < 2))) {
WRITE_ERROR(TL("The number of nodes must be positive and at least 2 in one direction if there are no attachments."));
hadError = true;
}
const double minAttachLength = minLength / 2 + POSITION_EPS / 2;
if (xLength < POSITION_EPS || yLength < POSITION_EPS) {
WRITE_ERROR("The distance between nodes must be at least " + toString(POSITION_EPS));
hadError = true;
} else if (xLength < minLength || yLength < minLength) {
WRITE_WARNINGF(TL("The distance between nodes should be at least % for the given lanenumber, lanewidth and junction radius"), toString(minLength));
}
if (xAttachLength != 0.0 && xAttachLength < POSITION_EPS) {
WRITE_ERROR("The length of attached streets must be at least " + toString(POSITION_EPS));
hadError = true;
} else if (xAttachLength != 0.0 && xAttachLength < minAttachLength) {
WRITE_WARNINGF(TL("The length of attached streets should be at least % for the given lanenumber, lanewidth and junction radius"), toString(minAttachLength));
} else if (yAttachLength != 0.0 && yAttachLength < POSITION_EPS) {
WRITE_ERROR("The length of attached streets must be at least " + toString(POSITION_EPS));
hadError = true;
} else if (yAttachLength != 0.0 && yAttachLength < minAttachLength) {
WRITE_WARNINGF(TL("The length of attached streets should be at least % for the given lanenumber, lanewidth and junction radius"), toString(minAttachLength));
}
if (hadError) {
throw ProcessError();
}
NGNet* net = new NGNet(nb);
net->createChequerBoard(xNo, yNo, xLength, yLength, xAttachLength, yAttachLength);
return net;
}
RandomDistributor<int> neighborDist;
neighborDist.add(1, oc.getFloat("rand.neighbor-dist1"));
neighborDist.add(2, oc.getFloat("rand.neighbor-dist2"));
neighborDist.add(3, oc.getFloat("rand.neighbor-dist3"));
neighborDist.add(4, oc.getFloat("rand.neighbor-dist4"));
neighborDist.add(5, oc.getFloat("rand.neighbor-dist5"));
neighborDist.add(6, oc.getFloat("rand.neighbor-dist6"));
NGNet* net = new NGNet(nb);
NGRandomNetBuilder randomNet(*net,
DEG2RAD(oc.getFloat("rand.min-angle")),
oc.getFloat("rand.min-distance"),
oc.getFloat("rand.max-distance"),
oc.getFloat("rand.connectivity"),
oc.getInt("rand.num-tries"),
neighborDist);
randomNet.createNet(oc.getInt("rand.iterations"), oc.getBool("rand.grid"));
return net;
}
int
main(int argc, char** argv) {
OptionsCont& oc = OptionsCont::getOptions();
oc.setApplicationDescription(TL("Synthetic network generator for the microscopic, multi-modal traffic simulation SUMO."));
oc.setApplicationName("netgenerate", "Eclipse SUMO netgenerate " 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();
}
GeoConvHelper::init("!",
Position(oc.getFloat("offset.x"), oc.getFloat("offset.y")),
Boundary(), Boundary());
RandHelper::initRandGlobal();
NBNetBuilder nb;
nb.applyOptions(oc);
if (oc.isSet("type-files")) {
NIXMLTypesHandler handler(nb.getTypeCont());
NITypeLoader::load(handler, oc.getStringVector("type-files"), toString(SUMO_TAG_TYPES));
}
NGNet* net = buildNetwork(nb);
oc.resetWritable();
net->toNB();
delete net;
WRITE_MESSAGE(TL(" Generation done;"));
WRITE_MESSAGEF(TL(" % nodes generated."), toString<int>(nb.getNodeCont().size()));
WRITE_MESSAGEF(TL(" % edges generated."), toString<int>(nb.getEdgeCont().size()));
if (oc.getBool("tls.discard-simple")) {
nb.getNodeCont().discardTrafficLights(nb.getTLLogicCont(), true);
int removed = nb.getTLLogicCont().getNumExtracted();
if (removed > 0) {
WRITE_MESSAGEF(TL(" Removed % traffic lights at geometry-like nodes"), toString(removed));
}
}
nb.compute(oc);
nb.getNodeCont().printBuiltNodesStatistics();
NWFrame::writeNetwork(oc, nb);
} 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;
}