From 06fb13552615579f6979e1541728149eecb848bd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Mon, 18 Sep 2023 23:26:22 +0200 Subject: [PATCH 01/11] First approach --- .gitmodules | 3 + CMakeLists.txt | 4 ++ README.md | 32 ++++++++++ lib/catch2 | 2 +- lib/libxlsxwriter | 1 + src/Platform/CMakeLists.txt | 4 +- src/Platform/ReportExcel.cc | 116 ++++++++++++++++++++++++++---------- src/Platform/ReportExcel.h | 33 ++++++---- 8 files changed, 150 insertions(+), 45 deletions(-) create mode 160000 lib/libxlsxwriter diff --git a/.gitmodules b/.gitmodules index 626d10f..235f4f1 100644 --- a/.gitmodules +++ b/.gitmodules @@ -13,3 +13,6 @@ [submodule "lib/openXLSX"] path = lib/openXLSX url = https://github.com/troldal/OpenXLSX.git +[submodule "lib/libxlsxwriter"] + path = lib/libxlsxwriter + url = https://github.com/jmcnamara/libxlsxwriter.git diff --git a/CMakeLists.txt b/CMakeLists.txt index be56d34..d74de50 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -55,6 +55,7 @@ add_git_submodule("lib/mdlp") add_git_submodule("lib/argparse") add_git_submodule("lib/json") add_git_submodule("lib/openXLSX") +#add_git_submodule("lib/libxlsxwriter") # Subdirectories # -------------- @@ -64,6 +65,9 @@ add_subdirectory(src/BayesNet) add_subdirectory(src/Platform) add_subdirectory(sample) +#find_package(PkgConfig REQUIRED) +#pkg_check_modules(Xlsxwriter REQUIRED IMPORTED_TARGET libxlsxwriter) + file(GLOB BayesNet_HEADERS CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/BayesNet/*.h ${BayesNet_SOURCE_DIR}/BayesNet/*.hpp) file(GLOB BayesNet_SOURCES CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/BayesNet/*.cc ${BayesNet_SOURCE_DIR}/src/BayesNet/*.cpp) file(GLOB Platform_SOURCES CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/Platform/*.cc ${BayesNet_SOURCE_DIR}/src/Platform/*.cpp) diff --git a/README.md b/README.md index c95d141..266bb8a 100644 --- a/README.md +++ b/README.md @@ -2,4 +2,36 @@ Bayesian Network Classifier with libtorch from scratch +## 0. Setup + +### libxlswriter + +Before compiling BayesNet. + +```bash +cd lib/libxlsxwriter +make +sudo make install +``` + +It has to be installed in /usr/local/lib otherwise CMakeLists.txt has to be modified accordingly + +Environment variable has to be set: + +```bash + export LD_LIBRARY_PATH=/usr/local/lib + ``` + +### Release + +```bash +make release +``` + +### Debug & Tests + +```bash +make debug +``` + ## 1. Introduction diff --git a/lib/catch2 b/lib/catch2 index 4acc518..9c541ca 160000 --- a/lib/catch2 +++ b/lib/catch2 @@ -1 +1 @@ -Subproject commit 4acc51828f7f93f3b2058a63f54d112af4034503 +Subproject commit 9c541ca72e7857dec71d8a41b97e42c2f1c92602 diff --git a/lib/libxlsxwriter b/lib/libxlsxwriter new file mode 160000 index 0000000..44e72c5 --- /dev/null +++ b/lib/libxlsxwriter @@ -0,0 +1 @@ +Subproject commit 44e72c5862f9d549453a4ff6e8ceab0da19705e5 diff --git a/src/Platform/CMakeLists.txt b/src/Platform/CMakeLists.txt index b5dc5b9..b885792 100644 --- a/src/Platform/CMakeLists.txt +++ b/src/Platform/CMakeLists.txt @@ -9,8 +9,8 @@ add_executable(manage manage.cc Results.cc ReportConsole.cc ReportExcel.cc Repor add_executable(list list.cc platformUtils Datasets.cc) target_link_libraries(main BayesNet ArffFiles mdlp "${TORCH_LIBRARIES}") if (${CMAKE_HOST_SYSTEM_NAME} MATCHES "Linux") - target_link_libraries(manage "${TORCH_LIBRARIES}" OpenXLSX::OpenXLSX stdc++fs) + target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so stdc++fs) + target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so) else() - target_link_libraries(manage "${TORCH_LIBRARIES}" OpenXLSX::OpenXLSX) endif() target_link_libraries(list ArffFiles mdlp "${TORCH_LIBRARIES}") \ No newline at end of file diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index 0ed5854..3c011aa 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -12,18 +12,72 @@ namespace platform { string do_grouping() const { return "\03"; } }; + void ReportExcel::writeString(int row, int col, const string& text, const string& style) + { + lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; + worksheet_write_string(worksheet, row, col, text.c_str(), efectiveStyle); + } + void ReportExcel::writeInt(int row, int col, const int number, const string& style) + { + lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; + worksheet_write_number(worksheet, row, col, number, efectiveStyle); + } + void ReportExcel::writeDouble(int row, int col, const double number, const string& style) + { + lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; + worksheet_write_number(worksheet, row, col, number, efectiveStyle); + } + + void ReportExcel::formatHeader() + { + worksheet_freeze_panes(worksheet, 8, 0); + } + + void ReportExcel::formatBody() + { + + } + + void ReportExcel::formatFooter() + { + + } + + void ReportExcel::createFormats() + { + lxw_format* bold = workbook_add_format(workbook); + format_set_bold(bold); + + lxw_format* result = workbook_add_format(workbook); + format_set_num_format(result, "0.0000000"); + + lxw_format* timeStyle = workbook_add_format(workbook); + format_set_num_format(timeStyle, "#,##0.00"); + + lxw_format* ints = workbook_add_format(workbook); + format_set_num_format(ints, "###,###"); + + lxw_format* floats = workbook_add_format(workbook); + format_set_num_format(floats, "#,###.00"); + + styles["bold"] = bold; + styles["result"] = result; + styles["time"] = timeStyle; + styles["ints"] = ints; + styles["floats"] = floats + } void ReportExcel::createFile() { - doc.create(Paths::excel() + "some_results.xlsx"); - wks = doc.workbook().worksheet("Sheet1"); - wks.setName(data["model"].get()); + workbook = workbook_new((Paths::excel() + "some_results.xlsx").c_str()); + const string name = data["model"].get(); + worksheet = workbook_add_worksheet(workbook, name.c_str()); + createFormats(); } void ReportExcel::closeFile() { - doc.save(); - doc.close(); + workbook_close(workbook); } void ReportExcel::header() @@ -32,17 +86,17 @@ namespace platform { locale::global(mylocale); cout.imbue(mylocale); stringstream oss; - wks.cell("A1").value().set( - "Report " + data["model"].get() + " ver. " + data["version"].get() + " with " + + writeString(0, 0, "Report " + data["model"].get() + " ver. " + data["version"].get() + " with " + to_string(data["folds"].get()) + " Folds cross validation and " + to_string(data["seeds"].size()) + - " random seeds. " + data["date"].get() + " " + data["time"].get()); - wks.cell("A2").value() = data["title"].get(); - wks.cell("A3").value() = "Random seeds: " + fromVector("seeds") + " Stratified: " + - (data["stratified"].get() ? "True" : "False"); + " random seeds. " + data["date"].get() + " " + data["time"].get(), "bold"); + writeString(1, 0, data["title"].get()); + writeString(2, 0, "Random seeds: " + fromVector("seeds") + " Stratified: " + + (data["stratified"].get() ? "True" : "False")); oss << "Execution took " << setprecision(2) << fixed << data["duration"].get() << " seconds, " << data["duration"].get() / 3600 << " hours, on " << data["platform"].get(); - wks.cell("A4").value() = oss.str(); - wks.cell("A5").value() = "Score is " + data["score_name"].get(); + writeString(3, 0, oss.str()); + writeString(4, 0, "Score is " + data["score_name"].get()); + formatHeader(); } void ReportExcel::body() @@ -52,7 +106,7 @@ namespace platform { "Time Std.", "Hyperparameters" }); int col = 1; for (const auto& item : head) { - wks.cell(8, col++).value() = item; + writeString(8, col++, item); } int row = 9; col = 1; @@ -60,17 +114,17 @@ namespace platform { double totalScore = 0.0; string hyperparameters; for (const auto& r : data["results"]) { - wks.cell(row, col).value() = r["dataset"].get(); - wks.cell(row, col + 1).value() = r["samples"].get(); - wks.cell(row, col + 2).value() = r["features"].get(); - wks.cell(row, col + 3).value() = r["classes"].get(); - wks.cell(row, col + 4).value() = r["nodes"].get(); - wks.cell(row, col + 5).value() = r["leaves"].get(); - wks.cell(row, col + 6).value() = r["depth"].get(); - wks.cell(row, col + 7).value() = r["score"].get(); - wks.cell(row, col + 8).value() = r["score_std"].get(); - wks.cell(row, col + 9).value() = r["time"].get(); - wks.cell(row, col + 10).value() = r["time_std"].get(); + writeString(row, col, r["dataset"].get()); + writeInt(row, col + 1, r["samples"].get(), "ints"); + writeInt(row, col + 2, r["features"].get(), "ints"); + writeInt(row, col + 3, r["classes"].get(), "ints"); + writeDouble(row, col + 4, r["nodes"].get(), "floats"); + writeDouble(row, col + 5, r["leaves"].get(), "floats"); + writeDouble(row, col + 6, r["depth"].get(), "floats"); + writeDouble(row, col + 7, r["score"].get(), "result"); + writeDouble(row, col + 8, r["score_std"].get(), "result"); + writeDouble(row, col + 9, r["time"].get(), "time"); + writeDouble(row, col + 10, r["time_std"].get(), "time"); try { hyperparameters = r["hyperparameters"].get(); } @@ -79,7 +133,7 @@ namespace platform { oss << r["hyperparameters"]; hyperparameters = oss.str(); } - wks.cell(row, col + 11).value() = hyperparameters; + writeString(row, col + 11, hyperparameters); lastResult = r; totalScore += r["score"].get(); row++; @@ -88,22 +142,24 @@ namespace platform { for (const string& group : { "scores_train", "scores_test", "times_train", "times_test" }) { row++; col = 1; - wks.cell(row, col).value() = group; + writeString(row, col, group); for (double item : lastResult[group]) { - wks.cell(row, ++col).value() = item; + writeDouble(row, ++col, item); } } } else { footer(totalScore, row); } + formatBody(); } void ReportExcel::footer(double totalScore, int row) { auto score = data["score_name"].get(); if (score == BestResult::scoreName()) { - wks.cell(row + 2, 1).value() = score + " compared to " + BestResult::title() + " .: "; - wks.cell(row + 2, 5).value() = totalScore / BestResult::score(); + writeString(row + 2, 1, score + " compared to " + BestResult::title() + " .: "); + writeDouble(row + 2, 7, totalScore / BestResult::score(), "result"); } + formatFooter(); } } \ No newline at end of file diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 3700681..69d9f69 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -1,25 +1,34 @@ #ifndef REPORTEXCEL_H #define REPORTEXCEL_H -#include +#include +#include "xlsxwriter.h" #include "ReportBase.h" #include "Paths.h" #include "Colors.h" namespace platform { using namespace std; - using namespace OpenXLSX; const int MAXLL = 128; - class ReportExcel : public ReportBase{ + class ReportExcel : public ReportBase { public: - explicit ReportExcel(json data_) : ReportBase(data_) {createFile();}; - virtual ~ReportExcel() {closeFile();}; - private: + explicit ReportExcel(json data_) : ReportBase(data_) { createFile(); }; + virtual ~ReportExcel() { closeFile(); }; + protected: + void writeString(int row, int col, const string& text, const string& style = ""); + void writeInt(int row, int col, const int number, const string& style = ""); + void writeDouble(int row, int col, const double number, const string& style = ""); + void formatHeader(); + void formatBody(); + void formatFooter(); + void createFormats(); + private: galeote void createFile(); - void closeFile(); - XLDocument doc; - XLWorksheet wks; - void header() override; - void body() override; - void footer(double totalScore, int row); + void closeFile(); + lxw_workbook* workbook; + lxw_worksheet* worksheet; + map styles; + void header() override; + void body() override; + void footer(double totalScore, int row); }; }; #endif // !REPORTEXCEL_H \ No newline at end of file From 1bdfbd1620e69ea76fb3158fd5205e4ea5f0fa41 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Tue, 19 Sep 2023 14:07:41 +0200 Subject: [PATCH 02/11] Complete adding color to format --- src/Platform/ReportExcel.cc | 207 +++++++++++++++++++++++++++--------- src/Platform/ReportExcel.h | 31 +++--- 2 files changed, 173 insertions(+), 65 deletions(-) diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index 3c011aa..37f9cb5 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -12,67 +12,154 @@ namespace platform { string do_grouping() const { return "\03"; } }; + + lxw_format* ReportExcel::efectiveStyle(const string& style) + { + lxw_format* efectiveStyle; + if (style == "") { + efectiveStyle = NULL; + } else { + string suffix = row % 2 ? "_odd" : "_even"; + efectiveStyle = styles.at(style + suffix); + } + return efectiveStyle; + } + void ReportExcel::writeString(int row, int col, const string& text, const string& style) { - lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; - worksheet_write_string(worksheet, row, col, text.c_str(), efectiveStyle); + worksheet_write_string(worksheet, row, col, text.c_str(), efectiveStyle(style)); } void ReportExcel::writeInt(int row, int col, const int number, const string& style) { - lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; - worksheet_write_number(worksheet, row, col, number, efectiveStyle); + worksheet_write_number(worksheet, row, col, number, efectiveStyle(style)); } void ReportExcel::writeDouble(int row, int col, const double number, const string& style) { - lxw_format* efectiveStyle = style == "" ? NULL : styles[style]; - worksheet_write_number(worksheet, row, col, number, efectiveStyle); + worksheet_write_number(worksheet, row, col, number, efectiveStyle(style)); } - void ReportExcel::formatHeader() + void ReportExcel::formatColumns() { - worksheet_freeze_panes(worksheet, 8, 0); + worksheet_freeze_panes(worksheet, 6, 1); + vector columns_sizes = { 22, 10, 9, 7, 12, 12, 12, 12, 12, 3, 14, 12, 50 }; + for (int i = 0; i < columns_sizes.size(); ++i) { + worksheet_set_column(worksheet, i, i, columns_sizes.at(i), NULL); + } } - void ReportExcel::formatBody() + void ReportExcel::addColor(lxw_format* style, bool odd) { - + uint32_t efectiveColor = odd ? colorEven : colorOdd; + format_set_bg_color(style, lxw_color_t(efectiveColor)); } - - void ReportExcel::formatFooter() + void ReportExcel::createStyle(const string& name, lxw_format* style, bool odd) { - + addColor(style, odd); + if (name == "textCentered") { + format_set_align(style, LXW_ALIGN_CENTER); + format_set_font_size(style, normalSize); + format_set_border(style, LXW_BORDER_THIN); + } else if (name == "text") { + format_set_font_size(style, normalSize); + format_set_border(style, LXW_BORDER_THIN); + } else if (name == "bodyHeader") { + format_set_bold(style); + format_set_font_size(style, normalSize); + format_set_align(style, LXW_ALIGN_VERTICAL_CENTER); + format_set_align(style, LXW_ALIGN_CENTER); + format_set_align(style, LXW_ALIGN_VERTICAL_CENTER); + format_set_bg_color(style, lxw_color_t(colorTitle)); + } else if (name == "result") { + format_set_font_size(style, normalSize); + format_set_border(style, LXW_BORDER_THIN); + format_set_num_format(style, "0.0000000"); + } else if (name == "time") { + format_set_font_size(style, normalSize); + format_set_border(style, LXW_BORDER_THIN); + format_set_num_format(style, "#,##0.000000"); + } else if (name == "ints") { + format_set_font_size(style, normalSize); + format_set_num_format(style, "###,###"); + format_set_border(style, LXW_BORDER_THIN); + } else if (name == "floats") { + format_set_border(style, LXW_BORDER_THIN); + format_set_font_size(style, normalSize); + format_set_num_format(style, "#,###.00"); + } } void ReportExcel::createFormats() { - lxw_format* bold = workbook_add_format(workbook); - format_set_bold(bold); + auto styleNames = { "text", "textCentered", "bodyHeader", "result", "time", "ints", "floats" }; + lxw_format* style; + for (string name : styleNames) { + lxw_format* style = workbook_add_format(workbook); + style = workbook_add_format(workbook); + createStyle(name, style, true); + styles[name + "_odd"] = style; + style = workbook_add_format(workbook); + createStyle(name, style, false); + styles[name + "_even"] = style; + } - lxw_format* result = workbook_add_format(workbook); - format_set_num_format(result, "0.0000000"); + // Header 1st line + lxw_format* headerFirst = workbook_add_format(workbook); + format_set_bold(headerFirst); + format_set_font_size(headerFirst, 18); + format_set_align(headerFirst, LXW_ALIGN_CENTER); + format_set_align(headerFirst, LXW_ALIGN_VERTICAL_CENTER); + format_set_border(headerFirst, LXW_BORDER_THIN); + format_set_bg_color(headerFirst, lxw_color_t(colorTitle)); - lxw_format* timeStyle = workbook_add_format(workbook); - format_set_num_format(timeStyle, "#,##0.00"); + // Header rest + lxw_format* headerRest = workbook_add_format(workbook); + format_set_bold(headerRest); + format_set_align(headerRest, LXW_ALIGN_CENTER); + format_set_font_size(headerRest, 16); + format_set_align(headerRest, LXW_ALIGN_VERTICAL_CENTER); + format_set_border(headerRest, LXW_BORDER_THIN); + format_set_bg_color(headerRest, lxw_color_t(colorOdd)); - lxw_format* ints = workbook_add_format(workbook); - format_set_num_format(ints, "###,###"); + // Header small + lxw_format* headerSmall = workbook_add_format(workbook); + format_set_bold(headerSmall); + format_set_align(headerSmall, LXW_ALIGN_LEFT); + format_set_font_size(headerSmall, 12); + format_set_border(headerSmall, LXW_BORDER_THIN); + format_set_align(headerSmall, LXW_ALIGN_VERTICAL_CENTER); + format_set_bg_color(headerSmall, lxw_color_t(colorOdd)); - lxw_format* floats = workbook_add_format(workbook); - format_set_num_format(floats, "#,###.00"); + styles["headerFirst"] = headerFirst; + styles["headerRest"] = headerRest; + styles["headerSmall"] = headerSmall; + } - styles["bold"] = bold; - styles["result"] = result; - styles["time"] = timeStyle; - styles["ints"] = ints; - styles["floats"] = floats + void ReportExcel::setProperties() + { + char line[data["title"].get().size() + 1]; + strcpy(line, data["title"].get().c_str()); + /* Create a properties structure and set some of the fields. */ + lxw_doc_properties properties = { + .title = line, + .subject = "Machine learning results", + .author = "Ricardo Montañana Gómez", + .manager = "Dr. J. A. Gámez, Dr. J. M. Puerta", + .company = "UCLM", + .comments = "Created with libxlsxwriter and c++", + }; + + /* Set the properties in the workbook. */ + workbook_set_properties(workbook, &properties); } void ReportExcel::createFile() { - workbook = workbook_new((Paths::excel() + "some_results.xlsx").c_str()); + workbook = workbook_new((Paths::excel() + fileName).c_str()); const string name = data["model"].get(); worksheet = workbook_add_worksheet(workbook, name.c_str()); + setProperties(); createFormats(); + formatColumns(); } void ReportExcel::closeFile() @@ -86,35 +173,49 @@ namespace platform { locale::global(mylocale); cout.imbue(mylocale); stringstream oss; - writeString(0, 0, "Report " + data["model"].get() + " ver. " + data["version"].get() + " with " + - to_string(data["folds"].get()) + " Folds cross validation and " + to_string(data["seeds"].size()) + - " random seeds. " + data["date"].get() + " " + data["time"].get(), "bold"); - writeString(1, 0, data["title"].get()); - writeString(2, 0, "Random seeds: " + fromVector("seeds") + " Stratified: " + - (data["stratified"].get() ? "True" : "False")); - oss << "Execution took " << setprecision(2) << fixed << data["duration"].get() << " seconds, " - << data["duration"].get() / 3600 << " hours, on " << data["platform"].get(); - writeString(3, 0, oss.str()); - writeString(4, 0, "Score is " + data["score_name"].get()); - formatHeader(); + string message = data["model"].get() + " ver. " + data["version"].get() + + data["language"].get() + " ver. " + data["language_version"].get() + + " with " + to_string(data["folds"].get()) + " Folds cross validation and " + to_string(data["seeds"].size()) + + " random seeds. " + data["date"].get() + " " + data["time"].get(); + worksheet_merge_range(worksheet, 0, 0, 0, 12, message.c_str(), styles["headerFirst"]); + worksheet_merge_range(worksheet, 1, 0, 1, 12, data["title"].get().c_str(), styles["headerRest"]); + worksheet_merge_range(worksheet, 2, 0, 3, 0, ("Score is " + data["score_name"].get()).c_str(), styles["headerRest"]); + worksheet_merge_range(worksheet, 2, 1, 3, 3, "Execution time", styles["headerRest"]); + oss << setprecision(2) << fixed << data["duration"].get() << " s"; + worksheet_merge_range(worksheet, 2, 4, 2, 5, oss.str().c_str(), styles["headerRest"]); + oss.str(""); + oss.clear(); + oss << setprecision(2) << fixed << data["duration"].get() / 3600 << " h"; + worksheet_merge_range(worksheet, 3, 4, 3, 5, oss.str().c_str(), styles["headerRest"]); + worksheet_merge_range(worksheet, 2, 6, 3, 7, "Platform", styles["headerRest"]); + worksheet_merge_range(worksheet, 2, 8, 3, 9, data["platform"].get().c_str(), styles["headerRest"]); + worksheet_merge_range(worksheet, 2, 10, 2, 12, ("Random seeds: " + fromVector("seeds")).c_str(), styles["headerSmall"]); + oss.str(""); + oss.clear(); + oss << "Stratified: " << (data["stratified"].get() ? "True" : "False"); + worksheet_merge_range(worksheet, 3, 10, 3, 11, oss.str().c_str(), styles["headerSmall"]); + oss.str(""); + oss.clear(); + oss << "Discretized: " << (data["discretized"].get() ? "True" : "False"); + worksheet_write_string(worksheet, 3, 12, oss.str().c_str(), styles["headerSmall"]); } void ReportExcel::body() { auto head = vector( - { "Dataset", "Samples", "Features", "Classes", "Nodes", "Edges", "States", "Score", "Score Std.", "Time", + { "Dataset", "Samples", "Features", "Classes", "Nodes", "Edges", "States", "Score", "Score Std.", "St.", "Time", "Time Std.", "Hyperparameters" }); - int col = 1; + int col = 0; for (const auto& item : head) { - writeString(8, col++, item); + writeString(5, col++, item, "bodyHeader"); } - int row = 9; - col = 1; + row = 6; + col = 0; json lastResult; double totalScore = 0.0; string hyperparameters; for (const auto& r : data["results"]) { - writeString(row, col, r["dataset"].get()); + writeString(row, col, r["dataset"].get(), "text"); writeInt(row, col + 1, r["samples"].get(), "ints"); writeInt(row, col + 2, r["features"].get(), "ints"); writeInt(row, col + 3, r["classes"].get(), "ints"); @@ -123,8 +224,10 @@ namespace platform { writeDouble(row, col + 6, r["depth"].get(), "floats"); writeDouble(row, col + 7, r["score"].get(), "result"); writeDouble(row, col + 8, r["score_std"].get(), "result"); - writeDouble(row, col + 9, r["time"].get(), "time"); - writeDouble(row, col + 10, r["time_std"].get(), "time"); + const string status = "X"; + writeString(row, col + 9, status, "textCentered"); + writeDouble(row, col + 10, r["time"].get(), "time"); + writeDouble(row, col + 11, r["time_std"].get(), "time"); try { hyperparameters = r["hyperparameters"].get(); } @@ -133,7 +236,7 @@ namespace platform { oss << r["hyperparameters"]; hyperparameters = oss.str(); } - writeString(row, col + 11, hyperparameters); + writeString(row, col + 12, hyperparameters, "text"); lastResult = r; totalScore += r["score"].get(); row++; @@ -150,16 +253,14 @@ namespace platform { } else { footer(totalScore, row); } - formatBody(); } void ReportExcel::footer(double totalScore, int row) { auto score = data["score_name"].get(); if (score == BestResult::scoreName()) { - writeString(row + 2, 1, score + " compared to " + BestResult::title() + " .: "); - writeDouble(row + 2, 7, totalScore / BestResult::score(), "result"); + worksheet_merge_range(worksheet, row + 2, 1, row + 2, 5, (score + " compared to " + BestResult::title() + " .:").c_str(), styles["text_even"]); + writeDouble(row + 2, 6, totalScore / BestResult::score(), "result"); } - formatFooter(); } } \ No newline at end of file diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 69d9f69..9dc3aa3 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -12,23 +12,30 @@ namespace platform { public: explicit ReportExcel(json data_) : ReportBase(data_) { createFile(); }; virtual ~ReportExcel() { closeFile(); }; - protected: + private: void writeString(int row, int col, const string& text, const string& style = ""); void writeInt(int row, int col, const int number, const string& style = ""); void writeDouble(int row, int col, const double number, const string& style = ""); - void formatHeader(); - void formatBody(); - void formatFooter(); + void formatColumns(); void createFormats(); - private: galeote + void setProperties(); void createFile(); - void closeFile(); - lxw_workbook* workbook; - lxw_worksheet* worksheet; - map styles; - void header() override; - void body() override; - void footer(double totalScore, int row); + void closeFile(); + lxw_workbook* workbook; + lxw_worksheet* worksheet; + map styles; + int row = 0; + int normalSize = 14; //font size for report body + uint32_t colorTitle = 0xB1A0C7; + uint32_t colorOdd = 0xDCE6F1; + uint32_t colorEven = 0xFDE9D9; + const string fileName = "some_results.xlsx"; + void header() override; + void body() override; + void footer(double totalScore, int row); + void createStyle(const string& name, lxw_format* style, bool odd); + void addColor(lxw_format* style, bool odd); + lxw_format* efectiveStyle(const string& name); }; }; #endif // !REPORTEXCEL_H \ No newline at end of file From f69f415b92f8807fad5502f9aa7089352891899b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Tue, 19 Sep 2023 17:55:03 +0200 Subject: [PATCH 03/11] Complete comparison with ZeroR --- src/Platform/CMakeLists.txt | 6 +-- src/Platform/ReportExcel.cc | 77 +++++++++++++++++++++++++++++++++++-- src/Platform/ReportExcel.h | 27 ++++++++++--- src/Platform/main.cc | 4 +- 4 files changed, 99 insertions(+), 15 deletions(-) diff --git a/src/Platform/CMakeLists.txt b/src/Platform/CMakeLists.txt index b885792..2a506b8 100644 --- a/src/Platform/CMakeLists.txt +++ b/src/Platform/CMakeLists.txt @@ -5,12 +5,12 @@ include_directories(${BayesNet_SOURCE_DIR}/lib/mdlp) include_directories(${BayesNet_SOURCE_DIR}/lib/argparse/include) include_directories(${BayesNet_SOURCE_DIR}/lib/json/include) add_executable(main main.cc Folding.cc platformUtils.cc Experiment.cc Datasets.cc Models.cc ReportConsole.cc ReportBase.cc) -add_executable(manage manage.cc Results.cc ReportConsole.cc ReportExcel.cc ReportBase.cc) +add_executable(manage manage.cc Results.cc ReportConsole.cc ReportExcel.cc ReportBase.cc Datasets.cc platformUtils.cc) add_executable(list list.cc platformUtils Datasets.cc) target_link_libraries(main BayesNet ArffFiles mdlp "${TORCH_LIBRARIES}") if (${CMAKE_HOST_SYSTEM_NAME} MATCHES "Linux") - target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so stdc++fs) - target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so) + target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so ArffFiles mdlp stdc++fs) else() + target_link_libraries(manage "${TORCH_LIBRARIES}" libxlsxwriter.so ArffFiles mdlp) endif() target_link_libraries(list ArffFiles mdlp "${TORCH_LIBRARIES}") \ No newline at end of file diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index 37f9cb5..acb2eaa 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -1,5 +1,6 @@ #include #include +#include "Datasets.h" #include "ReportExcel.h" #include "BestResult.h" @@ -12,6 +13,15 @@ namespace platform { string do_grouping() const { return "\03"; } }; + ReportExcel::ReportExcel(json data_) : ReportBase(data_), row(0) + { + normalSize = 14; //font size for report body + colorTitle = 0xB1A0C7; + colorOdd = 0xDCE6F1; + colorEven = 0xFDE9D9; + margin = .1; // margin to add to ZeroR comparison + createFile(); + } lxw_format* ReportExcel::efectiveStyle(const string& style) { @@ -41,7 +51,7 @@ namespace platform { void ReportExcel::formatColumns() { worksheet_freeze_panes(worksheet, 6, 1); - vector columns_sizes = { 22, 10, 9, 7, 12, 12, 12, 12, 12, 3, 14, 12, 50 }; + vector columns_sizes = { 22, 10, 9, 7, 12, 12, 12, 12, 12, 3, 15, 12, 23 }; for (int i = 0; i < columns_sizes.size(); ++i) { worksheet_set_column(worksheet, i, i, columns_sizes.at(i), NULL); } @@ -65,9 +75,9 @@ namespace platform { } else if (name == "bodyHeader") { format_set_bold(style); format_set_font_size(style, normalSize); - format_set_align(style, LXW_ALIGN_VERTICAL_CENTER); format_set_align(style, LXW_ALIGN_CENTER); format_set_align(style, LXW_ALIGN_VERTICAL_CENTER); + format_set_border(style, LXW_BORDER_THIN); format_set_bg_color(style, lxw_color_t(colorTitle)); } else if (name == "result") { format_set_font_size(style, normalSize); @@ -129,9 +139,17 @@ namespace platform { format_set_align(headerSmall, LXW_ALIGN_VERTICAL_CENTER); format_set_bg_color(headerSmall, lxw_color_t(colorOdd)); + // Summary style + lxw_format* summaryStyle = workbook_add_format(workbook); + format_set_bold(summaryStyle); + format_set_font_size(summaryStyle, 16); + format_set_border(summaryStyle, LXW_BORDER_THIN); + format_set_align(summaryStyle, LXW_ALIGN_VERTICAL_CENTER); + styles["headerFirst"] = headerFirst; styles["headerRest"] = headerRest; styles["headerSmall"] = headerSmall; + styles["summaryStyle"] = summaryStyle; } void ReportExcel::setProperties() @@ -173,7 +191,7 @@ namespace platform { locale::global(mylocale); cout.imbue(mylocale); stringstream oss; - string message = data["model"].get() + " ver. " + data["version"].get() + + string message = data["model"].get() + " ver. " + data["version"].get() + " " + data["language"].get() + " ver. " + data["language_version"].get() + " with " + to_string(data["folds"].get()) + " Folds cross validation and " + to_string(data["seeds"].size()) + " random seeds. " + data["date"].get() + " " + data["time"].get(); @@ -211,6 +229,7 @@ namespace platform { } row = 6; col = 0; + int hypSize = 22; json lastResult; double totalScore = 0.0; string hyperparameters; @@ -224,7 +243,7 @@ namespace platform { writeDouble(row, col + 6, r["depth"].get(), "floats"); writeDouble(row, col + 7, r["score"].get(), "result"); writeDouble(row, col + 8, r["score_std"].get(), "result"); - const string status = "X"; + const string status = compareResult(r["dataset"].get(), r["score"].get()); writeString(row, col + 9, status, "textCentered"); writeDouble(row, col + 10, r["time"].get(), "time"); writeDouble(row, col + 11, r["time_std"].get(), "time"); @@ -236,11 +255,18 @@ namespace platform { oss << r["hyperparameters"]; hyperparameters = oss.str(); } + if (hyperparameters.size() > hypSize) { + hypSize = hyperparameters.size(); + } writeString(row, col + 12, hyperparameters, "text"); lastResult = r; totalScore += r["score"].get(); row++; + } + // Set the right column width of hyperparameters with the maximum length + worksheet_set_column(worksheet, 12, 12, hypSize + 1, NULL); + // Show totals if only one dataset is present in the result if (data["results"].size() == 1) { for (const string& group : { "scores_train", "scores_test", "times_train", "times_test" }) { row++; @@ -254,9 +280,52 @@ namespace platform { footer(totalScore, row); } } + string ReportExcel::compareResult(const string& dataset, double result) + { + string status = " "; + if (data["score_name"].get() == "accuracy") { + auto dt = Datasets(Paths::datasets(), false); + dt.loadDataset(dataset); + auto numClasses = dt.getNClasses(dataset); + if (numClasses == 2) { + vector distribution = dt.getClassesCounts(dataset); + vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); + int maxCategory = distance(distribution.begin(), maxValue); + double mark = maxCategory * (1 + margin); + status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; + auto item = summary.find(status); + if (item != summary.end()) { + summary[status]++; + } else { + summary[status] = 1; + } + } + } + return status; + } + void ReportExcel::showSummary() + { + stringstream oss; + oss << "Better than ZeroR + " << setprecision(1) << fixed << margin * 100 << "%"; + + map meaning = { + {Symbols::equal_best, "Equal to best"}, + {Symbols::better_best, "Better than best"}, + {Symbols::cross, "Less than or equal to ZeroR"}, + {Symbols::upward_arrow, oss.str()} + }; + for (const auto& item : summary) { + worksheet_write_string(worksheet, row + 2, 1, item.first.c_str(), styles["summaryStyle"]); + worksheet_write_number(worksheet, row + 2, 2, item.second, styles["summaryStyle"]); + worksheet_merge_range(worksheet, row + 2, 3, row + 2, 5, meaning.at(item.first).c_str(), styles["summaryStyle"]); + row += 1; + } + } void ReportExcel::footer(double totalScore, int row) { + showSummary(); + row += 2 + summary.size(); auto score = data["score_name"].get(); if (score == BestResult::scoreName()) { worksheet_merge_range(worksheet, row + 2, 1, row + 2, 5, (score + " compared to " + BestResult::title() + " .:").c_str(), styles["text_even"]); diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 9dc3aa3..8c86788 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -8,9 +8,20 @@ namespace platform { using namespace std; const int MAXLL = 128; + class Symbols { + public: + inline static const string check_mark{ "\u2714" }; + inline static const string exclamation{ "\u2757" }; + inline static const string black_star{ "\u2605" }; + inline static const string cross{ "\u2717" }; + inline static const string upward_arrow{ "\u27B6" }; + inline static const string down_arrow{ "\u27B4" }; + inline static const string equal_best{ check_mark }; + inline static const string better_best{ black_star }; + }; class ReportExcel : public ReportBase { public: - explicit ReportExcel(json data_) : ReportBase(data_) { createFile(); }; + explicit ReportExcel(json data_); virtual ~ReportExcel() { closeFile(); }; private: void writeString(int row, int col, const string& text, const string& style = ""); @@ -21,14 +32,17 @@ namespace platform { void setProperties(); void createFile(); void closeFile(); + void showSummary(); lxw_workbook* workbook; lxw_worksheet* worksheet; map styles; - int row = 0; - int normalSize = 14; //font size for report body - uint32_t colorTitle = 0xB1A0C7; - uint32_t colorOdd = 0xDCE6F1; - uint32_t colorEven = 0xFDE9D9; + map summary; + int row; + int normalSize; //font size for report body + uint32_t colorTitle; + uint32_t colorOdd; + uint32_t colorEven; + double margin; const string fileName = "some_results.xlsx"; void header() override; void body() override; @@ -36,6 +50,7 @@ namespace platform { void createStyle(const string& name, lxw_format* style, bool odd); void addColor(lxw_format* style, bool odd); lxw_format* efectiveStyle(const string& name); + string compareResult(const string& dataset, double result); }; }; #endif // !REPORTEXCEL_H \ No newline at end of file diff --git a/src/Platform/main.cc b/src/Platform/main.cc index 9f8e00b..a122ad2 100644 --- a/src/Platform/main.cc +++ b/src/Platform/main.cc @@ -87,7 +87,7 @@ int main(int argc, char** argv) auto stratified = program.get("stratified"); auto n_folds = program.get("folds"); auto seeds = program.get>("seeds"); - auto hyperparameters =program.get("hyperparameters"); + auto hyperparameters = program.get("hyperparameters"); vector filesToTest; auto datasets = platform::Datasets(path, true, platform::ARFF); auto title = program.get("title"); @@ -102,7 +102,7 @@ int main(int argc, char** argv) } filesToTest.push_back(file_name); } else { - filesToTest = platform::Datasets(path, true, platform::ARFF).getNames(); + filesToTest = datasets.getNames(); saveResults = true; } /* From 925f71166cec25effdbc18cd9a0b74d5b936a1aa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Tue, 19 Sep 2023 23:46:49 +0200 Subject: [PATCH 04/11] Fix mistake in comparison --- src/Platform/ReportExcel.cc | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index acb2eaa..50293bf 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -156,7 +156,6 @@ namespace platform { { char line[data["title"].get().size() + 1]; strcpy(line, data["title"].get().c_str()); - /* Create a properties structure and set some of the fields. */ lxw_doc_properties properties = { .title = line, .subject = "Machine learning results", @@ -165,8 +164,6 @@ namespace platform { .company = "UCLM", .comments = "Created with libxlsxwriter and c++", }; - - /* Set the properties in the workbook. */ workbook_set_properties(workbook, &properties); } @@ -289,9 +286,12 @@ namespace platform { auto numClasses = dt.getNClasses(dataset); if (numClasses == 2) { vector distribution = dt.getClassesCounts(dataset); + double nSamples = dt.getNSamples(dataset); vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); - int maxCategory = distance(distribution.begin(), maxValue); - double mark = maxCategory * (1 + margin); + double mark = *maxValue / nSamples * (1 + margin); + if (mark > 1) { + mark = 0.9995; + } status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; auto item = summary.find(status); if (item != summary.end()) { @@ -325,11 +325,11 @@ namespace platform { void ReportExcel::footer(double totalScore, int row) { showSummary(); - row += 2 + summary.size(); + row += 4 + summary.size(); auto score = data["score_name"].get(); if (score == BestResult::scoreName()) { - worksheet_merge_range(worksheet, row + 2, 1, row + 2, 5, (score + " compared to " + BestResult::title() + " .:").c_str(), styles["text_even"]); - writeDouble(row + 2, 6, totalScore / BestResult::score(), "result"); + worksheet_merge_range(worksheet, row, 1, row, 5, (score + " compared to " + BestResult::title() + " .:").c_str(), efectiveStyle("text")); + writeDouble(row, 6, totalScore / BestResult::score(), "result"); } } } \ No newline at end of file From 20a6ebab7c78902548fceb733030676654f1caf5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 00:58:01 +0200 Subject: [PATCH 05/11] Support to add any number of sheets to excel --- src/Platform/ReportExcel.cc | 28 +++++++++++++++++++++++++--- src/Platform/ReportExcel.h | 4 ++-- src/Platform/Results.cc | 9 +++++++-- src/Platform/Results.h | 5 ++++- 4 files changed, 38 insertions(+), 8 deletions(-) diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index 50293bf..ab0b0a2 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -13,7 +13,8 @@ namespace platform { string do_grouping() const { return "\03"; } }; - ReportExcel::ReportExcel(json data_) : ReportBase(data_), row(0) + + ReportExcel::ReportExcel(json data_, lxw_workbook* workbook) : ReportBase(data_), row(0), workbook(workbook) { normalSize = 14; //font size for report body colorTitle = 0xB1A0C7; @@ -23,6 +24,11 @@ namespace platform { createFile(); } + lxw_workbook* ReportExcel::getWorkbook() + { + return workbook; + } + lxw_format* ReportExcel::efectiveStyle(const string& style) { lxw_format* efectiveStyle; @@ -169,9 +175,25 @@ namespace platform { void ReportExcel::createFile() { - workbook = workbook_new((Paths::excel() + fileName).c_str()); + if (workbook == NULL) { + workbook = workbook_new((Paths::excel() + fileName).c_str()); + } const string name = data["model"].get(); - worksheet = workbook_add_worksheet(workbook, name.c_str()); + string suffix = ""; + int num = 1; + // Create a sheet with the name of the model + while (true) { + string efectiveName = name + suffix; + worksheet = workbook_add_worksheet(workbook, efectiveName.c_str()); + if (worksheet == NULL) { + suffix = to_string(++num); + } else { + break; + } + if (num > 100) { + throw invalid_argument("Couldn't create sheet " + efectiveName); + } + } setProperties(); createFormats(); formatColumns(); diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 8c86788..6e0f6eb 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -21,8 +21,8 @@ namespace platform { }; class ReportExcel : public ReportBase { public: - explicit ReportExcel(json data_); - virtual ~ReportExcel() { closeFile(); }; + explicit ReportExcel(json data_, lxw_workbook* workbook); + lxw_workbook* getWorkbook(); private: void writeString(int row, int col, const string& text, const string& style = ""); void writeInt(int row, int col, const int number, const string& style = ""); diff --git a/src/Platform/Results.cc b/src/Platform/Results.cc index 3566ab7..e136a62 100644 --- a/src/Platform/Results.cc +++ b/src/Platform/Results.cc @@ -104,13 +104,15 @@ namespace platform { cout << "Invalid index" << endl; return -1; } - void Results::report(const int index, const bool excelReport) const + void Results::report(const int index, const bool excelReport) { cout << Colors::YELLOW() << "Reporting " << files.at(index).getFilename() << endl; auto data = files.at(index).load(); if (excelReport) { - ReportExcel reporter(data); + ReportExcel reporter(data, workbook); reporter.show(); + openExcel = true; + workbook = reporter.getWorkbook(); } else { ReportConsole reporter(data); reporter.show(); @@ -281,6 +283,9 @@ namespace platform { sortDate(); show(); menu(); + if (openExcel) { + workbook_close(workbook); + } cout << "Done!" << endl; } diff --git a/src/Platform/Results.h b/src/Platform/Results.h index 3f5655a..c418135 100644 --- a/src/Platform/Results.h +++ b/src/Platform/Results.h @@ -1,5 +1,6 @@ #ifndef RESULTS_H #define RESULTS_H +#include "xlsxwriter.h" #include #include #include @@ -44,10 +45,12 @@ namespace platform { bool complete; bool partial; bool indexList = true; + bool openExcel = false; + lxw_workbook* workbook = NULL; vector files; void load(); // Loads the list of results void show() const; - void report(const int index, const bool excelReport) const; + void report(const int index, const bool excelReport); void showIndex(const int index, const int idx) const; int getIndex(const string& intent) const; void menu(); From 3d0f29fda32902f04bc7c0395f7a12f29e88a4ad Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 01:01:40 +0200 Subject: [PATCH 06/11] Remove .vscode/settings.json from repository --- .gitignore | 1 + .vscode/settings.json | 109 ------------------------------------------ 2 files changed, 1 insertion(+), 109 deletions(-) delete mode 100644 .vscode/settings.json diff --git a/.gitignore b/.gitignore index 8855507..dea436c 100644 --- a/.gitignore +++ b/.gitignore @@ -36,3 +36,4 @@ build/ cmake-build*/** .idea puml/** +.vscode/settings.json diff --git a/.vscode/settings.json b/.vscode/settings.json deleted file mode 100644 index d7af13f..0000000 --- a/.vscode/settings.json +++ /dev/null @@ -1,109 +0,0 @@ -{ - "files.associations": { - "*.rmd": "markdown", - "*.py": "python", - "vector": "cpp", - "__bit_reference": "cpp", - "__bits": "cpp", - "__config": "cpp", - "__debug": "cpp", - "__errc": "cpp", - "__hash_table": "cpp", - "__locale": "cpp", - "__mutex_base": "cpp", - "__node_handle": "cpp", - "__nullptr": "cpp", - "__split_buffer": "cpp", - "__string": "cpp", - "__threading_support": "cpp", - "__tuple": "cpp", - "array": "cpp", - "atomic": "cpp", - "bitset": "cpp", - "cctype": "cpp", - "chrono": "cpp", - "clocale": "cpp", - "cmath": "cpp", - "compare": "cpp", - "complex": "cpp", - "concepts": "cpp", - "cstdarg": "cpp", - "cstddef": "cpp", - "cstdint": "cpp", - "cstdio": "cpp", - "cstdlib": "cpp", - "cstring": "cpp", - "ctime": "cpp", - "cwchar": "cpp", - "cwctype": "cpp", - "exception": "cpp", - "initializer_list": "cpp", - "ios": "cpp", - "iosfwd": "cpp", - "istream": "cpp", - "limits": "cpp", - "locale": "cpp", - "memory": "cpp", - "mutex": "cpp", - "new": "cpp", - "optional": "cpp", - "ostream": "cpp", - "ratio": "cpp", - "sstream": "cpp", - "stdexcept": "cpp", - "streambuf": "cpp", - "string": "cpp", - "string_view": "cpp", - "system_error": "cpp", - "tuple": "cpp", - "type_traits": "cpp", - "typeinfo": "cpp", - "unordered_map": "cpp", - "variant": "cpp", - "algorithm": "cpp", - "iostream": "cpp", - "iomanip": "cpp", - "numeric": "cpp", - "set": "cpp", - "__tree": "cpp", - "deque": "cpp", - "list": "cpp", - "map": "cpp", - "unordered_set": "cpp", - "any": "cpp", - "condition_variable": "cpp", - "forward_list": "cpp", - "fstream": "cpp", - "stack": "cpp", - "thread": "cpp", - "__memory": "cpp", - "filesystem": "cpp", - "*.toml": "toml", - "utility": "cpp", - "__verbose_abort": "cpp", - "bit": "cpp", - "random": "cpp", - "*.tcc": "cpp", - "functional": "cpp", - "iterator": "cpp", - "memory_resource": "cpp", - "format": "cpp", - "valarray": "cpp", - "regex": "cpp", - "span": "cpp", - "cfenv": "cpp", - "cinttypes": "cpp", - "csetjmp": "cpp", - "future": "cpp", - "queue": "cpp", - "typeindex": "cpp", - "shared_mutex": "cpp", - "*.ipp": "cpp", - "cassert": "cpp", - "charconv": "cpp", - "source_location": "cpp", - "ranges": "cpp" - }, - "cmake.configureOnOpen": false, - "C_Cpp.default.configurationProvider": "ms-vscode.cmake-tools" -} \ No newline at end of file From c280e254caeef83919398cd3fc2c4190eaedbb61 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 01:09:58 +0200 Subject: [PATCH 07/11] Remove OpenXLSX submodule --- .gitmodules | 3 --- CMakeLists.txt | 5 ----- lib/openXLSX | 1 - 3 files changed, 9 deletions(-) delete mode 160000 lib/openXLSX diff --git a/.gitmodules b/.gitmodules index 235f4f1..dbb94fc 100644 --- a/.gitmodules +++ b/.gitmodules @@ -10,9 +10,6 @@ [submodule "lib/json"] path = lib/json url = https://github.com/nlohmann/json.git -[submodule "lib/openXLSX"] - path = lib/openXLSX - url = https://github.com/troldal/OpenXLSX.git [submodule "lib/libxlsxwriter"] path = lib/libxlsxwriter url = https://github.com/jmcnamara/libxlsxwriter.git diff --git a/CMakeLists.txt b/CMakeLists.txt index d74de50..83feb88 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -54,8 +54,6 @@ endif (ENABLE_CLANG_TIDY) add_git_submodule("lib/mdlp") add_git_submodule("lib/argparse") add_git_submodule("lib/json") -add_git_submodule("lib/openXLSX") -#add_git_submodule("lib/libxlsxwriter") # Subdirectories # -------------- @@ -65,9 +63,6 @@ add_subdirectory(src/BayesNet) add_subdirectory(src/Platform) add_subdirectory(sample) -#find_package(PkgConfig REQUIRED) -#pkg_check_modules(Xlsxwriter REQUIRED IMPORTED_TARGET libxlsxwriter) - file(GLOB BayesNet_HEADERS CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/BayesNet/*.h ${BayesNet_SOURCE_DIR}/BayesNet/*.hpp) file(GLOB BayesNet_SOURCES CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/BayesNet/*.cc ${BayesNet_SOURCE_DIR}/src/BayesNet/*.cpp) file(GLOB Platform_SOURCES CONFIGURE_DEPENDS ${BayesNet_SOURCE_DIR}/src/Platform/*.cc ${BayesNet_SOURCE_DIR}/src/Platform/*.cpp) diff --git a/lib/openXLSX b/lib/openXLSX deleted file mode 160000 index b80da42..0000000 --- a/lib/openXLSX +++ /dev/null @@ -1 +0,0 @@ -Subproject commit b80da42d1454f361c29117095ebe1989437db390 From b9bc0088f300497a95958d4f83aa462ce4bed66d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 10:30:45 +0200 Subject: [PATCH 08/11] Add format to unique dataset results summary --- src/Platform/ReportExcel.cc | 21 +++++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index ab0b0a2..bfda442 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -180,20 +180,22 @@ namespace platform { } const string name = data["model"].get(); string suffix = ""; + string efectiveName; int num = 1; // Create a sheet with the name of the model while (true) { - string efectiveName = name + suffix; - worksheet = workbook_add_worksheet(workbook, efectiveName.c_str()); - if (worksheet == NULL) { + efectiveName = name + suffix; + if (workbook_get_worksheet_by_name(workbook, efectiveName.c_str())) { suffix = to_string(++num); } else { + worksheet = workbook_add_worksheet(workbook, efectiveName.c_str()); break; } if (num > 100) { throw invalid_argument("Couldn't create sheet " + efectiveName); } } + cout << "Adding sheet " << efectiveName << " to " << Paths::excel() + fileName << endl; setProperties(); createFormats(); formatColumns(); @@ -284,17 +286,24 @@ namespace platform { } // Set the right column width of hyperparameters with the maximum length - worksheet_set_column(worksheet, 12, 12, hypSize + 1, NULL); + worksheet_set_column(worksheet, 12, 12, hypSize + 5, NULL); // Show totals if only one dataset is present in the result if (data["results"].size() == 1) { for (const string& group : { "scores_train", "scores_test", "times_train", "times_test" }) { row++; col = 1; - writeString(row, col, group); + writeString(row, col, group, "text"); for (double item : lastResult[group]) { - writeDouble(row, ++col, item); + string style = group.find("scores") != string::npos ? "result" : "time"; + writeDouble(row, ++col, item, style); } } + // Set with of columns to show those totals completely + worksheet_set_column(worksheet, 1, 1, 12, NULL); + for (int i = 2; i < 7; ++i) { + // doesn't work with from col to col, so... + worksheet_set_column(worksheet, i, i, 15, NULL); + } } else { footer(totalScore, row); } From 68f22a673dc25fc53c33dff5dd541a61cece14b7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 11:40:01 +0200 Subject: [PATCH 09/11] Add comparison to report console --- src/Platform/ReportBase.cc | 38 +++++++++++++++++++++++++++++++++++ src/Platform/ReportBase.h | 19 +++++++++++++++++- src/Platform/ReportConsole.cc | 28 +++++++++++++++++++------- src/Platform/ReportConsole.h | 5 +++-- src/Platform/ReportExcel.cc | 38 +---------------------------------- src/Platform/ReportExcel.h | 16 +-------------- 6 files changed, 82 insertions(+), 62 deletions(-) diff --git a/src/Platform/ReportBase.cc b/src/Platform/ReportBase.cc index 24125f8..2289640 100644 --- a/src/Platform/ReportBase.cc +++ b/src/Platform/ReportBase.cc @@ -1,10 +1,22 @@ #include #include +#include "Datasets.h" #include "ReportBase.h" #include "BestResult.h" namespace platform { + ReportBase::ReportBase(json data_) : margin(0.1), data(data_) + { + stringstream oss; + oss << "Better than ZeroR + " << setprecision(1) << fixed << margin * 100 << "%"; + meaning = { + {Symbols::equal_best, "Equal to best"}, + {Symbols::better_best, "Better than best"}, + {Symbols::cross, "Less than or equal to ZeroR"}, + {Symbols::upward_arrow, oss.str()} + }; + } string ReportBase::fromVector(const string& key) { stringstream oss; @@ -34,4 +46,30 @@ namespace platform { header(); body(); } + string ReportBase::compareResult(const string& dataset, double result) + { + string status = " "; + if (data["score_name"].get() == "accuracy") { + auto dt = Datasets(Paths::datasets(), false); + dt.loadDataset(dataset); + auto numClasses = dt.getNClasses(dataset); + if (numClasses == 2) { + vector distribution = dt.getClassesCounts(dataset); + double nSamples = dt.getNSamples(dataset); + vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); + double mark = *maxValue / nSamples * (1 + margin); + if (mark > 1) { + mark = 0.9995; + } + status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; + auto item = summary.find(status); + if (item != summary.end()) { + summary[status]++; + } else { + summary[status] = 1; + } + } + } + return status; + } } \ No newline at end of file diff --git a/src/Platform/ReportBase.h b/src/Platform/ReportBase.h index 2acbbc7..f587c2e 100644 --- a/src/Platform/ReportBase.h +++ b/src/Platform/ReportBase.h @@ -2,14 +2,26 @@ #define REPORTBASE_H #include #include +#include "Paths.h" #include using json = nlohmann::json; namespace platform { using namespace std; + class Symbols { + public: + inline static const string check_mark{ "\u2714" }; + inline static const string exclamation{ "\u2757" }; + inline static const string black_star{ "\u2605" }; + inline static const string cross{ "\u2717" }; + inline static const string upward_arrow{ "\u27B6" }; + inline static const string down_arrow{ "\u27B4" }; + inline static const string equal_best{ check_mark }; + inline static const string better_best{ black_star }; + }; class ReportBase { public: - explicit ReportBase(json data_) { data = data_; }; + explicit ReportBase(json data_); virtual ~ReportBase() = default; void show(); protected: @@ -18,6 +30,11 @@ namespace platform { string fVector(const string& title, const json& data, const int width, const int precision); virtual void header() = 0; virtual void body() = 0; + virtual void showSummary() = 0; + string compareResult(const string& dataset, double result); + map summary; + double margin; + map meaning; }; }; #endif \ No newline at end of file diff --git a/src/Platform/ReportConsole.cc b/src/Platform/ReportConsole.cc index acbb602..0de1c11 100644 --- a/src/Platform/ReportConsole.cc +++ b/src/Platform/ReportConsole.cc @@ -11,11 +11,11 @@ namespace platform { string do_grouping() const { return "\03"; } }; - string ReportConsole::headerLine(const string& text) + string ReportConsole::headerLine(const string& text, int utf = 0) { int n = MAXL - text.length() - 3; n = n < 0 ? 0 : n; - return "* " + text + string(n, ' ') + "*\n"; + return "* " + text + string(n + utf, ' ') + "*\n"; } void ReportConsole::header() @@ -36,8 +36,8 @@ namespace platform { } void ReportConsole::body() { - cout << Colors::GREEN() << " # Dataset Sampl. Feat. Cls Nodes Edges States Score Time Hyperparameters" << endl; - cout << "=== ============================== ====== ===== === ========= ========= ========= =============== ================== ===============" << endl; + cout << Colors::GREEN() << " # Dataset Sampl. Feat. Cls Nodes Edges States Score Time Hyperparameters" << endl; + cout << "=== ========================= ====== ===== === ========= ========= ========= =============== =================== ====================" << endl; json lastResult; double totalScore = 0.0; bool odd = true; @@ -50,15 +50,17 @@ namespace platform { auto color = odd ? Colors::CYAN() : Colors::BLUE(); cout << color; cout << setw(3) << index++ << " "; - cout << setw(30) << left << r["dataset"].get() << " "; + cout << setw(25) << left << r["dataset"].get() << " "; cout << setw(6) << right << r["samples"].get() << " "; cout << setw(5) << right << r["features"].get() << " "; cout << setw(3) << right << r["classes"].get() << " "; cout << setw(9) << setprecision(2) << fixed << r["nodes"].get() << " "; cout << setw(9) << setprecision(2) << fixed << r["leaves"].get() << " "; cout << setw(9) << setprecision(2) << fixed << r["depth"].get() << " "; - cout << setw(8) << right << setprecision(6) << fixed << r["score"].get() << "±" << setw(6) << setprecision(4) << fixed << r["score_std"].get() << " "; - cout << setw(11) << right << setprecision(6) << fixed << r["time"].get() << "±" << setw(6) << setprecision(4) << fixed << r["time_std"].get() << " "; + cout << setw(8) << right << setprecision(6) << fixed << r["score"].get() << "±" << setw(6) << setprecision(4) << fixed << r["score_std"].get(); + const string status = compareResult(r["dataset"].get(), r["score"].get()); + cout << status; + cout << setw(12) << right << setprecision(6) << fixed << r["time"].get() << "±" << setw(6) << setprecision(4) << fixed << r["time_std"].get() << " "; try { cout << r["hyperparameters"].get(); } @@ -81,9 +83,21 @@ namespace platform { footer(totalScore); } } + void ReportConsole::showSummary() + { + for (const auto& item : summary) { + stringstream oss; + oss << setw(3) << left << item.first; + oss << setw(3) << right << item.second << " "; + oss << left << meaning.at(item.first); + cout << headerLine(oss.str(), 2); + } + } + void ReportConsole::footer(double totalScore) { cout << Colors::MAGENTA() << string(MAXL, '*') << endl; + showSummary(); auto score = data["score_name"].get(); if (score == BestResult::scoreName()) { stringstream oss; diff --git a/src/Platform/ReportConsole.h b/src/Platform/ReportConsole.h index b34e71f..7b3906c 100644 --- a/src/Platform/ReportConsole.h +++ b/src/Platform/ReportConsole.h @@ -7,17 +7,18 @@ namespace platform { using namespace std; - const int MAXL = 132; + const int MAXL = 133; class ReportConsole : public ReportBase { public: explicit ReportConsole(json data_, int index = -1) : ReportBase(data_), selectedIndex(index) {}; virtual ~ReportConsole() = default; private: int selectedIndex; - string headerLine(const string& text); + string headerLine(const string& text, int utf); void header() override; void body() override; void footer(double totalScore); + void showSummary(); }; }; #endif \ No newline at end of file diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index bfda442..c7d6ded 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -1,6 +1,5 @@ #include #include -#include "Datasets.h" #include "ReportExcel.h" #include "BestResult.h" @@ -20,7 +19,6 @@ namespace platform { colorTitle = 0xB1A0C7; colorOdd = 0xDCE6F1; colorEven = 0xFDE9D9; - margin = .1; // margin to add to ZeroR comparison createFile(); } @@ -308,43 +306,9 @@ namespace platform { footer(totalScore, row); } } - string ReportExcel::compareResult(const string& dataset, double result) - { - string status = " "; - if (data["score_name"].get() == "accuracy") { - auto dt = Datasets(Paths::datasets(), false); - dt.loadDataset(dataset); - auto numClasses = dt.getNClasses(dataset); - if (numClasses == 2) { - vector distribution = dt.getClassesCounts(dataset); - double nSamples = dt.getNSamples(dataset); - vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); - double mark = *maxValue / nSamples * (1 + margin); - if (mark > 1) { - mark = 0.9995; - } - status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; - auto item = summary.find(status); - if (item != summary.end()) { - summary[status]++; - } else { - summary[status] = 1; - } - } - } - return status; - } + void ReportExcel::showSummary() { - stringstream oss; - oss << "Better than ZeroR + " << setprecision(1) << fixed << margin * 100 << "%"; - - map meaning = { - {Symbols::equal_best, "Equal to best"}, - {Symbols::better_best, "Better than best"}, - {Symbols::cross, "Less than or equal to ZeroR"}, - {Symbols::upward_arrow, oss.str()} - }; for (const auto& item : summary) { worksheet_write_string(worksheet, row + 2, 1, item.first.c_str(), styles["summaryStyle"]); worksheet_write_number(worksheet, row + 2, 2, item.second, styles["summaryStyle"]); diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 6e0f6eb..289c808 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -3,22 +3,11 @@ #include #include "xlsxwriter.h" #include "ReportBase.h" -#include "Paths.h" #include "Colors.h" namespace platform { using namespace std; const int MAXLL = 128; - class Symbols { - public: - inline static const string check_mark{ "\u2714" }; - inline static const string exclamation{ "\u2757" }; - inline static const string black_star{ "\u2605" }; - inline static const string cross{ "\u2717" }; - inline static const string upward_arrow{ "\u27B6" }; - inline static const string down_arrow{ "\u27B4" }; - inline static const string equal_best{ check_mark }; - inline static const string better_best{ black_star }; - }; + class ReportExcel : public ReportBase { public: explicit ReportExcel(json data_, lxw_workbook* workbook); @@ -36,13 +25,11 @@ namespace platform { lxw_workbook* workbook; lxw_worksheet* worksheet; map styles; - map summary; int row; int normalSize; //font size for report body uint32_t colorTitle; uint32_t colorOdd; uint32_t colorEven; - double margin; const string fileName = "some_results.xlsx"; void header() override; void body() override; @@ -50,7 +37,6 @@ namespace platform { void createStyle(const string& name, lxw_format* style, bool odd); void addColor(lxw_format* style, bool odd); lxw_format* efectiveStyle(const string& name); - string compareResult(const string& dataset, double result); }; }; #endif // !REPORTEXCEL_H \ No newline at end of file From 03533461c824349e146eed3080b4ec480147d117 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 12:51:19 +0200 Subject: [PATCH 10/11] Add compare to best results in manage --- src/Platform/ReportBase.cc | 70 ++++++++++++++++++++++++++---------- src/Platform/ReportBase.h | 6 +++- src/Platform/ReportConsole.h | 2 +- src/Platform/ReportExcel.cc | 2 +- src/Platform/ReportExcel.h | 2 +- src/Platform/Results.cc | 6 ++-- src/Platform/Results.h | 7 +++- src/Platform/manage.cc | 5 ++- 8 files changed, 73 insertions(+), 27 deletions(-) diff --git a/src/Platform/ReportBase.cc b/src/Platform/ReportBase.cc index 2289640..6a5b885 100644 --- a/src/Platform/ReportBase.cc +++ b/src/Platform/ReportBase.cc @@ -6,7 +6,7 @@ namespace platform { - ReportBase::ReportBase(json data_) : margin(0.1), data(data_) + ReportBase::ReportBase(json data_, bool compare) : data(data_), compare(compare), margin(0.1) { stringstream oss; oss << "Better than ZeroR + " << setprecision(1) << fixed << margin * 100 << "%"; @@ -49,27 +49,59 @@ namespace platform { string ReportBase::compareResult(const string& dataset, double result) { string status = " "; - if (data["score_name"].get() == "accuracy") { - auto dt = Datasets(Paths::datasets(), false); - dt.loadDataset(dataset); - auto numClasses = dt.getNClasses(dataset); - if (numClasses == 2) { - vector distribution = dt.getClassesCounts(dataset); - double nSamples = dt.getNSamples(dataset); - vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); - double mark = *maxValue / nSamples * (1 + margin); - if (mark > 1) { - mark = 0.9995; - } - status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; - auto item = summary.find(status); - if (item != summary.end()) { - summary[status]++; - } else { - summary[status] = 1; + if (compare) { + double best = bestResult(dataset, data["model"].get()); + if (result == best) { + status = Symbols::equal_best; + } else if (result > best) { + status = Symbols::better_best; + } + } else { + if (data["score_name"].get() == "accuracy") { + auto dt = Datasets(Paths::datasets(), false); + dt.loadDataset(dataset); + auto numClasses = dt.getNClasses(dataset); + if (numClasses == 2) { + vector distribution = dt.getClassesCounts(dataset); + double nSamples = dt.getNSamples(dataset); + vector::iterator maxValue = max_element(distribution.begin(), distribution.end()); + double mark = *maxValue / nSamples * (1 + margin); + if (mark > 1) { + mark = 0.9995; + } + status = result < mark ? Symbols::cross : result > mark ? Symbols::upward_arrow : "="; } } } + if (status != " ") { + auto item = summary.find(status); + if (item != summary.end()) { + summary[status]++; + } else { + summary[status] = 1; + } + } return status; } + double ReportBase::bestResult(const string& dataset, const string& model) + { + double value = 0.0; + if (bestResults.size() == 0) { + // try to load the best results + string score = data["score_name"]; + replace(score.begin(), score.end(), '_', '-'); + string fileName = "best_results_" + score + "_" + model + ".json"; + ifstream resultData(Paths::results() + "/" + fileName); + if (resultData.is_open()) { + bestResults = json::parse(resultData); + } + } + try { + value = bestResults.at(dataset).at(0); + } + catch (exception) { + value = 1.0; + } + return value; + } } \ No newline at end of file diff --git a/src/Platform/ReportBase.h b/src/Platform/ReportBase.h index f587c2e..7695102 100644 --- a/src/Platform/ReportBase.h +++ b/src/Platform/ReportBase.h @@ -21,7 +21,7 @@ namespace platform { }; class ReportBase { public: - explicit ReportBase(json data_); + explicit ReportBase(json data_, bool compare); virtual ~ReportBase() = default; void show(); protected: @@ -35,6 +35,10 @@ namespace platform { map summary; double margin; map meaning; + private: + double bestResult(const string& dataset, const string& model); + bool compare; + json bestResults; }; }; #endif \ No newline at end of file diff --git a/src/Platform/ReportConsole.h b/src/Platform/ReportConsole.h index 7b3906c..3dcc719 100644 --- a/src/Platform/ReportConsole.h +++ b/src/Platform/ReportConsole.h @@ -10,7 +10,7 @@ namespace platform { const int MAXL = 133; class ReportConsole : public ReportBase { public: - explicit ReportConsole(json data_, int index = -1) : ReportBase(data_), selectedIndex(index) {}; + explicit ReportConsole(json data_, bool compare = false, int index = -1) : ReportBase(data_, compare), selectedIndex(index) {}; virtual ~ReportConsole() = default; private: int selectedIndex; diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index c7d6ded..ad816a2 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -13,7 +13,7 @@ namespace platform { string do_grouping() const { return "\03"; } }; - ReportExcel::ReportExcel(json data_, lxw_workbook* workbook) : ReportBase(data_), row(0), workbook(workbook) + ReportExcel::ReportExcel(json data_, bool compare, lxw_workbook* workbook) : ReportBase(data_, compare), row(0), workbook(workbook) { normalSize = 14; //font size for report body colorTitle = 0xB1A0C7; diff --git a/src/Platform/ReportExcel.h b/src/Platform/ReportExcel.h index 289c808..c5d462f 100644 --- a/src/Platform/ReportExcel.h +++ b/src/Platform/ReportExcel.h @@ -10,7 +10,7 @@ namespace platform { class ReportExcel : public ReportBase { public: - explicit ReportExcel(json data_, lxw_workbook* workbook); + explicit ReportExcel(json data_, bool compare, lxw_workbook* workbook); lxw_workbook* getWorkbook(); private: void writeString(int row, int col, const string& text, const string& style = ""); diff --git a/src/Platform/Results.cc b/src/Platform/Results.cc index e136a62..8568600 100644 --- a/src/Platform/Results.cc +++ b/src/Platform/Results.cc @@ -109,12 +109,12 @@ namespace platform { cout << Colors::YELLOW() << "Reporting " << files.at(index).getFilename() << endl; auto data = files.at(index).load(); if (excelReport) { - ReportExcel reporter(data, workbook); + ReportExcel reporter(data, compare, workbook); reporter.show(); openExcel = true; workbook = reporter.getWorkbook(); } else { - ReportConsole reporter(data); + ReportConsole reporter(data, compare); reporter.show(); } } @@ -150,6 +150,7 @@ namespace platform { if (all_of(line.begin(), line.end(), ::isdigit)) { int idx = stoi(line); if (indexList) { + // The value is about the files list index = idx; if (index >= 0 && index < files.size()) { report(index, false); @@ -157,6 +158,7 @@ namespace platform { continue; } } else { + // The value is about the result showed on screen showIndex(index, idx); continue; } diff --git a/src/Platform/Results.h b/src/Platform/Results.h index c418135..60748ba 100644 --- a/src/Platform/Results.h +++ b/src/Platform/Results.h @@ -35,7 +35,11 @@ namespace platform { }; class Results { public: - Results(const string& path, const int max, const string& model, const string& score, bool complete, bool partial) : path(path), max(max), model(model), scoreName(score), complete(complete), partial(partial) { load(); }; + Results(const string& path, const int max, const string& model, const string& score, bool complete, bool partial, bool compare) : + path(path), max(max), model(model), scoreName(score), complete(complete), partial(partial), compare(compare) + { + load(); + }; void manage(); private: string path; @@ -46,6 +50,7 @@ namespace platform { bool partial; bool indexList = true; bool openExcel = false; + bool compare; lxw_workbook* workbook = NULL; vector files; void load(); // Loads the list of results diff --git a/src/Platform/manage.cc b/src/Platform/manage.cc index aec19e7..cf699d6 100644 --- a/src/Platform/manage.cc +++ b/src/Platform/manage.cc @@ -14,6 +14,7 @@ argparse::ArgumentParser manageArguments(int argc, char** argv) program.add_argument("-s", "--score").default_value("any").help("Filter results of the score name supplied"); program.add_argument("--complete").help("Show only results with all datasets").default_value(false).implicit_value(true); program.add_argument("--partial").help("Show only partial results").default_value(false).implicit_value(true); + program.add_argument("--compare").help("Compare with best results").default_value(false).implicit_value(true); try { program.parse_args(argc, argv); auto number = program.get("number"); @@ -24,6 +25,7 @@ argparse::ArgumentParser manageArguments(int argc, char** argv) auto score = program.get("score"); auto complete = program.get("complete"); auto partial = program.get("partial"); + auto compare = program.get("compare"); } catch (const exception& err) { cerr << err.what() << endl; @@ -41,9 +43,10 @@ int main(int argc, char** argv) auto score = program.get("score"); auto complete = program.get("complete"); auto partial = program.get("partial"); + auto compare = program.get("compare"); if (complete) partial = false; - auto results = platform::Results(platform::Paths::results(), number, model, score, complete, partial); + auto results = platform::Results(platform::Paths::results(), number, model, score, complete, partial, compare); results.manage(); return 0; } From d8734ff0823f84310c14e43774fca02532d4071d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ricardo=20Monta=C3=B1ana?= Date: Wed, 20 Sep 2023 13:15:33 +0200 Subject: [PATCH 11/11] Separate contextual menu from general --- src/Platform/ReportExcel.cc | 4 ++-- src/Platform/Results.cc | 16 ++++++++++++++-- 2 files changed, 16 insertions(+), 4 deletions(-) diff --git a/src/Platform/ReportExcel.cc b/src/Platform/ReportExcel.cc index ad816a2..41891e9 100644 --- a/src/Platform/ReportExcel.cc +++ b/src/Platform/ReportExcel.cc @@ -93,12 +93,12 @@ namespace platform { format_set_num_format(style, "#,##0.000000"); } else if (name == "ints") { format_set_font_size(style, normalSize); - format_set_num_format(style, "###,###"); + format_set_num_format(style, "###,##0"); format_set_border(style, LXW_BORDER_THIN); } else if (name == "floats") { format_set_border(style, LXW_BORDER_THIN); format_set_font_size(style, normalSize); - format_set_num_format(style, "#,###.00"); + format_set_num_format(style, "#,##0.00"); } } diff --git a/src/Platform/Results.cc b/src/Platform/Results.cc index 8568600..725952c 100644 --- a/src/Platform/Results.cc +++ b/src/Platform/Results.cc @@ -126,7 +126,7 @@ namespace platform { return; } cout << Colors::YELLOW() << "Showing " << files.at(index).getFilename() << endl; - ReportConsole reporter(data, idx); + ReportConsole reporter(data, compare, idx); reporter.show(); } void Results::menu() @@ -134,9 +134,21 @@ namespace platform { char option; int index; bool finished = false; + string color, context; string filename, line, options = "qldhsre"; while (!finished) { - cout << Colors::RESET() << "Choose option (quit='q', list='l', delete='d', hide='h', sort='s', report='r', excel='e'): "; + if (indexList) { + color = Colors::GREEN(); + context = " (quit='q', list='l', delete='d', hide='h', sort='s', report='r', excel='e'): "; + options = "qldhsre"; + } else { + color = Colors::MAGENTA(); + context = " (quit='q', list='l'): "; + options = "ql"; + } + cout << Colors::RESET() << color; + + cout << "Choose option " << context; getline(cin, line); if (line.size() == 0) continue;