mirror of https://github.com/tstack/lnav
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
191 lines
5.1 KiB
C++
191 lines
5.1 KiB
C++
/**
|
|
* Copyright (c) 2023, 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.
|
|
*/
|
|
|
|
#ifndef piper_looper_hh
|
|
#define piper_looper_hh
|
|
|
|
#include <future>
|
|
#include <memory>
|
|
#include <string>
|
|
|
|
#include "base/auto_fd.hh"
|
|
#include "base/piper.file.hh"
|
|
#include "base/result.h"
|
|
#include "ghc/filesystem.hpp"
|
|
#include "safe/safe.h"
|
|
#include "yajlpp/yajlpp_def.hh"
|
|
|
|
namespace lnav {
|
|
namespace piper {
|
|
|
|
enum class state {
|
|
running,
|
|
finished,
|
|
};
|
|
|
|
using safe_demux_id = safe::Safe<std::string>;
|
|
|
|
struct options {
|
|
bool o_demux{false};
|
|
bool o_tail{true};
|
|
|
|
options& with_demux(bool v)
|
|
{
|
|
this->o_demux = v;
|
|
return *this;
|
|
}
|
|
|
|
options& with_tail(bool v)
|
|
{
|
|
this->o_tail = v;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
class looper {
|
|
public:
|
|
looper(std::string name,
|
|
auto_fd stdout_fd,
|
|
auto_fd stderr_fd,
|
|
options opts);
|
|
|
|
~looper();
|
|
|
|
std::string get_name() const { return this->l_name; }
|
|
|
|
ghc::filesystem::path get_out_dir() const { return this->l_out_dir; }
|
|
|
|
ghc::filesystem::path get_out_pattern() const
|
|
{
|
|
return this->l_out_dir / "out.*";
|
|
}
|
|
|
|
std::string get_demux_id() const { return *this->l_demux_id.readAccess(); }
|
|
|
|
std::string get_url() const
|
|
{
|
|
return fmt::format(FMT_STRING("piper://{}"),
|
|
this->l_out_dir.filename().string());
|
|
}
|
|
|
|
int get_loop_count() const
|
|
{
|
|
return this->l_loop_count.load();
|
|
}
|
|
|
|
bool is_finished() const
|
|
{
|
|
return this->l_future.wait_for(std::chrono::seconds(0))
|
|
== std::future_status::ready;
|
|
}
|
|
|
|
size_t consume_finished()
|
|
{
|
|
if (!this->is_finished()) {
|
|
return 0;
|
|
}
|
|
|
|
if (this->l_finished.fetch_or(1) == 0) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
private:
|
|
void loop();
|
|
|
|
std::atomic<bool> l_looping{true};
|
|
const std::string l_name;
|
|
const std::string l_cwd;
|
|
const std::map<std::string, std::string> l_env;
|
|
ghc::filesystem::path l_out_dir;
|
|
auto_fd l_stdout;
|
|
auto_fd l_stderr;
|
|
options l_options;
|
|
std::future<void> l_future;
|
|
std::atomic<int> l_finished{0};
|
|
std::atomic<int> l_loop_count{0};
|
|
safe_demux_id l_demux_id;
|
|
};
|
|
|
|
template<state LooperState>
|
|
class handle {
|
|
public:
|
|
explicit handle(std::shared_ptr<looper> looper)
|
|
: h_looper(std::move(looper))
|
|
{
|
|
}
|
|
|
|
std::string get_name() const { return this->h_looper->get_name(); }
|
|
|
|
ghc::filesystem::path get_out_dir() const
|
|
{
|
|
return this->h_looper->get_out_dir();
|
|
}
|
|
|
|
ghc::filesystem::path get_out_pattern() const
|
|
{
|
|
return this->h_looper->get_out_pattern();
|
|
}
|
|
|
|
std::string get_demux_id() const { return this->h_looper->get_demux_id(); }
|
|
|
|
std::string get_url() const { return this->h_looper->get_url(); }
|
|
|
|
int get_loop_count() const { return this->h_looper->get_loop_count(); }
|
|
|
|
bool is_finished() const { return this->h_looper->is_finished(); }
|
|
|
|
size_t consume_finished() { return this->h_looper->consume_finished(); }
|
|
|
|
bool operator==(const handle& other) const
|
|
{
|
|
return this->h_looper.get() == other.h_looper.get();
|
|
}
|
|
|
|
private:
|
|
std::shared_ptr<looper> h_looper;
|
|
};
|
|
|
|
extern const typed_json_path_container<lnav::piper::header> header_handlers;
|
|
|
|
using running_handle = handle<state::running>;
|
|
|
|
Result<handle<state::running>, std::string> create_looper(std::string name,
|
|
auto_fd stdout_fd,
|
|
auto_fd stderr_fd,
|
|
options opts = {});
|
|
|
|
void cleanup();
|
|
|
|
} // namespace piper
|
|
} // namespace lnav
|
|
|
|
#endif
|