mirror of
				https://github.com/rad4day/Waybar.git
				synced 2025-10-24 22:52:32 +02:00 
			
		
		
		
	refactor: move command execution into their own file
This commit is contained in:
		| @@ -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; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										35
									
								
								include/util/command.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								include/util/command.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -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(); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Alexis
					Alexis