summaryrefslogtreecommitdiffstats
path: root/src/logfile_sub_source.hh
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 17:44:55 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 17:44:55 +0000
commit5068d34c08f951a7ea6257d305a1627b09a95817 (patch)
tree08213e2be853396a3b07ce15dbe222644dcd9a89 /src/logfile_sub_source.hh
parentInitial commit. (diff)
downloadlnav-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/logfile_sub_source.hh')
-rw-r--r--src/logfile_sub_source.hh1021
1 files changed, 1021 insertions, 0 deletions
diff --git a/src/logfile_sub_source.hh b/src/logfile_sub_source.hh
new file mode 100644
index 0000000..712f220
--- /dev/null
+++ b/src/logfile_sub_source.hh
@@ -0,0 +1,1021 @@
+/**
+ * Copyright (c) 2007-2012, 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.
+ *
+ * @file logfile_sub_source.hh
+ */
+
+#ifndef logfile_sub_source_hh
+#define logfile_sub_source_hh
+
+#include <array>
+#include <list>
+#include <map>
+#include <sstream>
+#include <utility>
+#include <vector>
+
+#include <limits.h>
+
+#include "base/lnav.console.hh"
+#include "base/lnav_log.hh"
+#include "base/time_util.hh"
+#include "big_array.hh"
+#include "bookmarks.hh"
+#include "document.sections.hh"
+#include "filter_observer.hh"
+#include "lnav_config_fwd.hh"
+#include "log_accel.hh"
+#include "log_format.hh"
+#include "logfile.hh"
+#include "strong_int.hh"
+#include "textview_curses.hh"
+
+STRONG_INT_TYPE(uint64_t, content_line);
+
+struct sqlite3_stmt;
+extern "C"
+{
+int sqlite3_finalize(sqlite3_stmt* pStmt);
+}
+
+class logfile_sub_source;
+
+class index_delegate {
+public:
+ virtual ~index_delegate() = default;
+
+ virtual void index_start(logfile_sub_source& lss) {}
+
+ virtual void index_line(logfile_sub_source& lss,
+ logfile* lf,
+ logfile::iterator ll)
+ {
+ }
+
+ virtual void index_complete(logfile_sub_source& lss) {}
+};
+
+class pcre_filter : public text_filter {
+public:
+ pcre_filter(type_t type,
+ const std::string& id,
+ size_t index,
+ std::shared_ptr<lnav::pcre2pp::code> code)
+ : text_filter(type, filter_lang_t::REGEX, id, index),
+ pf_pcre(std::move(code))
+ {
+ }
+
+ ~pcre_filter() override = default;
+
+ bool matches(const logfile& lf,
+ logfile::const_iterator ll,
+ shared_buffer_ref& line) override
+ {
+ return this->pf_pcre->find_in(line.to_string_fragment())
+ .ignore_error()
+ .has_value();
+ }
+
+ std::string to_command() const override
+ {
+ return (this->lf_type == text_filter::INCLUDE ? "filter-in "
+ : "filter-out ")
+ + this->lf_id;
+ }
+
+protected:
+ std::shared_ptr<lnav::pcre2pp::code> pf_pcre;
+};
+
+class sql_filter : public text_filter {
+public:
+ sql_filter(logfile_sub_source& lss,
+ std::string stmt_str,
+ sqlite3_stmt* stmt)
+ : text_filter(EXCLUDE, filter_lang_t::SQL, std::move(stmt_str), 0),
+ sf_log_source(lss)
+ {
+ this->sf_filter_stmt = stmt;
+ }
+
+ bool matches(const logfile& lf,
+ logfile::const_iterator ll,
+ shared_buffer_ref& line) override;
+
+ std::string to_command() const override;
+
+ auto_mem<sqlite3_stmt> sf_filter_stmt{sqlite3_finalize};
+ logfile_sub_source& sf_log_source;
+};
+
+class log_location_history : public location_history {
+public:
+ explicit log_location_history(logfile_sub_source& lss)
+ : llh_history(std::begin(this->llh_backing),
+ std::end(this->llh_backing)),
+ llh_log_source(lss)
+ {
+ }
+
+ ~log_location_history() override = default;
+
+ void loc_history_append(vis_line_t top) override;
+
+ nonstd::optional<vis_line_t> loc_history_back(
+ vis_line_t current_top) override;
+
+ nonstd::optional<vis_line_t> loc_history_forward(
+ vis_line_t current_top) override;
+
+private:
+ nonstd::ring_span<content_line_t> llh_history;
+ logfile_sub_source& llh_log_source;
+ content_line_t llh_backing[MAX_SIZE];
+};
+
+class logline_window {
+public:
+ logline_window(logfile_sub_source& lss,
+ vis_line_t start_vl,
+ vis_line_t end_vl)
+ : lw_source(lss), lw_start_line(start_vl), lw_end_line(end_vl)
+ {
+ }
+
+ class iterator;
+
+ class logmsg_info {
+ public:
+ logmsg_info(logfile_sub_source& lss, vis_line_t vl);
+
+ vis_line_t get_vis_line() const { return this->li_line; }
+
+ const logline& get_logline() const { return *this->li_logline; }
+
+ const string_attrs_t& get_attrs() const
+ {
+ this->load_msg();
+ return this->li_string_attrs;
+ }
+
+ const logline_value_vector& get_values() const
+ {
+ this->load_msg();
+ return this->li_line_values;
+ }
+
+ std::string to_string(const struct line_range& lr) const;
+
+ private:
+ friend iterator;
+
+ void next_msg();
+ void load_msg() const;
+
+ logfile_sub_source& li_source;
+ vis_line_t li_line;
+ logfile* li_file{nullptr};
+ logfile::const_iterator li_logline;
+ mutable string_attrs_t li_string_attrs;
+ mutable logline_value_vector li_line_values;
+ };
+
+ class iterator {
+ public:
+ iterator(logfile_sub_source& lss, vis_line_t vl) : i_info(lss, vl) {}
+
+ iterator& operator++();
+
+ bool operator!=(const iterator& rhs) const
+ {
+ return this->i_info.get_vis_line() != rhs.i_info.get_vis_line();
+ }
+
+ const logmsg_info& operator*() const { return this->i_info; }
+
+ private:
+ logmsg_info i_info;
+ };
+
+ iterator begin();
+
+ iterator end();
+
+private:
+ logfile_sub_source& lw_source;
+ vis_line_t lw_start_line;
+ vis_line_t lw_end_line;
+};
+
+/**
+ * Delegate class that merges the contents of multiple log files into a single
+ * source of data for a text view.
+ */
+class logfile_sub_source
+ : public text_sub_source
+ , public text_time_translator
+ , public list_input_delegate {
+public:
+ const static bookmark_type_t BM_ERRORS;
+ const static bookmark_type_t BM_WARNINGS;
+ const static bookmark_type_t BM_FILES;
+
+ virtual void text_filters_changed();
+
+ logfile_sub_source();
+
+ ~logfile_sub_source() = default;
+
+ void toggle_time_offset()
+ {
+ this->lss_flags ^= F_TIME_OFFSET;
+ this->clear_line_size_cache();
+ }
+
+ void increase_line_context()
+ {
+ auto old_flags = this->lss_flags;
+
+ if (this->lss_flags & F_FILENAME) {
+ // Nothing to do
+ } else if (this->lss_flags & F_BASENAME) {
+ this->lss_flags &= ~F_NAME_MASK;
+ this->lss_flags |= F_FILENAME;
+ } else {
+ this->lss_flags |= F_BASENAME;
+ }
+ if (old_flags != this->lss_flags) {
+ this->clear_line_size_cache();
+ }
+ }
+
+ bool decrease_line_context()
+ {
+ auto old_flags = this->lss_flags;
+
+ if (this->lss_flags & F_FILENAME) {
+ this->lss_flags &= ~F_NAME_MASK;
+ this->lss_flags |= F_BASENAME;
+ } else if (this->lss_flags & F_BASENAME) {
+ this->lss_flags &= ~F_NAME_MASK;
+ }
+ if (old_flags != this->lss_flags) {
+ this->clear_line_size_cache();
+
+ return true;
+ }
+
+ return false;
+ }
+
+ size_t get_filename_offset() const
+ {
+ if (this->lss_flags & F_FILENAME) {
+ return this->lss_filename_width;
+ } else if (this->lss_flags & F_BASENAME) {
+ return this->lss_basename_width;
+ }
+
+ return 0;
+ }
+
+ void set_time_offset(bool enabled)
+ {
+ if (enabled)
+ this->lss_flags |= F_TIME_OFFSET;
+ else
+ this->lss_flags &= ~F_TIME_OFFSET;
+ this->clear_line_size_cache();
+ }
+
+ bool is_time_offset_enabled() const
+ {
+ return (bool) (this->lss_flags & F_TIME_OFFSET);
+ }
+
+ bool is_filename_enabled() const
+ {
+ return (bool) (this->lss_flags & F_FILENAME);
+ }
+
+ bool is_basename_enabled() const
+ {
+ return (bool) (this->lss_flags & F_BASENAME);
+ }
+
+ log_level_t get_min_log_level() const { return this->lss_min_log_level; }
+
+ void set_force_rebuild() { this->lss_force_rebuild = true; }
+
+ void set_min_log_level(log_level_t level)
+ {
+ if (this->lss_min_log_level != level) {
+ this->lss_min_log_level = level;
+ this->text_filters_changed();
+ }
+ }
+
+ bool get_min_log_time(struct timeval& tv_out) const
+ {
+ tv_out = this->lss_min_log_time;
+ return (this->lss_min_log_time.tv_sec != 0
+ || this->lss_min_log_time.tv_usec != 0);
+ }
+
+ void set_min_log_time(const struct timeval& tv)
+ {
+ if (this->lss_min_log_time != tv) {
+ this->lss_min_log_time = tv;
+ this->text_filters_changed();
+ }
+ }
+
+ bool get_max_log_time(struct timeval& tv_out) const
+ {
+ tv_out = this->lss_max_log_time;
+ return (this->lss_max_log_time.tv_sec
+ != std::numeric_limits<time_t>::max()
+ || this->lss_max_log_time.tv_usec != 0);
+ }
+
+ void set_max_log_time(struct timeval& tv)
+ {
+ if (this->lss_max_log_time != tv) {
+ this->lss_max_log_time = tv;
+ this->text_filters_changed();
+ }
+ }
+
+ void clear_min_max_log_times()
+ {
+ if (this->lss_min_log_time.tv_sec != 0
+ || this->lss_min_log_time.tv_usec != 0
+ || this->lss_max_log_time.tv_sec
+ != std::numeric_limits<time_t>::max()
+ || this->lss_max_log_time.tv_usec != 0)
+ {
+ memset(&this->lss_min_log_time, 0, sizeof(this->lss_min_log_time));
+ this->lss_max_log_time.tv_sec = std::numeric_limits<time_t>::max();
+ this->lss_max_log_time.tv_usec = 0;
+ this->text_filters_changed();
+ }
+ }
+
+ bool list_input_handle_key(listview_curses& lv, int ch);
+
+ void set_marked_only(bool val)
+ {
+ if (this->lss_marked_only != val) {
+ this->lss_marked_only = val;
+ this->text_filters_changed();
+ }
+ }
+
+ bool get_marked_only() { return this->lss_marked_only; }
+
+ size_t text_line_count() { return this->lss_filtered_index.size(); }
+
+ size_t text_line_width(textview_curses& curses)
+ {
+ return this->lss_longest_line;
+ }
+
+ size_t file_count() const
+ {
+ size_t retval = 0;
+ const_iterator iter;
+
+ for (iter = this->cbegin(); iter != this->cend(); ++iter) {
+ if (*iter != nullptr && (*iter)->get_file() != nullptr) {
+ retval += 1;
+ }
+ }
+
+ return retval;
+ }
+
+ bool empty() const { return this->lss_filtered_index.empty(); }
+
+ void text_value_for_line(textview_curses& tc,
+ int row,
+ std::string& value_out,
+ line_flags_t flags);
+
+ void text_attrs_for_line(textview_curses& tc,
+ int row,
+ string_attrs_t& value_out);
+
+ size_t text_size_for_line(textview_curses& tc, int row, line_flags_t flags)
+ {
+ size_t index = row % LINE_SIZE_CACHE_SIZE;
+
+ if (this->lss_line_size_cache[index].first != row) {
+ std::string value;
+
+ this->text_value_for_line(tc, row, value, flags);
+ this->lss_line_size_cache[index].second = value.size();
+ this->lss_line_size_cache[index].first = row;
+ }
+ return this->lss_line_size_cache[index].second;
+ }
+
+ void text_mark(const bookmark_type_t* bm, vis_line_t line, bool added);
+
+ void text_clear_marks(const bookmark_type_t* bm);
+
+ bool insert_file(const std::shared_ptr<logfile>& lf);
+
+ void remove_file(std::shared_ptr<logfile> lf);
+
+ enum class rebuild_result {
+ rr_no_change,
+ rr_appended_lines,
+ rr_partial_rebuild,
+ rr_full_rebuild,
+ };
+
+ rebuild_result rebuild_index(nonstd::optional<ui_clock::time_point> deadline
+ = nonstd::nullopt);
+
+ void text_update_marks(vis_bookmarks& bm);
+
+ void set_user_mark(const bookmark_type_t* bm, content_line_t cl)
+ {
+ this->lss_user_marks[bm].insert_once(cl);
+ }
+
+ bookmarks<content_line_t>::type& get_user_bookmarks()
+ {
+ return this->lss_user_marks;
+ }
+
+ bookmark_metadata& get_bookmark_metadata(content_line_t cl);
+
+ bookmark_metadata& get_bookmark_metadata(vis_line_t vl)
+ {
+ return this->get_bookmark_metadata(this->at(vl));
+ }
+
+ nonstd::optional<bookmark_metadata*> find_bookmark_metadata(
+ content_line_t cl);
+
+ nonstd::optional<bookmark_metadata*> find_bookmark_metadata(vis_line_t vl)
+ {
+ return this->find_bookmark_metadata(this->at(vl));
+ }
+
+ void erase_bookmark_metadata(content_line_t cl);
+
+ void erase_bookmark_metadata(vis_line_t vl)
+ {
+ this->erase_bookmark_metadata(this->at(vl));
+ }
+
+ void clear_bookmark_metadata();
+
+ int get_filtered_count() const
+ {
+ return this->lss_index.size() - this->lss_filtered_index.size();
+ }
+
+ int get_filtered_count_for(size_t filter_index) const
+ {
+ int retval = 0;
+
+ for (const auto& ld : this->lss_files) {
+ retval += ld->ld_filter_state.lfo_filter_state
+ .tfs_filter_hits[filter_index];
+ }
+
+ return retval;
+ }
+
+ Result<void, lnav::console::user_message> set_sql_filter(
+ std::string stmt_str, sqlite3_stmt* stmt);
+
+ Result<void, lnav::console::user_message> set_sql_marker(
+ std::string stmt_str, sqlite3_stmt* stmt);
+
+ Result<void, lnav::console::user_message> set_preview_sql_filter(
+ sqlite3_stmt* stmt);
+
+ std::string get_sql_filter_text()
+ {
+ auto filt = this->get_sql_filter();
+
+ if (filt) {
+ return filt.value()->get_id();
+ }
+ return "";
+ }
+
+ nonstd::optional<std::shared_ptr<text_filter>> get_sql_filter();
+
+ std::string get_sql_marker_text() const
+ {
+ return this->lss_marker_stmt_text;
+ }
+
+ std::shared_ptr<logfile> find(const char* fn, content_line_t& line_base);
+
+ std::shared_ptr<logfile> find(content_line_t& line) const
+ {
+ std::shared_ptr<logfile> retval;
+
+ retval = this->lss_files[line / MAX_LINES_PER_FILE]->get_file();
+ line = content_line_t(line % MAX_LINES_PER_FILE);
+
+ return retval;
+ }
+
+ logfile* find_file_ptr(content_line_t& line)
+ {
+ auto retval
+ = this->lss_files[line / MAX_LINES_PER_FILE]->get_file_ptr();
+ line = content_line_t(line % MAX_LINES_PER_FILE);
+
+ return retval;
+ }
+
+ logline* find_line(content_line_t line) const
+ {
+ logline* retval = nullptr;
+ std::shared_ptr<logfile> lf = this->find(line);
+
+ if (lf != nullptr) {
+ auto ll_iter = lf->begin() + line;
+
+ retval = &(*ll_iter);
+ }
+
+ return retval;
+ }
+
+ nonstd::optional<std::pair<std::shared_ptr<logfile>, logfile::iterator>>
+ find_line_with_file(content_line_t line) const
+ {
+ std::shared_ptr<logfile> lf = this->find(line);
+
+ if (lf != nullptr) {
+ auto ll_iter = lf->begin() + line;
+
+ return std::make_pair(lf, ll_iter);
+ }
+
+ return nonstd::nullopt;
+ }
+
+ nonstd::optional<std::pair<std::shared_ptr<logfile>, logfile::iterator>>
+ find_line_with_file(vis_line_t vl) const
+ {
+ if (vl >= 0_vl && vl <= vis_line_t(this->lss_filtered_index.size())) {
+ return this->find_line_with_file(this->at(vl));
+ }
+
+ return nonstd::nullopt;
+ }
+
+ nonstd::optional<vis_line_t> find_from_time(
+ const struct timeval& start) const;
+
+ nonstd::optional<vis_line_t> find_from_time(time_t start) const
+ {
+ struct timeval tv = {start, 0};
+
+ return this->find_from_time(tv);
+ }
+
+ nonstd::optional<vis_line_t> find_from_time(const exttm& etm) const
+ {
+ return this->find_from_time(etm.to_timeval());
+ }
+
+ nonstd::optional<vis_line_t> find_from_content(content_line_t cl);
+
+ nonstd::optional<struct timeval> time_for_row(vis_line_t row)
+ {
+ if (row < (ssize_t) this->text_line_count()) {
+ return this->find_line(this->at(row))->get_timeval();
+ }
+ return nonstd::nullopt;
+ }
+
+ nonstd::optional<vis_line_t> row_for_time(struct timeval time_bucket)
+ {
+ return this->find_from_time(time_bucket);
+ }
+
+ content_line_t at(vis_line_t vl) const
+ {
+ return this->lss_index[this->lss_filtered_index[vl]];
+ }
+
+ content_line_t at_base(vis_line_t vl)
+ {
+ while (this->find_line(this->at(vl))->get_sub_offset() != 0) {
+ --vl;
+ }
+
+ return this->at(vl);
+ }
+
+ logline_window window_at(vis_line_t start_vl, vis_line_t end_vl)
+ {
+ return logline_window(*this, start_vl, end_vl);
+ }
+
+ log_accel::direction_t get_line_accel_direction(vis_line_t vl);
+
+ /**
+ * Container for logfile references that keeps of how many lines in the
+ * logfile have been indexed.
+ */
+ struct logfile_data {
+ logfile_data(size_t index,
+ filter_stack& fs,
+ const std::shared_ptr<logfile>& lf)
+ : ld_file_index(index), ld_filter_state(fs, lf),
+ ld_visible(lf->is_indexing())
+ {
+ lf->set_logline_observer(&this->ld_filter_state);
+ }
+
+ void clear() { this->ld_filter_state.lfo_filter_state.clear(); }
+
+ void set_file(const std::shared_ptr<logfile>& lf)
+ {
+ this->ld_filter_state.lfo_filter_state.tfs_logfile = lf;
+ lf->set_logline_observer(&this->ld_filter_state);
+ }
+
+ std::shared_ptr<logfile> get_file() const
+ {
+ return this->ld_filter_state.lfo_filter_state.tfs_logfile;
+ }
+
+ logfile* get_file_ptr() const
+ {
+ return this->ld_filter_state.lfo_filter_state.tfs_logfile.get();
+ }
+
+ bool is_visible() const
+ {
+ return this->get_file_ptr() != nullptr && this->ld_visible;
+ }
+
+ void set_visibility(bool vis) { this->ld_visible = vis; }
+
+ size_t ld_file_index;
+ line_filter_observer ld_filter_state;
+ size_t ld_lines_indexed{0};
+ size_t ld_lines_watched{0};
+ bool ld_visible;
+ };
+
+ using iterator = std::vector<std::unique_ptr<logfile_data>>::iterator;
+ using const_iterator
+ = std::vector<std::unique_ptr<logfile_data>>::const_iterator;
+
+ iterator begin() { return this->lss_files.begin(); }
+
+ iterator end() { return this->lss_files.end(); }
+
+ const_iterator cbegin() const { return this->lss_files.begin(); }
+
+ const_iterator cend() const { return this->lss_files.end(); }
+
+ iterator find_data(content_line_t& line)
+ {
+ auto retval = this->lss_files.begin();
+ std::advance(retval, line / MAX_LINES_PER_FILE);
+ line = content_line_t(line % MAX_LINES_PER_FILE);
+
+ return retval;
+ }
+
+ iterator find_data(content_line_t line, uint64_t& offset_out)
+ {
+ auto retval = this->lss_files.begin();
+ std::advance(retval, line / MAX_LINES_PER_FILE);
+ offset_out = line % MAX_LINES_PER_FILE;
+
+ return retval;
+ }
+
+ nonstd::optional<logfile_data*> find_data(
+ const std::shared_ptr<logfile>& lf)
+ {
+ for (auto& ld : *this) {
+ if (ld->ld_filter_state.lfo_filter_state.tfs_logfile == lf) {
+ return ld.get();
+ }
+ }
+ return nonstd::nullopt;
+ }
+
+ iterator find_data_i(const std::shared_ptr<const logfile>& lf)
+ {
+ for (auto iter = this->begin(); iter != this->end(); ++iter) {
+ if ((*iter)->ld_filter_state.lfo_filter_state.tfs_logfile == lf) {
+ return iter;
+ }
+ }
+
+ return this->end();
+ }
+
+ content_line_t get_file_base_content_line(iterator iter)
+ {
+ ssize_t index = std::distance(this->begin(), iter);
+
+ return content_line_t(index * MAX_LINES_PER_FILE);
+ }
+
+ void set_index_delegate(index_delegate* id)
+ {
+ if (id != this->lss_index_delegate) {
+ this->lss_index_delegate = id;
+ this->reload_index_delegate();
+ }
+ }
+
+ index_delegate* get_index_delegate() const
+ {
+ return this->lss_index_delegate;
+ }
+
+ void reload_index_delegate();
+
+ class meta_grepper
+ : public grep_proc_source<vis_line_t>
+ , public grep_proc_sink<vis_line_t> {
+ public:
+ meta_grepper(logfile_sub_source& source) : lmg_source(source) {}
+
+ bool grep_value_for_line(vis_line_t line,
+ std::string& value_out) override;
+
+ vis_line_t grep_initial_line(vis_line_t start,
+ vis_line_t highest) override;
+
+ void grep_next_line(vis_line_t& line) override;
+
+ void grep_begin(grep_proc<vis_line_t>& gp,
+ vis_line_t start,
+ vis_line_t stop) override;
+
+ void grep_end(grep_proc<vis_line_t>& gp) override;
+
+ void grep_match(grep_proc<vis_line_t>& gp,
+ vis_line_t line,
+ int start,
+ int end) override;
+
+ logfile_sub_source& lmg_source;
+ bool lmg_done{false};
+ };
+
+ nonstd::optional<
+ std::pair<grep_proc_source<vis_line_t>*, grep_proc_sink<vis_line_t>*>>
+ get_grepper();
+
+ nonstd::optional<location_history*> get_location_history()
+ {
+ return &this->lss_location_history;
+ }
+
+ void text_crumbs_for_line(int line, std::vector<breadcrumb::crumb>& crumbs);
+
+ Result<bool, lnav::console::user_message> eval_sql_filter(
+ sqlite3_stmt* stmt, iterator ld, logfile::const_iterator ll);
+
+ void invalidate_sql_filter();
+
+ void set_line_meta_changed() { this->lss_line_meta_changed = true; }
+
+ bool is_line_meta_changed() const { return this->lss_line_meta_changed; }
+
+ void set_exec_context(exec_context* ec) { this->lss_exec_context = ec; }
+
+ static const uint64_t MAX_CONTENT_LINES = (1ULL << 40) - 1;
+ static const uint64_t MAX_LINES_PER_FILE = 256 * 1024 * 1024;
+ static const uint64_t MAX_FILES = (MAX_CONTENT_LINES / MAX_LINES_PER_FILE);
+
+ std::function<void(logfile_sub_source&, file_off_t, file_size_t)>
+ lss_sorting_observer;
+
+ uint32_t lss_index_generation{0};
+
+ void quiesce();
+
+private:
+ static const size_t LINE_SIZE_CACHE_SIZE = 512;
+
+ enum {
+ B_SCRUB,
+ B_TIME_OFFSET,
+ B_FILENAME,
+ B_BASENAME,
+ };
+
+ enum {
+ F_SCRUB = (1UL << B_SCRUB),
+ F_TIME_OFFSET = (1UL << B_TIME_OFFSET),
+ F_FILENAME = (1UL << B_FILENAME),
+ F_BASENAME = (1UL << B_BASENAME),
+
+ F_NAME_MASK = (F_FILENAME | F_BASENAME),
+ };
+
+ struct __attribute__((__packed__)) indexed_content {
+ indexed_content() = default;
+
+ indexed_content(content_line_t cl) : ic_value(cl) {}
+
+ operator content_line_t() const
+ {
+ return content_line_t(this->ic_value);
+ }
+
+ uint64_t ic_value : 40;
+ };
+
+ struct logline_cmp {
+ logline_cmp(logfile_sub_source& lc) : llss_controller(lc) {}
+
+ bool operator()(const content_line_t& lhs,
+ const content_line_t& rhs) const
+ {
+ logline* ll_lhs = this->llss_controller.find_line(lhs);
+ logline* ll_rhs = this->llss_controller.find_line(rhs);
+
+ return (*ll_lhs) < (*ll_rhs);
+ }
+
+ bool operator()(const uint32_t& lhs, const uint32_t& rhs) const
+ {
+ content_line_t cl_lhs
+ = (content_line_t) llss_controller.lss_index[lhs];
+ content_line_t cl_rhs
+ = (content_line_t) llss_controller.lss_index[rhs];
+ logline* ll_lhs = this->llss_controller.find_line(cl_lhs);
+ logline* ll_rhs = this->llss_controller.find_line(cl_rhs);
+
+ return (*ll_lhs) < (*ll_rhs);
+ }
+#if 0
+ bool operator()(const indexed_content &lhs, const indexed_content &rhs)
+ {
+ logline *ll_lhs = this->llss_controller.find_line(lhs.ic_value);
+ logline *ll_rhs = this->llss_controller.find_line(rhs.ic_value);
+
+ return (*ll_lhs) < (*ll_rhs);
+ }
+#endif
+
+ bool operator()(const content_line_t& lhs, const time_t& rhs) const
+ {
+ logline* ll_lhs = this->llss_controller.find_line(lhs);
+
+ return *ll_lhs < rhs;
+ }
+
+ bool operator()(const content_line_t& lhs,
+ const struct timeval& rhs) const
+ {
+ logline* ll_lhs = this->llss_controller.find_line(lhs);
+
+ return *ll_lhs < rhs;
+ }
+
+ logfile_sub_source& llss_controller;
+ };
+
+ struct filtered_logline_cmp {
+ filtered_logline_cmp(const logfile_sub_source& lc) : llss_controller(lc)
+ {
+ }
+
+ bool operator()(const uint32_t& lhs, const uint32_t& rhs) const
+ {
+ content_line_t cl_lhs
+ = (content_line_t) llss_controller.lss_index[lhs];
+ content_line_t cl_rhs
+ = (content_line_t) llss_controller.lss_index[rhs];
+ logline* ll_lhs = this->llss_controller.find_line(cl_lhs);
+ logline* ll_rhs = this->llss_controller.find_line(cl_rhs);
+
+ return (*ll_lhs) < (*ll_rhs);
+ }
+
+ bool operator()(const uint32_t& lhs, const struct timeval& rhs) const
+ {
+ content_line_t cl_lhs
+ = (content_line_t) llss_controller.lss_index[lhs];
+ logline* ll_lhs = this->llss_controller.find_line(cl_lhs);
+
+ return (*ll_lhs) < rhs;
+ }
+
+ const logfile_sub_source& llss_controller;
+ };
+
+ /**
+ * Functor for comparing the ld_file field of the logfile_data struct.
+ */
+ struct logfile_data_eq {
+ explicit logfile_data_eq(std::shared_ptr<logfile> lf)
+ : lde_file(std::move(lf))
+ {
+ }
+
+ bool operator()(const std::unique_ptr<logfile_data>& ld) const
+ {
+ return this->lde_file == ld->get_file();
+ }
+
+ std::shared_ptr<logfile> lde_file;
+ };
+
+ void clear_line_size_cache()
+ {
+ this->lss_line_size_cache.fill(std::make_pair(0, 0));
+ this->lss_line_size_cache[0].first = -1;
+ }
+
+ bool check_extra_filters(iterator ld, logfile::iterator ll);
+
+ size_t lss_basename_width = 0;
+ size_t lss_filename_width = 0;
+ unsigned long lss_flags{0};
+ bool lss_force_rebuild{false};
+ std::vector<std::unique_ptr<logfile_data>> lss_files;
+
+ big_array<indexed_content> lss_index;
+ std::vector<uint32_t> lss_filtered_index;
+ auto_mem<sqlite3_stmt> lss_preview_filter_stmt{sqlite3_finalize};
+
+ bookmarks<content_line_t>::type lss_user_marks;
+ auto_mem<sqlite3_stmt> lss_marker_stmt{sqlite3_finalize};
+ std::string lss_marker_stmt_text;
+
+ line_flags_t lss_token_flags{0};
+ iterator lss_token_file_data;
+ std::shared_ptr<logfile> lss_token_file;
+ std::string lss_token_value;
+ string_attrs_t lss_token_attrs;
+ lnav::document::metadata lss_token_meta;
+ int lss_token_meta_line{-1};
+ int lss_token_meta_size{0};
+ logline_value_vector lss_token_values;
+ int lss_token_shift_start{0};
+ int lss_token_shift_size{0};
+ shared_buffer lss_share_manager;
+ logfile::iterator lss_token_line;
+ std::array<std::pair<int, size_t>, LINE_SIZE_CACHE_SIZE>
+ lss_line_size_cache;
+ log_level_t lss_min_log_level{LEVEL_UNKNOWN};
+ struct timeval lss_min_log_time {
+ 0, 0
+ };
+ struct timeval lss_max_log_time {
+ std::numeric_limits<time_t>::max(), 0
+ };
+ bool lss_marked_only{false};
+ index_delegate* lss_index_delegate{nullptr};
+ size_t lss_longest_line{0};
+ meta_grepper lss_meta_grepper;
+ log_location_history lss_location_history;
+ exec_context* lss_exec_context{nullptr};
+
+ bool lss_in_value_for_line{false};
+ bool lss_line_meta_changed{false};
+};
+
+#endif