refactor: move command execution into their own file
parent
b794ca63d1
commit
ce50a627be
|
@ -2,8 +2,8 @@
|
||||||
|
|
||||||
#include <fmt/format.h>
|
#include <fmt/format.h>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <sys/wait.h>
|
|
||||||
#include "util/chrono.hpp"
|
#include "util/chrono.hpp"
|
||||||
|
#include "util/command.hpp"
|
||||||
#include "ALabel.hpp"
|
#include "ALabel.hpp"
|
||||||
|
|
||||||
namespace waybar::modules {
|
namespace waybar::modules {
|
||||||
|
|
|
@ -8,85 +8,85 @@
|
||||||
|
|
||||||
namespace waybar::chrono {
|
namespace waybar::chrono {
|
||||||
|
|
||||||
using namespace std::chrono;
|
using namespace std::chrono;
|
||||||
|
|
||||||
using clock = std::chrono::system_clock;
|
using clock = std::chrono::system_clock;
|
||||||
using duration = clock::duration;
|
using duration = clock::duration;
|
||||||
using time_point = std::chrono::time_point<clock, duration>;
|
using time_point = std::chrono::time_point<clock, duration>;
|
||||||
|
|
||||||
inline struct timespec to_timespec(time_point t) noexcept
|
inline struct timespec to_timespec(time_point t) noexcept
|
||||||
{
|
{
|
||||||
long secs = duration_cast<seconds>(t.time_since_epoch()).count();
|
long secs = duration_cast<seconds>(t.time_since_epoch()).count();
|
||||||
long nsc = duration_cast<nanoseconds>(t.time_since_epoch() % seconds(1)).count();
|
long nsc = duration_cast<nanoseconds>(t.time_since_epoch() % seconds(1)).count();
|
||||||
return {secs, nsc};
|
return {secs, nsc};
|
||||||
}
|
}
|
||||||
|
|
||||||
inline time_point to_time_point(struct timespec t) noexcept
|
inline time_point to_time_point(struct timespec t) noexcept
|
||||||
{
|
{
|
||||||
return time_point(duration_cast<duration>(seconds(t.tv_sec) + nanoseconds(t.tv_nsec)));
|
return time_point(duration_cast<duration>(seconds(t.tv_sec) + nanoseconds(t.tv_nsec)));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace waybar::util {
|
namespace waybar::util {
|
||||||
|
|
||||||
struct SleeperThread {
|
struct SleeperThread {
|
||||||
SleeperThread() = default;
|
SleeperThread() = default;
|
||||||
|
|
||||||
SleeperThread(std::function<void()> func)
|
SleeperThread(std::function<void()> func)
|
||||||
: thread{[this, func] {
|
: thread{[this, func] {
|
||||||
do {
|
|
||||||
func();
|
|
||||||
} while (do_run);
|
|
||||||
}}
|
|
||||||
{
|
|
||||||
defined = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
SleeperThread& operator=(std::function<void()> func)
|
|
||||||
{
|
|
||||||
thread = std::thread([this, func] {
|
|
||||||
do {
|
do {
|
||||||
func();
|
func();
|
||||||
} while (do_run);
|
} while (do_run);
|
||||||
});
|
}}
|
||||||
defined = true;
|
{
|
||||||
return *this;
|
defined = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SleeperThread& operator=(std::function<void()> func)
|
||||||
|
{
|
||||||
|
thread = std::thread([this, func] {
|
||||||
|
do {
|
||||||
|
func();
|
||||||
|
} while (do_run);
|
||||||
|
});
|
||||||
|
defined = true;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
auto sleep_for(chrono::duration dur)
|
auto sleep_for(chrono::duration dur)
|
||||||
{
|
{
|
||||||
auto lock = std::unique_lock(mutex);
|
auto lock = std::unique_lock(mutex);
|
||||||
return condvar.wait_for(lock, dur);
|
return condvar.wait_for(lock, dur);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto sleep_until(chrono::time_point time)
|
auto sleep_until(chrono::time_point time)
|
||||||
{
|
{
|
||||||
auto lock = std::unique_lock(mutex);
|
auto lock = std::unique_lock(mutex);
|
||||||
return condvar.wait_until(lock, time);
|
return condvar.wait_until(lock, time);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto wake_up()
|
auto wake_up()
|
||||||
{
|
{
|
||||||
|
condvar.notify_all();
|
||||||
|
}
|
||||||
|
|
||||||
|
~SleeperThread()
|
||||||
|
{
|
||||||
|
do_run = false;
|
||||||
|
if (defined) {
|
||||||
condvar.notify_all();
|
condvar.notify_all();
|
||||||
|
thread.join();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
~SleeperThread()
|
private:
|
||||||
{
|
std::thread thread;
|
||||||
do_run = false;
|
std::condition_variable condvar;
|
||||||
if (defined) {
|
std::mutex mutex;
|
||||||
condvar.notify_all();
|
bool defined = false;
|
||||||
thread.join();
|
bool do_run = true;
|
||||||
}
|
};
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::thread thread;
|
|
||||||
std::condition_variable condvar;
|
|
||||||
std::mutex mutex;
|
|
||||||
bool defined = false;
|
|
||||||
bool do_run = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,35 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <sys/wait.h>
|
||||||
|
|
||||||
|
namespace waybar::util::command {
|
||||||
|
|
||||||
|
struct cmd_res {
|
||||||
|
int exit_code;
|
||||||
|
std::string out;
|
||||||
|
};
|
||||||
|
|
||||||
|
inline struct cmd_res exec(const std::string cmd)
|
||||||
|
{
|
||||||
|
FILE* fp(popen(cmd.c_str(), "r"));
|
||||||
|
if (!fp) {
|
||||||
|
return { -1, "" };
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<char, 128> buffer = {0};
|
||||||
|
std::string output;
|
||||||
|
while (feof(fp) == 0) {
|
||||||
|
if (fgets(buffer.data(), 128, fp) != nullptr) {
|
||||||
|
output += buffer.data();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove last newline
|
||||||
|
if (!output.empty() && output[output.length()-1] == '\n') {
|
||||||
|
output.erase(output.length()-1);
|
||||||
|
}
|
||||||
|
int exit_code = WEXITSTATUS(pclose(fp));
|
||||||
|
return { exit_code, output };
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,31 +4,31 @@
|
||||||
|
|
||||||
namespace waybar::util {
|
namespace waybar::util {
|
||||||
|
|
||||||
struct JsonParser {
|
struct JsonParser {
|
||||||
|
|
||||||
JsonParser()
|
JsonParser()
|
||||||
: _reader(_builder.newCharReader())
|
: _reader(_builder.newCharReader())
|
||||||
{}
|
{}
|
||||||
|
|
||||||
Json::Value parse(const std::string data)
|
Json::Value parse(const std::string data)
|
||||||
{
|
{
|
||||||
Json::Value root;
|
Json::Value root;
|
||||||
std::string err;
|
std::string err;
|
||||||
bool res =
|
bool res =
|
||||||
_reader->parse(data.c_str(), data.c_str() + data.size(), &root, &err);
|
_reader->parse(data.c_str(), data.c_str() + data.size(), &root, &err);
|
||||||
if (!res)
|
if (!res)
|
||||||
throw std::runtime_error(err);
|
throw std::runtime_error(err);
|
||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
|
|
||||||
~JsonParser()
|
~JsonParser()
|
||||||
{
|
{
|
||||||
delete _reader;
|
delete _reader;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Json::CharReaderBuilder _builder;
|
Json::CharReaderBuilder _builder;
|
||||||
Json::CharReader *_reader;
|
Json::CharReader *_reader;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,22 +10,8 @@ waybar::modules::Custom::Custom(std::string name, Json::Value config)
|
||||||
thread_ = [this, interval] {
|
thread_ = [this, interval] {
|
||||||
bool can_update = true;
|
bool can_update = true;
|
||||||
if (config_["exec-if"]) {
|
if (config_["exec-if"]) {
|
||||||
auto pid = fork();
|
auto res = waybar::util::command::exec(config_["exec-if"].asString());
|
||||||
int res = 0;
|
if (res.exit_code != 0) {
|
||||||
if (pid == 0) {
|
|
||||||
std::istringstream iss(config_["exec-if"].asString());
|
|
||||||
std::vector<char*> av;
|
|
||||||
for (std::string s; iss >> s;) {
|
|
||||||
// Need to copy otherwise values are the same
|
|
||||||
char *str = new char[s.size() + 1];
|
|
||||||
memcpy(str, s.c_str(), s.size() + 1);
|
|
||||||
av.push_back(str);
|
|
||||||
}
|
|
||||||
av.push_back(0);
|
|
||||||
execvp(av.front(), av.data());
|
|
||||||
_exit(127);
|
|
||||||
} else if (pid > 0 && waitpid(pid, &res, 0) != -1
|
|
||||||
&& WEXITSTATUS(res) != 0) {
|
|
||||||
can_update = false;
|
can_update = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,33 +24,16 @@ waybar::modules::Custom::Custom(std::string name, Json::Value config)
|
||||||
|
|
||||||
auto waybar::modules::Custom::update() -> void
|
auto waybar::modules::Custom::update() -> void
|
||||||
{
|
{
|
||||||
std::array<char, 128> buffer = {0};
|
auto res = waybar::util::command::exec(config_["exec"].asString());
|
||||||
std::string output;
|
|
||||||
std::shared_ptr<FILE> fp(popen(config_["exec"].asCString(), "r"), pclose);
|
|
||||||
if (!fp) {
|
|
||||||
std::cerr << name_ + " can't exec " + config_["exec"].asString() << std::endl;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (feof(fp.get()) == 0) {
|
|
||||||
if (fgets(buffer.data(), 128, fp.get()) != nullptr) {
|
|
||||||
output += buffer.data();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove last newline
|
|
||||||
if (!output.empty() && output[output.length()-1] == '\n') {
|
|
||||||
output.erase(output.length()-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hide label if output is empty
|
// Hide label if output is empty
|
||||||
if (output.empty()) {
|
if (res.out.empty() || res.exit_code != 0) {
|
||||||
label_.hide();
|
label_.hide();
|
||||||
label_.set_name("");
|
label_.set_name("");
|
||||||
} else {
|
} else {
|
||||||
label_.set_name("custom-" + name_);
|
label_.set_name("custom-" + name_);
|
||||||
auto format = config_["format"] ? config_["format"].asString() : "{}";
|
auto format = config_["format"] ? config_["format"].asString() : "{}";
|
||||||
auto str = fmt::format(format, output);
|
auto str = fmt::format(format, res.out);
|
||||||
label_.set_text(str);
|
label_.set_text(str);
|
||||||
label_.set_tooltip_text(str);
|
label_.set_tooltip_text(str);
|
||||||
label_.show();
|
label_.show();
|
||||||
|
|
Loading…
Reference in New Issue