diff --git a/Applications/FileIO/SWMM/SWMMInterface.cpp b/Applications/FileIO/SWMM/SWMMInterface.cpp
index cf049b5a159813ab035ec635edbcb1362e43673d..78548f6fc42c994e1cfbd6b5e936745b23c9332f 100644
--- a/Applications/FileIO/SWMM/SWMMInterface.cpp
+++ b/Applications/FileIO/SWMM/SWMMInterface.cpp
@@ -914,21 +914,26 @@ std::size_t SwmmInterface::getNumberOfParameters(SwmmObject obj_type) const
     if (OpenSwmmOutFile(const_cast<char*>(outfile.c_str())) != 0)
         return 0;
 
+    std::size_t n_params(0);
     switch (obj_type)
     {
         case SwmmObject::SUBCATCHMENT:
-            return (n_obj_params[0] - 1 + SWMM_Npolluts);
+            n_params = n_obj_params[0] + SWMM_Npolluts;
+            break;
         case SwmmObject::NODE:
-            return (n_obj_params[1] - 1 + SWMM_Npolluts);
+            n_params = n_obj_params[1] + SWMM_Npolluts;
+            break;
         case SwmmObject::LINK:
-            return (n_obj_params[2] - 1 + SWMM_Npolluts);
+            n_params = n_obj_params[2] + SWMM_Npolluts;
+            break;
         case SwmmObject::SYSTEM:
-            return n_obj_params[3];
+            n_params = n_obj_params[3];
+            break;
         default:
             ERR ("Object type not recognised.");
     }
     CloseSwmmOutFile();
-    return 0;
+    return n_params;
 }
 
 std::size_t SwmmInterface::getNumberOfTimeSteps() const
@@ -944,7 +949,7 @@ std::size_t SwmmInterface::getNumberOfTimeSteps() const
 bool SwmmInterface::addResultsToMesh(MeshLib::Mesh &mesh, SwmmObject const swmm_type,
     std::string const& vec_name, std::vector<double> const& data)
 {
-    if (!(swmm_type == SwmmObject::NODE) || (swmm_type == SwmmObject::LINK))
+    if (!(swmm_type == SwmmObject::NODE || swmm_type == SwmmObject::LINK))
     {
         ERR ("Information of this object type cannot be added to mesh.");
         return false;
@@ -968,18 +973,17 @@ bool SwmmInterface::addResultsToMesh(MeshLib::Mesh &mesh, SwmmObject const swmm_
         return false;
     }
 
-    MeshLib::Properties& p = mesh.getProperties();
-    MeshLib::MeshItemType item_type = (swmm_type == SwmmObject::NODE) ?
-        MeshLib::MeshItemType::Node : MeshLib::MeshItemType::Cell;
-    auto* const prop =
-        p.createNewPropertyVector<double>(vec_name, item_type, 1);
+    MeshLib::MeshItemType const item_type = (swmm_type == SwmmObject::NODE)
+                                                ? MeshLib::MeshItemType::Node
+                                                : MeshLib::MeshItemType::Cell;
+    MeshLib::PropertyVector<double>* prop =
+        MeshLib::getOrCreateMeshProperty<double>(mesh, vec_name, item_type, 1);
     if (!prop)
     {
-        ERR ("Error creating array \"%s\".", vec_name.c_str());
+        ERR("Error fetching array \"%s\".", vec_name.c_str());
         return false;
     }
-    prop->reserve(data.size());
-    std::copy(data.cbegin(), data.cend(), std::back_inserter(*prop));
+    std::copy(data.cbegin(), data.cend(), prop->begin());
     return true;
 }
 
@@ -1112,7 +1116,6 @@ std::vector<double> SwmmInterface::getArrayForObject(SwmmObject obj_type, std::s
         return data;
     }
 
-    INFO ("Fetching \"%s\"-data...", getArrayName(obj_type, var_idx, SWMM_Npolluts).c_str());
     std::size_t const n_time_steps (static_cast<std::size_t>(SWMM_Nperiods));
     for (std::size_t i=0; i<n_time_steps; ++i)
     {
@@ -1151,14 +1154,16 @@ std::string SwmmInterface::getArrayName(SwmmObject obj_type, std::size_t var_idx
         if (var_idx < n_obj_params[1])
             return node_vars[var_idx];
         if (var_idx < n_obj_params[1]+n_pollutants)
-            return _pollutant_names[var_idx-n_obj_params[1]];
+            return std::string("Node_" +
+                               _pollutant_names[var_idx - n_obj_params[1]]);
     }
     if (obj_type == SwmmObject::LINK)
     {
         if (var_idx < n_obj_params[2])
             return link_vars[var_idx];
         if (var_idx < n_obj_params[2]+n_pollutants)
-            return _pollutant_names[var_idx-n_obj_params[2]];
+            return std::string("Link_" +
+                               _pollutant_names[var_idx - n_obj_params[2]]);
     }
     if (obj_type == SwmmObject::SYSTEM && var_idx < n_obj_params[3])
     {
@@ -1266,6 +1271,19 @@ bool SwmmInterface::getLinkPointIds(std::vector<std::size_t> &inlets, std::vecto
     return true;
 }
 
+std::string SwmmInterface::swmmObjectTypeToString(SwmmObject const obj_type)
+{
+    if (obj_type == SwmmObject::NODE)
+        return "node";
+    if (obj_type == SwmmObject::LINK)
+        return "link";
+    if (obj_type == SwmmObject::SUBCATCHMENT)
+        return "subcatchment";
+    if (obj_type == SwmmObject::SYSTEM)
+        return "system";
+    return "undefined";
+}
+
 bool SwmmInterface::writeCsvForTimestep(std::string const& file_name, SwmmObject obj_type, std::size_t time_step) const
 {
     FileIO::CsvInterface csv;
@@ -1291,6 +1309,8 @@ bool SwmmInterface::writeCsvForTimestep(std::string const& file_name, SwmmObject
 bool SwmmInterface::writeCsvForObject(std::string const& file_name, SwmmObject obj_type, std::size_t obj_idx) const
 {
     FileIO::CsvInterface csv;
+    INFO("Writing data for %s %d.", swmmObjectTypeToString(obj_type).c_str(),
+         obj_idx);
     csv.addIndexVectorForWriting(getNumberOfTimeSteps());
     std::size_t const n_params (getNumberOfParameters(obj_type));
     for (std::size_t i=0; i<n_params; ++i)
diff --git a/Applications/FileIO/SWMM/SWMMInterface.h b/Applications/FileIO/SWMM/SWMMInterface.h
index da001e9d810d6bd123701f4736748e6dfbd00976..f697fb036f9bd4215a090aea23967664a0efb932 100644
--- a/Applications/FileIO/SWMM/SWMMInterface.h
+++ b/Applications/FileIO/SWMM/SWMMInterface.h
@@ -70,7 +70,7 @@ public:
         std::string const& vec_name, std::vector<double> const& data);
 
     /// Returns the mesh generated from SWMM file content.
-    MeshLib::Mesh const& getMesh() const { return *_mesh; }
+    MeshLib::Mesh& getMesh() const { return *_mesh; }
 
     /// Returns the name of the data array for the given object type and parameter index.
     std::string getArrayName(SwmmObject obj_type, std::size_t var_idx) const;
@@ -114,6 +114,9 @@ public:
     /// Checks if a SWMM output file exists for the current input
     bool existsSwmmOutputFile() const;
 
+    /// Returns a string with the name of the object type
+    static std::string swmmObjectTypeToString(SwmmObject const obj_type);
+
     /// Reading a SWMM input file and conversion into OGS geometry.
     static bool convertSwmmInputToGeometry(std::string const& inp_file_name,
         GeoLib::GEOObjects &geo_objects, bool add_subcatchments);
diff --git a/Applications/Utils/SWMMConverter/SWMMConverter.cpp b/Applications/Utils/SWMMConverter/SWMMConverter.cpp
index bab9608b52a2a9c49c581430260ad7c8c5860a2f..52527b66472aaeb5b9ddee6667bcce5eef202aa2 100644
--- a/Applications/Utils/SWMMConverter/SWMMConverter.cpp
+++ b/Applications/Utils/SWMMConverter/SWMMConverter.cpp
@@ -12,6 +12,7 @@
 
 #include "Applications/ApplicationsLib/LogogSetup.h"
 
+#include "BaseLib/FileTools.h"
 #include "BaseLib/StringTools.h"
 
 #include "GeoLib/GEOObjects.h"
@@ -23,109 +24,211 @@
 
 #include "Applications/FileIO/CsvInterface.h"
 
-int main(int argc, char *argv[])
+int writeGeoOutput(std::string input_file, std::string output_file)
 {
-    ApplicationsLib::LogogSetup setup;
+    GeoLib::GEOObjects geo_objects;
+    if (!FileIO::SwmmInterface::convertSwmmInputToGeometry(input_file,
+                                                           geo_objects, true))
+        return -1;
 
-    TCLAP::CmdLine cmd
-        ("Read files for the Storm Water Management Model (SWMM) and converts them to OGS.", ' ', "0.1");
-    TCLAP::ValueArg<std::string> mesh_output_arg
-        ("m","mesh", "mesh output file (*.vtu)", false, "", "mesh output file");
-    cmd.add(mesh_output_arg);
-    TCLAP::ValueArg<std::string> geo_output_arg
-        ("g","geo", "geometry output file (*.gml)", false, "", "geometry output file");
-    cmd.add(geo_output_arg);
-    TCLAP::ValueArg<std::string> swmm_input_arg
-        ("i","input", "SWMM input file (*.inp)", true, "", "input file");
-    cmd.add(swmm_input_arg);
-    cmd.parse( argc, argv );
+    GeoLib::IO::BoostXmlGmlInterface xml(geo_objects);
+    xml.setNameForExport(BaseLib::extractBaseNameWithoutExtension(input_file));
+    xml.writeToFile(output_file);
+    return 0;
+}
 
-    if (!(geo_output_arg.isSet() || mesh_output_arg.isSet()))
+int addObjectsToMesh(FileIO::SwmmInterface& swmm,
+                     MeshLib::Mesh& mesh,
+                     FileIO::SwmmObject const type,
+                     std::size_t const timestep)
+{
+    std::size_t const n_node_params(swmm.getNumberOfParameters(type));
+    for (std::size_t j = 0; j < n_node_params; ++j)
     {
-        ERR ("No output format given. Please specify OGS geometry or mesh output file.");
+        std::string const vec_name(swmm.getArrayName(type, j));
+        if (vec_name.empty())
+            return -2;
+        std::vector<double> data_vec =
+            swmm.getArrayAtTimeStep(type, timestep, j);
+        if (!swmm.addResultsToMesh(mesh, type, vec_name, data_vec))
+            return -3;
+    }
+    return 0;
+}
+
+int writeMeshOutput(std::string const& input_file,
+                    std::string const& output_file,
+                    bool const node_args,
+                    bool const link_args)
+{
+    std::unique_ptr<FileIO::SwmmInterface> swmm =
+        FileIO::SwmmInterface::create(input_file);
+    if (swmm == nullptr)
         return -1;
+
+    MeshLib::Mesh& mesh = swmm->getMesh();
+
+    bool const no_output_file = !swmm->existsSwmmOutputFile();
+    if (!(node_args || link_args) || no_output_file)
+    {
+        if (no_output_file)
+            INFO("No output file found.");
+        MeshLib::IO::VtuInterface vtkIO(&mesh, 0, false);
+        vtkIO.writeToFile(output_file);
+        return 0;
     }
 
-    if (geo_output_arg.isSet())
+    std::string const basename = BaseLib::dropFileExtension(output_file);
+    std::string const extension =
+        std::string("." + BaseLib::getFileExtension(output_file));
+    std::size_t const n_time_steps(swmm->getNumberOfTimeSteps());
+    INFO("Number of simulation time steps: %d", n_time_steps);
+    for (std::size_t i = 0; i < n_time_steps; i++)
     {
-        GeoLib::GEOObjects geo_objects;
-        if (!FileIO::SwmmInterface::convertSwmmInputToGeometry(swmm_input_arg.getValue(), geo_objects, true))
-            return -1;
+        if (node_args)
+            addObjectsToMesh(*swmm, mesh, FileIO::SwmmObject::NODE, i);
 
-        GeoLib::IO::BoostXmlGmlInterface xml(geo_objects);
-        xml.setNameForExport(BaseLib::extractBaseNameWithoutExtension(swmm_input_arg.getValue()));
-        xml.writeToFile(geo_output_arg.getValue());
-        return 0;
+        if (link_args)
+            addObjectsToMesh(*swmm, mesh, FileIO::SwmmObject::LINK, i);
+
+        MeshLib::IO::VtuInterface vtkio(&mesh, 0, false);
+        std::string name(basename + BaseLib::tostring(i) + extension);
+        vtkio.writeToFile(name);
     }
+    return 0;
+}
 
-    if (!mesh_output_arg.isSet())
-        return -1;
+void writeObjectsOfSwmmTypeToCsv(FileIO::SwmmInterface& swmm,
+                                 FileIO::SwmmObject const type,
+                                 std::string const& base,
+                                 std::string const& ext)
+{
+    std::size_t n_objects = swmm.getNumberOfObjects(type);
+    std::string const& type_str(swmm.swmmObjectTypeToString(type));
+    for (std::size_t i = 0; i < n_objects; ++i)
+    {
+        std::string const obj_name = swmm.getName(type, i);
+        std::string const obj_file_name =
+            std::string(base + "_" + type_str + "_" + obj_name + ext);
+        swmm.writeCsvForObject(obj_file_name, type, i);
+    }
+}
 
-    std::unique_ptr<FileIO::SwmmInterface> swmm = nullptr;
-    swmm = FileIO::SwmmInterface::create(swmm_input_arg.getValue());
+int writeCsvOutput(std::string input_file,
+                   std::string output_file,
+                   bool const node_args,
+                   bool const link_args,
+                   bool const catchment_args,
+                   bool const system_args)
+{
+    std::unique_ptr<FileIO::SwmmInterface> swmm =
+        FileIO::SwmmInterface::create(input_file);
     if (swmm == nullptr)
         return -1;
 
-    MeshLib::Mesh mesh = swmm->getMesh();
-    MeshLib::IO::VtuInterface vtkIO(&mesh, 0, false);
-    vtkIO.writeToFile(mesh_output_arg.getValue());
-
     if (!swmm->existsSwmmOutputFile())
     {
-        INFO ("No output file found.");
+        INFO("No output file found, skipping data conversion to CSV.");
+        return -1;
+    }
+
+    if (!(node_args || link_args || catchment_args || system_args))
+    {
+        INFO("No data category selected. Nothing to write.");
         return 0;
     }
 
-    INFO ("Simulation time steps: %d", swmm->getNumberOfTimeSteps());
-
-    // Example code: Writing node information to csv file
-    FileIO::CsvInterface csv;
-    csv.addIndexVectorForWriting(swmm->getNumberOfObjects(FileIO::SwmmObject::NODE));
-    std::vector<double> x;
-    std::vector<double> y;
-    std::vector<double> z;
-    swmm->getNodeCoordinateVectors(x,y,z);
-    csv.addVectorForWriting("x", x);
-    csv.addVectorForWriting("y", y);
-    csv.addVectorForWriting("z", z);
-    for (std::size_t i=0; i<9; ++i)
+    std::string const basename = BaseLib::dropFileExtension(output_file);
+    std::string const extension =
+        std::string("." + BaseLib::getFileExtension(output_file));
+
+    if (node_args)
+        writeObjectsOfSwmmTypeToCsv(*swmm, FileIO::SwmmObject::NODE, basename,
+                                    extension);
+
+    if (link_args)
+        writeObjectsOfSwmmTypeToCsv(*swmm, FileIO::SwmmObject::LINK, basename,
+                                    extension);
+
+    if (catchment_args)
+        writeObjectsOfSwmmTypeToCsv(*swmm, FileIO::SwmmObject::SUBCATCHMENT,
+                                    basename, extension);
+
+    if (system_args)
     {
-        std::vector<double> data_vec = swmm->getArrayAtTimeStep(FileIO::SwmmObject::NODE, 10, i);
-        csv.addVectorForWriting(swmm->getArrayName(FileIO::SwmmObject::NODE, i), data_vec);
+        std::string const obj_file_name =
+            std::string(basename + "_system" + extension);
+        swmm->writeCsvForObject(obj_file_name, FileIO::SwmmObject::SYSTEM, 0);
     }
-    csv.writeToFile("d:/csvtest.csv");
-    INFO ("CSV written.");
-    std::cin.ignore();
+    return 0;
+}
 
-    for (std::size_t i=0; i<swmm->getNumberOfParameters(FileIO::SwmmObject::NODE); ++i)
-        INFO ("%d\t%s", i, swmm->getArrayName(FileIO::SwmmObject::NODE, i).c_str());
-    std::cin.ignore();
+int main(int argc, char* argv[])
+{
+    ApplicationsLib::LogogSetup setup;
 
-    // Example code: Add simulated parameter to mesh for each timestep and write result
-    std::size_t n_time_steps (swmm->getNumberOfTimeSteps());
-    for (std::size_t i=0; i<n_time_steps; i++)
+    TCLAP::CmdLine cmd(
+        "Read files for the Storm Water Management Model (SWMM) and converts "
+        "them into OGS data structures.",
+        ' ', "0.1");
+    TCLAP::ValueArg<std::string> mesh_output_arg(
+        "m", "mesh", "mesh output file (*.vtu)", false, "", "mesh output file");
+    cmd.add(mesh_output_arg);
+    TCLAP::ValueArg<std::string> geo_output_arg(
+        "g", "geo", "geometry output file (*.gml)", false, "",
+        "geometry output file");
+    cmd.add(geo_output_arg);
+    TCLAP::ValueArg<std::string> csv_output_arg(
+        "c", "csv", "csv output file (*.csv)", false, "", "CSV output file");
+    cmd.add(csv_output_arg);
+    TCLAP::ValueArg<std::string> swmm_input_arg(
+        "i", "input", "SWMM input file (*.inp)", true, "", "input file");
+    cmd.add(swmm_input_arg);
+    TCLAP::SwitchArg add_nodes_arg(
+        "", "node_vars", "Read node variables and add to output mesh");
+    cmd.add(add_nodes_arg);
+    TCLAP::SwitchArg add_links_arg(
+        "", "link_vars", "Read link variables and add to output mesh");
+    cmd.add(add_links_arg);
+    TCLAP::SwitchArg add_subcatchments_arg(
+        "", "subcatchment_vars",
+        "Read subcatchment variables and write to CSV-file");
+    cmd.add(add_subcatchments_arg);
+    TCLAP::SwitchArg add_system_arg(
+        "", "system_vars", "Read system variables and write to CSV-file");
+    cmd.add(add_system_arg);
+    cmd.parse(argc, argv);
+
+    if (!(geo_output_arg.isSet() || mesh_output_arg.isSet() ||
+          csv_output_arg.isSet()))
     {
-        FileIO::SwmmObject type = FileIO::SwmmObject::NODE;
-        for (std::size_t j=6; j<9; ++j)
-        {
-            std::string vec_name (swmm->getArrayName(type, j));
-            if (vec_name.empty())
-                return -10;
-            std::vector<double> data_vec = swmm->getArrayAtTimeStep(type, i, j);
-            if (data_vec.empty())
-                return -20;
-            bool done = swmm->addResultsToMesh(mesh, type, vec_name, data_vec);
-            if (!done)
-                return -30;
-        }
+        ERR("No output format given. Please specify OGS geometry or mesh "
+            "output file.");
+        return -1;
+    }
 
-        MeshLib::IO::VtuInterface vtkio(&mesh, 0, false);
-        std::string name ("d:/swmmresults" + BaseLib::tostring(i) + ".vtu");
-        vtkio.writeToFile(name);
-        mesh.getProperties().removePropertyVector("P");
-        mesh.getProperties().removePropertyVector("NH4");
-        mesh.getProperties().removePropertyVector("CSB");
+    if ((add_subcatchments_arg.getValue() || add_system_arg.getValue()) &&
+        !csv_output_arg.isSet())
+    {
+        ERR("Please specify csv output file for exporting subcatchment or "
+            "system parameters.");
+        return -1;
     }
 
+    if (geo_output_arg.isSet())
+        writeGeoOutput(swmm_input_arg.getValue(), geo_output_arg.getValue());
+
+    if (mesh_output_arg.isSet())
+        writeMeshOutput(swmm_input_arg.getValue(), mesh_output_arg.getValue(),
+                        add_nodes_arg.getValue(), add_links_arg.getValue());
+
+    if (csv_output_arg.isSet())
+        writeCsvOutput(swmm_input_arg.getValue(),
+                       csv_output_arg.getValue(),
+                       add_nodes_arg.getValue(),
+                       add_links_arg.getValue(),
+                       add_subcatchments_arg.getValue(),
+                       add_system_arg.getValue());
+
     return 0;
 }