Line data Source code
1 : //* This file is part of the MOOSE framework 2 : //* https://mooseframework.inl.gov 3 : //* 4 : //* All rights reserved, see COPYRIGHT for full restrictions 5 : //* https://github.com/idaholab/moose/blob/master/COPYRIGHT 6 : //* 7 : //* Licensed under LGPL 2.1, please see LICENSE for details 8 : //* https://www.gnu.org/licenses/lgpl-2.1.html 9 : 10 : #include "JsonInputFileFormatter.h" 11 : #include "MooseUtils.h" 12 : 13 : #include <vector> 14 : 15 14 : JsonInputFileFormatter::JsonInputFileFormatter() : _spaces(2), _level(0) {} 16 : 17 : std::string 18 14 : JsonInputFileFormatter::toString(const nlohmann::json & root) 19 : { 20 14 : _stream.clear(); 21 28 : _stream.str(""); 22 534 : for (auto && el : root["blocks"].items()) 23 534 : addBlock(el.key(), el.value(), true); 24 14 : return _stream.str(); 25 : } 26 : 27 : void 28 975158 : JsonInputFileFormatter::addLine(const std::string & line, 29 : size_t max_line_len, 30 : const std::string & comment) 31 : { 32 975158 : if (line.empty() && comment.empty()) 33 : { 34 17890 : _stream << "\n"; 35 53670 : return; 36 : } 37 : 38 1914536 : std::string indent(_level * _spaces, ' '); 39 957268 : auto doc = MooseUtils::trim(comment); 40 957268 : if (doc.empty()) // Not comment so just print out the line normally 41 : { 42 35780 : _stream << indent << line << "\n"; 43 35780 : return; 44 : } 45 : 46 : // We have a comment so we need to break it up over multiple lines if necessary 47 : // and make sure that they all start at the same spot. 48 921488 : _stream << indent << line; 49 921488 : std::vector<std::string> elements; 50 : 51 : // if the line is empty we can just start the comment right away 52 921488 : int extra = 1; 53 921488 : if (line.empty()) 54 619749 : extra = 0; 55 : 56 : // be careful of really long lines. 57 921488 : int len = 100 - max_line_len - indent.size(); 58 921488 : if (len < 0) 59 3136 : len = 20; 60 : 61 1842976 : MooseUtils::tokenize(doc, elements, len, " \t"); 62 921488 : std::string first(max_line_len - line.size() + extra, ' '); 63 921488 : _stream << first << "# " << elements[0] << "\n"; 64 921488 : std::string cindent(max_line_len + indent.size() + extra, ' '); 65 1698680 : for (size_t i = 1; i < elements.size(); ++i) 66 777192 : _stream << cindent << "# " << elements[i] << "\n"; 67 : 68 921488 : _stream << std::flush; 69 993048 : } 70 : 71 : void 72 17419 : JsonInputFileFormatter::addBlock(const std::string & name, 73 : const nlohmann::json & block, 74 : bool toplevel) 75 : { 76 52257 : addLine(""); 77 17419 : if (toplevel) 78 1040 : addLine("[" + name + "]"); 79 : else 80 33798 : addLine("[./" + name + "]"); 81 : 82 17419 : _level++; 83 18567 : std::string desc = block.contains("description") ? nlohmann::to_string(block["description"]) : ""; 84 17419 : if (!desc.empty()) 85 32542 : addLine("", 0, desc); 86 : 87 17419 : if (block.contains("parameters")) 88 16271 : addParameters(block["parameters"]); 89 : 90 17419 : if (block.contains("actions")) 91 : { 92 : // there could be duplicate parameters across actions, last one wins 93 716 : nlohmann::json all_params; 94 716 : auto & actions = block["actions"]; 95 1523 : for (auto && el : actions.items()) 96 : { 97 807 : auto & params = el.value()["parameters"]; 98 6275 : for (auto && param_el : params.items()) 99 6275 : all_params[param_el.key()] = param_el.value(); 100 716 : } 101 716 : addParameters(all_params); 102 716 : } 103 : 104 17419 : if (block.contains("star")) 105 1380 : addBlock("*", block["star"]); 106 : 107 34838 : addTypes("subblock_types", block); 108 17419 : addTypes("types", block); 109 : 110 17419 : if (block.contains("subblocks")) 111 : { 112 77 : auto & subblocks = block["subblocks"]; 113 77 : if (!subblocks.is_null()) 114 245 : for (auto && el : subblocks.items()) 115 245 : addBlock(el.key(), el.value()); 116 : } 117 : 118 17419 : _level--; 119 17419 : if (toplevel) 120 2080 : addLine("[]"); 121 : else 122 67596 : addLine("[../]"); 123 17419 : } 124 : 125 : void 126 34838 : JsonInputFileFormatter::addTypes(const std::string & key, const nlohmann::json & block) 127 : { 128 34838 : if (!block.contains(key)) 129 34122 : return; 130 716 : auto & types = block[key]; 131 716 : if (types.is_null()) 132 245 : return; 133 : 134 1884 : addLine(""); 135 1413 : addLine("[./<types>]"); 136 471 : _level++; 137 16742 : for (auto && el : types.items()) 138 16742 : addBlock("<" + el.key() + ">", el.value()); 139 471 : _level--; 140 1884 : addLine("[../]"); 141 : } 142 : 143 : void 144 16987 : JsonInputFileFormatter::addParameters(const nlohmann::json & params) 145 : { 146 16987 : size_t max_name = 0; 147 318726 : for (auto & el : params.items()) 148 301739 : if (el.key().size() > max_name) 149 54904 : max_name = el.key().size(); 150 : 151 16987 : size_t max_len = 0; 152 16987 : std::map<std::string, std::string> lines; 153 318726 : for (auto && el : params.items()) 154 : { 155 301739 : auto & name = el.key(); 156 301739 : auto & param = el.value(); 157 : auto def = 158 699544 : param.contains("default") ? MooseUtils::trim(nlohmann::to_string(param["default"])) : ""; 159 301739 : if (!def.empty()) 160 205673 : def = def.substr(1, def.size() - 2); 161 301739 : if (def.find(' ') != std::string::npos) 162 5966 : def = "'" + def + "'"; 163 301739 : std::string indent(max_name - name.size(), ' '); 164 301739 : std::string required; 165 301739 : if (param["required"]) 166 28368 : required = "(required)"; 167 301739 : if (def.size() == 0 && required.size() == 0) 168 96721 : def = "(no_default)"; 169 301739 : std::string l = name + indent + " = " + def + required; 170 301739 : if (l.size() > max_len) 171 45628 : max_len = l.size(); 172 301739 : lines[name] = l; 173 318726 : } 174 318726 : for (auto & el : params.items()) 175 : { 176 301739 : auto & name = el.key(); 177 301739 : auto & param = el.value(); 178 301739 : auto & l = lines[name]; 179 301739 : auto desc = nlohmann::to_string(param["description"]); 180 301739 : addLine(l, max_len, desc); 181 : 182 301739 : const auto doc_unit = nlohmann::to_string(param["doc_unit"]); 183 301739 : if (!doc_unit.empty()) 184 603478 : addLine("", max_len + 1, 185 603478 : "Unit: " + doc_unit); // a +1 to account for an empty line 186 : 187 301739 : const auto group = nlohmann::to_string(param["group_name"]); 188 301739 : if (!group.empty()) 189 603478 : addLine("", max_len + 1, 190 603478 : "Group: " + group); // a +1 to account for an empty line 191 318726 : } 192 16987 : }