#include #include "libslic3r/Preset.hpp" #include "libslic3r/PrintConfig.hpp" #include "libslic3r/PrintConfigConstants.hpp" #include "libslic3r/LocalesUtils.hpp" #include #include #include #include using namespace Slic3r; SCENARIO("Generic config validation performs as expected.", "[Config]") { GIVEN("A config generated from default options") { Slic3r::DynamicPrintConfig config = Slic3r::DynamicPrintConfig::full_print_config(); WHEN( "outer_wall_line_width is set to 250%, a valid value") { config.set_deserialize_strict("outer_wall_line_width", "250%"); THEN( "The config is read as valid.") { REQUIRE(config.validate().empty()); } } WHEN( "outer_wall_line_width is set to -10, an invalid value") { config.set("outer_wall_line_width", -10); THEN( "Validate returns error") { REQUIRE_FALSE(config.validate().empty()); } } WHEN( "wall_loops is set to -10, an invalid value") { config.set("wall_loops", -10); THEN( "Validate returns error") { REQUIRE_FALSE(config.validate().empty()); } } } } SCENARIO("Config accessor functions perform as expected.", "[Config]") { GIVEN("A config generated from default options") { Slic3r::DynamicPrintConfig config = Slic3r::DynamicPrintConfig::full_print_config(); WHEN("A boolean option is set to a boolean value") { REQUIRE_NOTHROW(config.set("gcode_comments", true)); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("gcode_comments")->getBool() == true); } } WHEN("A boolean option is set to a string value representing a 0 or 1") { CHECK_NOTHROW(config.set_deserialize_strict("gcode_comments", "1")); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("gcode_comments")->getBool() == true); } } WHEN("A boolean option is set to a string value representing something other than 0 or 1") { THEN("A BadOptionTypeException exception is thrown.") { REQUIRE_THROWS_AS(config.set("gcode_comments", "Z"), BadOptionTypeException); } AND_THEN("Value is unchanged.") { REQUIRE(config.opt("gcode_comments")->getBool() == false); } } WHEN("A boolean option is set to an int value") { THEN("A BadOptionTypeException exception is thrown.") { REQUIRE_THROWS_AS(config.set("gcode_comments", 1), BadOptionTypeException); } } WHEN("A numeric option is set from serialized string") { config.set_deserialize_strict("raft_layers", "20"); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("raft_layers")->getInt() == 20); } } WHEN("An integer-based option is set through the integer interface") { config.set("raft_layers", 100); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("raft_layers")->getInt() == 100); } } WHEN("An floating-point option is set through the integer interface") { config.set("default_acceleration", 10); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("default_acceleration")->getFloat() == 10.0); } } WHEN("A floating-point option is set through the double interface") { config.set("default_acceleration", 5.5); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("default_acceleration")->getFloat() == 5.5); } } WHEN("An integer-based option is set through the double interface") { THEN("A BadOptionTypeException exception is thrown.") { REQUIRE_THROWS_AS(config.set("top_shell_layers", 5.5), BadOptionTypeException); } } WHEN("A numeric option is set to a non-numeric value.") { auto prev_value = config.opt("default_acceleration")->getFloat(); THEN("A BadOptionTypeException exception is thrown.") { REQUIRE_THROWS_AS(config.set_deserialize_strict("default_acceleration", "zzzz"), BadOptionValueException); } THEN("The value does not change.") { REQUIRE(config.opt("default_acceleration")->getFloat() == prev_value); } } WHEN("A string option is set through the string interface") { config.set("machine_end_gcode", "100"); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("machine_end_gcode")->value == "100"); } } WHEN("A string option is set through the integer interface") { config.set("machine_end_gcode", 100); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("machine_end_gcode")->value == "100"); } } WHEN("A string option is set through the double interface") { config.set("machine_end_gcode", 100.5); THEN("The underlying value is set correctly.") { REQUIRE(config.opt("machine_end_gcode")->value == float_to_string_decimal_point(100.5)); } } WHEN("A float or percent is set as a percent through the string interface.") { config.set_deserialize_strict("initial_layer_line_width", "100%"); THEN("Value and percent flag are 100/true") { auto tmp = config.opt("initial_layer_line_width"); REQUIRE(tmp->percent == true); REQUIRE(tmp->value == 100); } } WHEN("A float or percent is set as a float through the string interface.") { config.set_deserialize_strict("initial_layer_line_width", "100"); THEN("Value and percent flag are 100/false") { auto tmp = config.opt("initial_layer_line_width"); REQUIRE(tmp->percent == false); REQUIRE(tmp->value == 100); } } WHEN("A float or percent is set as a float through the int interface.") { config.set("initial_layer_line_width", 100); THEN("Value and percent flag are 100/false") { auto tmp = config.opt("initial_layer_line_width"); REQUIRE(tmp->percent == false); REQUIRE(tmp->value == 100); } } WHEN("A float or percent is set as a float through the double interface.") { config.set("initial_layer_line_width", 100.5); THEN("Value and percent flag are 100.5/false") { auto tmp = config.opt("initial_layer_line_width"); REQUIRE(tmp->percent == false); REQUIRE(tmp->value == 100.5); } } WHEN("A numeric vector is set from serialized string") { config.set_deserialize_strict("temperature_vitrification", "10,20"); THEN("The underlying value is set correctly.") { CHECK(config.opt("temperature_vitrification")->get_at(0) == 10); CHECK(config.opt("temperature_vitrification")->get_at(1) == 20); } } // FIXME: Design better accessors for vector elements // The following isn't supported and probably shouldn't be: // WHEN("An integer-based vector option is set through the integer interface") { // config.set("temperature_vitrification", 100); // THEN("The underlying value is set correctly.") { // REQUIRE(config.opt("temperature_vitrification")->get_at(0) == 100); // } // } WHEN("An integer-based vector option is set through the set_key_value interface") { config.set_key_value("temperature_vitrification", new ConfigOptionInts{10,20}); THEN("The underlying value is set correctly.") { CHECK(config.opt("temperature_vitrification")->get_at(0) == 10); CHECK(config.opt("temperature_vitrification")->get_at(1) == 20); } } WHEN("An invalid option is requested during set.") { THEN("A BadOptionTypeException exception is thrown.") { REQUIRE_THROWS_AS(config.set("deadbeef_invalid_option", 1), UnknownOptionException); REQUIRE_THROWS_AS(config.set("deadbeef_invalid_option", 1.0), UnknownOptionException); REQUIRE_THROWS_AS(config.set("deadbeef_invalid_option", "1"), UnknownOptionException); REQUIRE_THROWS_AS(config.set("deadbeef_invalid_option", true), UnknownOptionException); } } WHEN("An invalid option is requested during get.") { THEN("A UnknownOptionException exception is thrown.") { REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); } } WHEN("An invalid option is requested during opt.") { THEN("A UnknownOptionException exception is thrown.") { REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); REQUIRE_THROWS_AS(config.option_throw("deadbeef_invalid_option", false), UnknownOptionException); } } WHEN("getX called on an unset option.") { THEN("The default is returned.") { REQUIRE(config.opt_float("layer_height") == INITIAL_LAYER_HEIGHT); REQUIRE(config.opt_int("raft_layers") == INITIAL_RAFT_LAYERS); REQUIRE(config.opt_bool("reduce_crossing_wall") == INITIAL_REDUCE_CROSSING_WALL); } } WHEN("opt_float called on an option that has been set.") { config.set("layer_height", INITIAL_LAYER_HEIGHT*2); THEN("The set value is returned.") { REQUIRE(config.opt_float("layer_height") == INITIAL_LAYER_HEIGHT*2); } } } } SCENARIO("Config ini load/save interface", "[Config]") { WHEN("new_from_ini is called") { Slic3r::DynamicPrintConfig config; std::string path = std::string(TEST_DATA_DIR) + "/test_config/new_from_ini.ini"; config.load_from_ini(path, ForwardCompatibilitySubstitutionRule::Disable); THEN("Config object contains ini file options.") { REQUIRE(config.option_throw("filament_colour", false)->values.size() == 1); REQUIRE(config.option_throw("filament_colour", false)->values.front() == "#ABCD"); } } } // TODO: https://github.com/SoftFever/OrcaSlicer/issues/11269 - Is this test still relevant? Delete if not. // It was failing so at least "nozzle_type" and "extruder_printable_area" could not be serialized // and an exception was thrown, but "nozzle_type" has been around for at least 3 months now. // So maybe this test and the serialization logic in Config.?pp should be deleted if it doesn't get used. SCENARIO("DynamicPrintConfig serialization", "[Config]") { WHEN("DynamicPrintConfig is serialized and deserialized") { FullPrintConfig full_print_config; DynamicPrintConfig cfg; cfg.apply(full_print_config, false); std::string serialized; // try { std::ostringstream ss; cereal::BinaryOutputArchive oarchive(ss); oarchive(cfg); serialized = ss.str(); // } catch (const std::runtime_error & /* e */) { // // e.what(); // } CAPTURE(serialized.length()); THEN("Config object contains ini file options.") { DynamicPrintConfig cfg2; // try { std::stringstream ss(serialized); cereal::BinaryInputArchive iarchive(ss); iarchive(cfg2); // } catch (const std::runtime_error & /* e */) { // // e.what(); // } CAPTURE(cfg.diff_report(cfg2)); REQUIRE(cfg == cfg2); } } } SCENARIO("update_non_diff_values_to_base_config preserves child vectors when child has more extruders than parent", "[Config][Variant]") { GIVEN("A 2-extruder child printer config inheriting from a 1-extruder parent") { Slic3r::DynamicPrintConfig child; Slic3r::DynamicPrintConfig parent; child.set_key_value("nozzle_diameter", new Slic3r::ConfigOptionFloats({0.4, 0.4})); child.set_key_value("printer_extruder_id", new Slic3r::ConfigOptionInts({1, 2})); child.set_key_value("printer_extruder_variant", new Slic3r::ConfigOptionStrings({"Direct Drive Standard", "Direct Drive Standard"})); child.set_key_value("retraction_length", new Slic3r::ConfigOptionFloats({1.5, 1.5})); parent.set_key_value("nozzle_diameter", new Slic3r::ConfigOptionFloats({0.4})); parent.set_key_value("printer_extruder_id", new Slic3r::ConfigOptionInts({1})); parent.set_key_value("printer_extruder_variant", new Slic3r::ConfigOptionStrings({"Direct Drive Standard"})); parent.set_key_value("retraction_length", new Slic3r::ConfigOptionFloats({0.8})); const Slic3r::t_config_option_keys keys = { "retraction_length", "printer_extruder_id", "printer_extruder_variant" }; const std::set different_keys = { "retraction_length", "printer_extruder_id", "printer_extruder_variant" }; WHEN("update_non_diff_values_to_base_config is called") { std::string id_name = "printer_extruder_id"; std::string var_name = "printer_extruder_variant"; child.update_non_diff_values_to_base_config( parent, keys, different_keys, id_name, var_name, Slic3r::printer_options_with_variant_1, Slic3r::printer_options_with_variant_2); THEN("printer_extruder_id retains size 2") { REQUIRE(child.option("printer_extruder_id")->values.size() == 2); } THEN("printer_extruder_variant retains size 2") { REQUIRE(child.option("printer_extruder_variant")->values.size() == 2); } THEN("retraction_length retains size 2") { REQUIRE(child.option("retraction_length")->values.size() == 2); } THEN("printer_extruder_id values are preserved for both extruders") { auto* pe_id = child.option("printer_extruder_id"); REQUIRE(pe_id->values.size() == 2); REQUIRE(pe_id->values[0] == 1); REQUIRE(pe_id->values[1] == 2); } } } } SCENARIO("extend_default_config_length collapses selectable nozzle flow variants to Standard", "[Config][Variant]") { GIVEN("A single-extruder BBL-style profile with Standard and High Flow variants") { Slic3r::DynamicPrintConfig config; config.set_key_value("nozzle_diameter", new Slic3r::ConfigOptionFloats({0.4})); config.set_key_value("extruder_type", new Slic3r::ConfigOptionEnumsGeneric({Slic3r::etDirectDrive})); config.set_key_value("extruder_variant_list", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard,Direct Drive High Flow" })); config.set_key_value("printer_extruder_id", new Slic3r::ConfigOptionInts({1, 1})); config.set_key_value("printer_extruder_variant", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard", "Direct Drive High Flow" })); config.set_key_value("machine_max_speed_x", new Slic3r::ConfigOptionFloats({ 500, 200, 200, 200 })); WHEN("default config lengths are normalized") { Slic3r::extend_default_config_length(config, false, {}); THEN("only the Standard nozzle flow variant remains") { auto* variants = config.option("printer_extruder_variant"); REQUIRE(variants->values == std::vector({ "Direct Drive Standard" })); } THEN("Standard machine limits are kept") { auto* speed_x = config.option("machine_max_speed_x"); REQUIRE(speed_x->values == std::vector({500, 200})); } THEN("printer extruder ids are rebuilt per physical extruder") { auto* ids = config.option("printer_extruder_id"); REQUIRE(ids->values == std::vector({1})); } } } } SCENARIO("extend_default_config_length collapses mixed physical extruder variants", "[Config][Variant]") { GIVEN("An X2D-style profile with one Direct Drive and one Bowden extruder") { Slic3r::DynamicPrintConfig config; config.set_key_value("nozzle_diameter", new Slic3r::ConfigOptionFloats({0.4, 0.4})); config.set_key_value("extruder_type", new Slic3r::ConfigOptionEnumsGeneric({ Slic3r::etDirectDrive, Slic3r::etBowden })); config.set_key_value("extruder_variant_list", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard,Direct Drive High Flow", "Bowden Standard,Bowden High Flow" })); config.set_key_value("printer_extruder_id", new Slic3r::ConfigOptionInts({1, 1, 2, 2})); config.set_key_value("printer_extruder_variant", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard", "Direct Drive High Flow", "Bowden Standard", "Bowden High Flow" })); config.set_key_value("retraction_length", new Slic3r::ConfigOptionFloats({ 0.8, 0.8, 2.0, 2.0 })); config.set_key_value("machine_max_speed_e", new Slic3r::ConfigOptionFloats({ 30, 30, 30, 30, 120, 120, 120, 120 })); WHEN("default config lengths are normalized") { Slic3r::extend_default_config_length(config, false, {}); THEN("one Standard variant is kept for each physical extruder") { auto* variants = config.option("printer_extruder_variant"); REQUIRE(variants->values == std::vector({ "Direct Drive Standard", "Bowden Standard" })); } THEN("per-extruder retraction settings are aligned") { auto* retraction = config.option("retraction_length"); REQUIRE(retraction->values == std::vector({0.8, 2.0})); } THEN("stride-2 machine limits are aligned") { auto* speed_e = config.option("machine_max_speed_e"); REQUIRE(speed_e->values == std::vector({30, 30, 120, 120})); } } } } SCENARIO("extend_default_config_length collapses symmetric physical extruder variants", "[Config][Variant]") { GIVEN("An H2D-style profile with matching Direct Drive extruder variant lists") { Slic3r::DynamicPrintConfig config; config.set_key_value("nozzle_diameter", new Slic3r::ConfigOptionFloats({0.4, 0.4})); config.set_key_value("extruder_type", new Slic3r::ConfigOptionEnumsGeneric({ Slic3r::etDirectDrive, Slic3r::etDirectDrive })); config.set_key_value("extruder_variant_list", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard,Direct Drive High Flow", "Direct Drive Standard,Direct Drive High Flow" })); config.set_key_value("printer_extruder_id", new Slic3r::ConfigOptionInts({1, 1, 2, 2})); config.set_key_value("printer_extruder_variant", new Slic3r::ConfigOptionStrings({ "Direct Drive Standard", "Direct Drive High Flow", "Direct Drive Standard", "Direct Drive High Flow" })); config.set_key_value("machine_max_speed_e", new Slic3r::ConfigOptionFloats({ 50, 50, 60, 60, 70, 70, 80, 80 })); WHEN("default config lengths are normalized") { Slic3r::extend_default_config_length(config, false, {}); THEN("the Standard variant is kept for each physical extruder, not the first matching name globally") { auto* variants = config.option("printer_extruder_variant"); REQUIRE(variants->values == std::vector({ "Direct Drive Standard", "Direct Drive Standard" })); } THEN("stride-2 machine limits are kept from each extruder's Standard slot") { auto* speed_e = config.option("machine_max_speed_e"); REQUIRE(speed_e->values == std::vector({50, 50, 70, 70})); } } } } // SCENARIO("DynamicPrintConfig JSON serialization", "[Config]") { // WHEN("DynamicPrintConfig is serialized and deserialized") { // auto now = std::chrono::high_resolution_clock::now(); // auto timestamp = now.time_since_epoch().count(); // std::stringstream ss; // ss << "catch_test_serialization_" << timestamp << ".json"; // std::string filename = (fs::temp_directory_path() / ss.str()).string(); // TODO: Finish making a unit test for JSON serialization // FullPrintConfig full_print_config; // DynamicPrintConfig cfg; // cfg.apply(full_print_config, false); // std::string serialized; // try { // std::ostringstream ss; // cereal::BinaryOutputArchive oarchive(ss); // oarchive(cfg); // serialized = ss.str(); // } catch (const std::runtime_error & /* e */) { // // e.what(); // } // CAPTURE(serialized.length()); // THEN("Config object contains ini file options.") { // DynamicPrintConfig cfg2; // try { // std::stringstream ss(serialized); // cereal::BinaryInputArchive iarchive(ss); // iarchive(cfg2); // } catch (const std::runtime_error & /* e */) { // // e.what(); // } // CAPTURE(cfg.diff_report(cfg2)); // REQUIRE(cfg == cfg2); // } // } // }