/** * 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. * * @file file_collection.hh */ #ifndef lnav_file_collection_hh #define lnav_file_collection_hh #include #include #include #include #include #include #include "archive_manager.hh" #include "base/future_util.hh" #include "file_format.hh" #include "logfile_fwd.hh" #include "safe/safe.h" #include "tailer/tailer.looper.hh" struct tailer_progress { std::string tp_message; }; struct scan_progress { std::list sp_extractions; std::map sp_tailers; }; using safe_scan_progress = safe::Safe; struct other_file_descriptor { file_format_t ofd_format; std::string ofd_description; other_file_descriptor(file_format_t format = file_format_t::UNKNOWN, std::string description = "") : ofd_format(format), ofd_description(std::move(description)) { } }; struct file_error_info { const time_t fei_mtime; const std::string fei_description; }; struct file_collection; enum class child_poll_result_t { ALIVE, FINISHED, }; class child_poller { public: explicit child_poller( auto_pid child, std::function&)> finalizer) : cp_child(std::move(child)), cp_finalizer(std::move(finalizer)) { ensure(this->cp_finalizer); } child_poller(child_poller&& other) noexcept : cp_child(std::move(other.cp_child)), cp_finalizer(std::move(other.cp_finalizer)) { ensure(this->cp_finalizer); } child_poller& operator=(child_poller&& other) noexcept { require(other.cp_finalizer); this->cp_child = std::move(other.cp_child); this->cp_finalizer = std::move(other.cp_finalizer); return *this; } ~child_poller() noexcept = default; child_poller(const child_poller&) = delete; child_poller& operator=(const child_poller&) = delete; child_poll_result_t poll(file_collection& fc); private: nonstd::optional> cp_child; std::function&)> cp_finalizer; }; struct file_collection { bool fc_invalidate_merge{false}; bool fc_recursive{false}; bool fc_rotated{false}; std::map fc_name_to_errors; std::map fc_file_names; std::vector> fc_files; int fc_files_generation{0}; std::vector, std::string>> fc_renamed_files; std::set fc_closed_files; std::map fc_other_files; std::set fc_synced_files; std::shared_ptr fc_progress; std::vector fc_new_stats; std::list fc_child_pollers; size_t fc_largest_path_length{0}; file_collection() : fc_progress(std::make_shared>()) { } void clear() { this->fc_name_to_errors.clear(); this->fc_file_names.clear(); this->fc_files.clear(); this->fc_closed_files.clear(); this->fc_other_files.clear(); this->fc_new_stats.clear(); } file_collection rescan_files(bool required = false); void expand_filename(lnav::futures::future_queue& fq, const std::string& path, logfile_open_options& loo, bool required); std::future watch_logfile(const std::string& filename, logfile_open_options& loo, bool required); void merge(file_collection& other); void request_close(const std::shared_ptr& lf); void close_files(const std::vector>& files); void regenerate_unique_file_names(); }; #endif