Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

adapt fmi4cpp for distributed systems #152

Open
wants to merge 14 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion include/fmi4cpp/fmi2/fmu.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ class fmu : public virtual fmu_provider<model_description, cs_fmu, me_fmu>
std::shared_ptr<const model_description> modelDescription_;

public:
explicit fmu(const std::filesystem::path& fmuPath);
explicit fmu(const std::filesystem::path& fmuPath, bool unzipFmu = true);

[[nodiscard]] std::string get_model_description_xml() const;
[[nodiscard]] std::shared_ptr<const model_description> get_model_description() const override;
Expand Down
58 changes: 33 additions & 25 deletions src/fmi4cpp/fmi2/fmu.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -11,34 +11,42 @@
using namespace fmi4cpp;
using namespace fmi4cpp::fmi2;

fmu::fmu(const std::filesystem::path& fmuPath)
fmu::fmu(const std::filesystem::path& fmuPath, bool unzipFmu)
{

if (!exists(fmuPath)) {
const auto err = "No such file '" + absolute(fmuPath).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
if ( std::filesystem::is_directory(fmuPath) && fs::exists(fmuPath / "modelDescription.xml") ) {
MLOG_DEBUG("fmuPath is a directory, assuming it is an unzipped FMU");
unzipFmu = false;
}

const std::string fmuName = fmuPath.stem().string();
std::filesystem::path tmpPath(std::filesystem::temp_directory_path() /= std::filesystem::path("fmi4cpp_" + fmuName + "_" + generate_simple_id(8)));

if (!create_directories(tmpPath)) {
const auto err = "Failed to create temporary directory '" + tmpPath.string() + "' !";
MLOG_FATAL(err);
throw std::runtime_error(err);
if (unzipFmu) {
if (!std::filesystem::exists(fmuPath)) {
const auto err = "No such file '" + absolute(fmuPath).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

const std::string fmuName = fmuPath.stem().string();
fs::path tmpPath(fs::temp_directory_path() /= fs::path("fmi4cpp_" + fmuName + "_" + generate_simple_id(8)));

if (!create_directories(tmpPath)) {
const auto err = "Failed to create temporary directory '" + tmpPath.string() + "' !";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

MLOG_DEBUG("Created temporary directory '" << tmpPath.string());

if (!unzip(fmuPath, tmpPath.string())) {
const auto err = "Failed to extract FMU '" + absolute(fmuPath).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

resource_ = std::make_shared<fmu_resource>(tmpPath);
modelDescription_ = std::move(parse_model_description(resource_->model_description_path()));
} else {
resource_ = std::make_shared<fmu_resource>(fmuPath);
modelDescription_ = std::move(parse_model_description(resource_->model_description_path()));
}

MLOG_DEBUG("Created temporary directory '" << tmpPath.string());

if (!unzip(fmuPath, tmpPath.string())) {
const auto err = "Failed to extract FMU '" + absolute(fmuPath).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

resource_ = std::make_shared<fmu_resource>(tmpPath);
modelDescription_ = std::move(parse_model_description(resource_->model_description_path()));
}

std::string fmu::get_model_description_xml() const
Expand Down
3 changes: 2 additions & 1 deletion tests/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@

add_executable(test_controlled_temperature test_controlled_temperature.cpp)
target_link_libraries(test_controlled_temperature PRIVATE fmi4cpp::fmi4cpp Catch2::Catch2)
target_link_libraries(test_controlled_temperature PRIVATE fmi4cpp::fmi4cpp Catch2::Catch2 libzip::libzip)
add_test(NAME test_controlled_temperature COMMAND test_controlled_temperature)
target_include_directories(test_controlled_temperature PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/../src)

add_executable(test_model_description1 test_modeldescription1.cpp)
target_link_libraries(test_model_description1 PRIVATE fmi4cpp::fmi4cpp Catch2::Catch2)
Expand Down
57 changes: 57 additions & 0 deletions tests/test_controlled_temperature.cpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,10 @@

#include <fmi4cpp/fmi2/fmi2.hpp>
#include <fmi4cpp/fs_portability.hpp>
#include <fmi4cpp/mlog.hpp>
#include <fmi4cpp/tools/os_util.hpp>
#include <fmi4cpp/tools/simple_id.hpp>
#include <fmi4cpp/tools/unzipper.hpp>

#define CATCH_CONFIG_MAIN
#include <catch2/catch.hpp>
Expand Down Expand Up @@ -37,5 +42,57 @@ TEST_CASE("ControlledTemperature_test1")

CHECK(298.15 == Approx(ref));

CHECK(slave->terminate());
}


TEST_CASE("ControlledTemperature_test2")
{
const std::string fmu_path = "../resources/fmus/2.0/cs/20sim/4.6.4.8004/"
"ControlledTemperature/ControlledTemperature.fmu";
namespace fs = std::filesystem;
if (!fs::exists(fmu_path)) {
const auto err = "No such file '" + fs::absolute(fmu_path).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

const std::string fmuName = fs::path(fmu_path).stem().string();
fs::path tmpPath(fs::temp_directory_path() /= fs::path("fmi4cpp_" + fmuName + "_" + generate_simple_id(8)));

if (!fs::create_directories(tmpPath)) {
const auto err = "Failed to create temporary directory '" + tmpPath.string() + "' !";
MLOG_FATAL(err);
throw std::runtime_error(err);
}

MLOG_DEBUG("Created temporary directory '" << tmpPath.string());

if (!unzip(fmu_path, tmpPath.string())) {
const auto err = "Failed to extract FMU '" + fs::absolute(fs::path(fmu_path)).string() + "'!";
MLOG_FATAL(err);
throw std::runtime_error(err);
}
auto fmu = fmi2::fmu(tmpPath,false).as_cs_fmu();

size_t numOutputs = 0;
for (const auto& v : *fmu->get_model_description()->model_variables) {
if (v.causality == fmi2::causality::output) {
numOutputs++;
}
}
CHECK(2 == numOutputs);

auto slave = fmu->new_instance();
CHECK(slave->setup_experiment());
CHECK(slave->enter_initialization_mode());
CHECK(slave->exit_initialization_mode());

double ref;
CHECK(slave->step(step_size));
CHECK(slave->read_real(vr, ref));

CHECK(298.15 == Approx(ref));

CHECK(slave->terminate());
}