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/log_data_table.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 'src/log_data_table.cc')
-rw-r--r-- | src/log_data_table.cc | 196 |
1 files changed, 196 insertions, 0 deletions
diff --git a/src/log_data_table.cc b/src/log_data_table.cc new file mode 100644 index 0000000..7a7c537 --- /dev/null +++ b/src/log_data_table.cc @@ -0,0 +1,196 @@ +/** + * Copyright (c) 2020, 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 "log_data_table.hh" + +#include "config.h" +#include "scn/scn.h" + +log_data_table::log_data_table(logfile_sub_source& lss, + log_vtab_manager& lvm, + content_line_t template_line, + intern_string_t table_name) + : log_vtab_impl(table_name), ldt_log_source(lss), + ldt_template_line(template_line) +{ + auto lf = lss.find(template_line); + auto format = lf->get_format(); + + this->vi_supports_indexes = false; + this->ldt_format_impl = lvm.lookup_impl(format->get_name()); + this->get_columns_int(); +} + +void +log_data_table::get_columns_int() +{ + auto& cols = this->ldt_cols; + auto& metas = this->ldt_value_metas; + content_line_t cl_copy = this->ldt_template_line; + auto lf = this->ldt_log_source.find(cl_copy); + struct line_range body; + string_attrs_t sa; + logline_value_vector line_values; + auto format = lf->get_format(); + + if (this->ldt_format_impl != nullptr) { + this->ldt_format_impl->get_columns(cols); + } + lf->read_full_message(lf->begin() + cl_copy, line_values.lvv_sbr); + line_values.lvv_sbr.erase_ansi(); + format->annotate(cl_copy, sa, line_values, false); + body = find_string_attr_range(sa, &SA_BODY); + if (body.lr_end == -1) { + this->ldt_schema_id.clear(); + return; + } + + data_scanner ds(line_values.lvv_sbr, body.lr_start, body.lr_end); + data_parser dp(&ds); + column_namer cn{column_namer::language::SQL}; + + dp.parse(); + + for (auto pair_iter = dp.dp_pairs.begin(); pair_iter != dp.dp_pairs.end(); + ++pair_iter) + { + std::string key_str + = dp.get_element_string(pair_iter->e_sub_elements->front()); + auto colname = cn.add_column(key_str).to_string(); + int sql_type = SQLITE3_TEXT; + value_kind_t kind = value_kind_t::VALUE_TEXT; + std::string collator; + + switch (pair_iter->e_sub_elements->back().value_token()) { + case DT_IPV4_ADDRESS: + case DT_IPV6_ADDRESS: + collator = "ipaddress"; + break; + + case DT_NUMBER: + sql_type = SQLITE_FLOAT; + kind = value_kind_t::VALUE_FLOAT; + break; + + default: + collator = "naturalnocase"; + break; + } + metas.emplace_back( + intern_string::lookup(colname), kind, cols.size(), format.get()); + cols.emplace_back(colname, sql_type, collator); + } + this->ldt_schema_id = dp.dp_schema_id; +} + +bool +log_data_table::next(log_cursor& lc, logfile_sub_source& lss) +{ + if (lc.is_eof()) { + return true; + } + + content_line_t cl; + + cl = lss.at(lc.lc_curr_line); + std::shared_ptr<logfile> lf = lss.find(cl); + auto lf_iter = lf->begin() + cl; + + if (!lf_iter->is_message()) { + return false; + } + + if (lf_iter->has_schema() && !lf_iter->match_schema(this->ldt_schema_id)) { + return false; + } + + string_attrs_t sa; + struct line_range body; + logline_value_vector line_values; + + lf->read_full_message(lf_iter, line_values.lvv_sbr); + line_values.lvv_sbr.erase_ansi(); + lf->get_format()->annotate(cl, sa, line_values, false); + body = find_string_attr_range(sa, &SA_BODY); + if (body.lr_end == -1) { + return false; + } + + data_scanner ds(line_values.lvv_sbr, body.lr_start, body.lr_end); + data_parser dp(&ds); + dp.parse(); + + lf_iter->set_schema(dp.dp_schema_id); + + /* The cached schema ID in the log line is not complete, so we still */ + /* need to check for a full match. */ + if (dp.dp_schema_id != this->ldt_schema_id) { + return false; + } + + this->ldt_pairs.clear(); + this->ldt_pairs.swap(dp.dp_pairs, __FILE__, __LINE__); + + return true; +} + +void +log_data_table::extract(logfile* lf, + uint64_t line_number, + logline_value_vector& values) +{ + auto& line = values.lvv_sbr; + auto meta_iter = this->ldt_value_metas.begin(); + + this->ldt_format_impl->extract(lf, line_number, values); + for (const auto& ldt_pair : this->ldt_pairs) { + const auto& pvalue = ldt_pair.get_pair_value(); + auto lr = line_range{ + pvalue.e_capture.c_begin, + pvalue.e_capture.c_end, + }; + + switch (pvalue.value_token()) { + case DT_NUMBER: { + auto num_view = line.to_string_view(lr); + auto num_scan_res = scn::scan_value<double>(num_view); + auto num = num_scan_res ? num_scan_res.value() : 0.0; + + values.lvv_values.emplace_back(*meta_iter, num); + break; + } + + default: { + values.lvv_values.emplace_back(*meta_iter, line, lr); + break; + } + } + ++meta_iter; + } +} |