diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-04 17:44:55 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-04 17:44:55 +0000 |
commit | 5068d34c08f951a7ea6257d305a1627b09a95817 (patch) | |
tree | 08213e2be853396a3b07ce15dbe222644dcd9a89 /src/pretty_printer.cc | |
parent | Initial commit. (diff) | |
download | lnav-5068d34c08f951a7ea6257d305a1627b09a95817.tar.xz lnav-5068d34c08f951a7ea6257d305a1627b09a95817.zip |
Adding upstream version 0.11.1.upstream/0.11.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r-- | src/pretty_printer.cc | 378 |
1 files changed, 378 insertions, 0 deletions
diff --git a/src/pretty_printer.cc b/src/pretty_printer.cc new file mode 100644 index 0000000..2ff3e11 --- /dev/null +++ b/src/pretty_printer.cc @@ -0,0 +1,378 @@ +/** + * Copyright (c) 2015, Timothy Stack + * + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * * Neither the name of Timothy Stack nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ''AS IS'' AND ANY + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND + * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include "pretty_printer.hh" + +#include "base/string_util.hh" +#include "config.h" + +void +pretty_printer::append_to(attr_line_t& al) +{ + if (this->pp_scanner->get_init_offset() > 0) { + data_scanner::capture_t leading_cap = { + 0, + this->pp_scanner->get_init_offset(), + }; + + // this->pp_stream << pi.get_substr(&leading_cap); + this->pp_values.emplace_back(DT_WORD, leading_cap); + } + + this->pp_scanner->reset(); + while (true) { + auto tok_res = this->pp_scanner->tokenize2(); + if (!tok_res) { + break; + } + + element el(tok_res->tr_token, tok_res->tr_capture); + + switch (el.e_token) { + case DT_XML_DECL_TAG: + case DT_XML_EMPTY_TAG: + if (this->pp_is_xml && this->pp_line_length > 0) { + this->start_new_line(); + } + this->pp_values.emplace_back(el); + if (this->pp_is_xml) { + this->start_new_line(); + } + continue; + case DT_XML_OPEN_TAG: + if (this->pp_is_xml) { + this->start_new_line(); + this->write_element(el); + this->pp_interval_state.back().is_start + = this->pp_stream.tellp(); + this->pp_interval_state.back().is_name + = tok_res->to_string(); + this->descend(); + } else { + this->pp_values.emplace_back(el); + } + continue; + case DT_XML_CLOSE_TAG: + this->flush_values(); + this->ascend(); + this->append_child_node(); + this->write_element(el); + this->start_new_line(); + continue; + case DT_LCURLY: + case DT_LSQUARE: + case DT_LPAREN: + this->flush_values(true); + this->pp_values.emplace_back(el); + this->descend(); + this->pp_interval_state.back().is_start + = this->pp_stream.tellp(); + continue; + case DT_RCURLY: + case DT_RSQUARE: + case DT_RPAREN: + this->flush_values(); + if (this->pp_body_lines.top()) { + this->start_new_line(); + } + this->ascend(); + this->write_element(el); + continue; + case DT_COMMA: + if (this->pp_depth > 0) { + this->flush_values(true); + if (!this->pp_is_xml) { + this->append_child_node(); + } + this->write_element(el); + this->start_new_line(); + this->pp_interval_state.back().is_start + = this->pp_stream.tellp(); + continue; + } + break; + case DT_WHITE: + if (this->pp_values.empty() && this->pp_depth == 0 + && this->pp_line_length == 0) + { + this->pp_leading_indent = el.e_capture.length(); + continue; + } + break; + default: + break; + } + this->pp_values.emplace_back(el); + } + while (this->pp_depth > 0) { + this->ascend(); + } + this->flush_values(); + + attr_line_t combined; + combined.get_string() = this->pp_stream.str(); + combined.get_attrs() = this->pp_attrs; + + if (!al.empty()) { + al.append("\n"); + } + al.append(combined); + + if (this->pp_hier_stage != nullptr) { + this->pp_hier_stage->hn_parent = this->pp_hier_nodes.back().get(); + this->pp_hier_nodes.back()->hn_children.push_back( + std::move(this->pp_hier_stage)); + } + this->pp_hier_stage = std::move(this->pp_hier_nodes.back()); + this->pp_hier_nodes.pop_back(); + if (this->pp_hier_stage->hn_children.size() == 1 + && this->pp_hier_stage->hn_named_children.empty()) + { + this->pp_hier_stage + = std::move(this->pp_hier_stage->hn_children.front()); + this->pp_hier_stage->hn_parent = nullptr; + } +} + +void +pretty_printer::write_element(const pretty_printer::element& el) +{ + if (this->pp_leading_indent == 0 && this->pp_line_length == 0 + && el.e_token == DT_WHITE) + { + if (this->pp_depth == 0) { + this->pp_soft_indent += el.e_capture.length(); + } + return; + } + if (((this->pp_leading_indent == 0) + || (this->pp_line_length <= this->pp_leading_indent)) + && el.e_token == DT_LINE) + { + this->pp_soft_indent = 0; + if (this->pp_line_length > 0) { + this->pp_line_length = 0; + this->pp_stream << std::endl; + this->pp_body_lines.top() += 1; + } + return; + } + if (this->pp_line_length == 0) { + this->append_indent(); + } + ssize_t start_size = this->pp_stream.tellp(); + if (el.e_token == DT_QUOTED_STRING) { + auto_mem<char> unquoted_str((char*) malloc(el.e_capture.length() + 1)); + const char* start + = this->pp_scanner->to_string_fragment(el.e_capture).data(); + auto unq_len = unquote(unquoted_str.in(), start, el.e_capture.length()); + data_scanner ds( + string_fragment::from_bytes(unquoted_str.in(), unq_len)); + string_attrs_t sa; + pretty_printer str_pp( + &ds, sa, this->pp_leading_indent + this->pp_depth * 4); + attr_line_t result; + str_pp.append_to(result); + if (result.get_string().find('\n') != std::string::npos) { + switch (start[0]) { + case 'r': + case 'u': + this->pp_stream << start[0]; + this->pp_stream << start[1] << start[1]; + break; + default: + this->pp_stream << start[0] << start[0]; + break; + } + this->pp_stream << std::endl << result.get_string(); + if (result.empty() || result.get_string().back() != '\n') { + this->pp_stream << std::endl; + } + this->pp_stream << start[el.e_capture.length() - 1] + << start[el.e_capture.length() - 1]; + } else { + this->pp_stream + << this->pp_scanner->to_string_fragment(el.e_capture); + } + } else { + this->pp_stream << this->pp_scanner->to_string_fragment(el.e_capture); + int shift_amount + = start_size - el.e_capture.c_begin - this->pp_shift_accum; + shift_string_attrs(this->pp_attrs, el.e_capture.c_begin, shift_amount); + this->pp_shift_accum = start_size - el.e_capture.c_begin; + } + this->pp_line_length += el.e_capture.length(); + if (el.e_token == DT_LINE) { + this->pp_line_length = 0; + this->pp_body_lines.top() += 1; + } +} + +void +pretty_printer::append_indent() +{ + this->pp_stream << std::string( + this->pp_leading_indent + this->pp_soft_indent, ' '); + this->pp_soft_indent = 0; + if (this->pp_stream.tellp() == this->pp_leading_indent) { + return; + } + for (int lpc = 0; lpc < this->pp_depth; lpc++) { + this->pp_stream << " "; + } +} + +bool +pretty_printer::flush_values(bool start_on_depth) +{ + nonstd::optional<data_scanner::capture_t> last_key; + bool retval = false; + + while (!this->pp_values.empty()) { + { + auto& el = this->pp_values.front(); + this->write_element(this->pp_values.front()); + switch (el.e_token) { + case DT_SYMBOL: + case DT_CONSTANT: + case DT_WORD: + case DT_QUOTED_STRING: + last_key = el.e_capture; + break; + case DT_COLON: + case DT_EQUALS: + if (last_key) { + this->pp_interval_state.back().is_name + = this->pp_scanner + ->to_string_fragment(last_key.value()) + .to_string(); + if (!this->pp_interval_state.back().is_name.empty()) { + this->pp_interval_state.back().is_start + = static_cast<ssize_t>(this->pp_stream.tellp()); + } + last_key = nonstd::nullopt; + } + break; + default: + break; + } + if (start_on_depth + && (el.e_token == DT_LSQUARE || el.e_token == DT_LCURLY)) + { + if (this->pp_line_length > 0) { + this->pp_stream << std::endl; + } + this->pp_line_length = 0; + } + } + this->pp_values.pop_front(); + retval = true; + } + return retval; +} + +void +pretty_printer::start_new_line() +{ + bool has_output; + + if (this->pp_line_length > 0) { + this->pp_stream << std::endl; + this->pp_line_length = 0; + } + has_output = this->flush_values(); + if (has_output && this->pp_line_length > 0) { + this->pp_stream << std::endl; + } + this->pp_line_length = 0; + this->pp_body_lines.top() += 1; +} + +void +pretty_printer::ascend() +{ + if (this->pp_depth > 0) { + int lines = this->pp_body_lines.top(); + this->pp_depth -= 1; + this->pp_body_lines.pop(); + this->pp_body_lines.top() += lines; + + if (!this->pp_is_xml) { + this->append_child_node(); + } + this->pp_interval_state.pop_back(); + this->pp_hier_stage = std::move(this->pp_hier_nodes.back()); + this->pp_hier_nodes.pop_back(); + } else { + this->pp_body_lines.top() = 0; + } +} + +void +pretty_printer::descend() +{ + this->pp_depth += 1; + this->pp_body_lines.push(0); + this->pp_interval_state.resize(this->pp_depth + 1); + this->pp_hier_nodes.push_back( + std::make_unique<lnav::document::hier_node>()); +} + +void +pretty_printer::append_child_node() +{ + auto& ivstate = this->pp_interval_state.back(); + if (!ivstate.is_start) { + return; + } + + auto* top_node = this->pp_hier_nodes.back().get(); + auto new_key = ivstate.is_name.empty() + ? lnav::document::section_key_t{top_node->hn_children.size()} + : lnav::document::section_key_t{ivstate.is_name}; + this->pp_intervals.emplace_back( + ivstate.is_start.value(), + static_cast<ssize_t>(this->pp_stream.tellp()), + new_key); + auto new_node = this->pp_hier_stage != nullptr + ? std::move(this->pp_hier_stage) + : std::make_unique<lnav::document::hier_node>(); + auto* retval = new_node.get(); + new_node->hn_parent = top_node; + new_node->hn_start = this->pp_intervals.back().start; + if (!ivstate.is_name.empty()) { + top_node->hn_named_children.insert({ + ivstate.is_name, + retval, + }); + } + top_node->hn_children.emplace_back(std::move(new_node)); + ivstate.is_start = nonstd::nullopt; + ivstate.is_name.clear(); +} |