2010-01-02 23:59:24 -08:00
|
|
|
/* -*- mode: c++; c-basic-offset: 4; tab-width: 4; indent-tabs-mode: t -*-
|
|
|
|
* vim: ts=4 sw=4 noet ai cindent syntax=cpp
|
2009-10-15 21:51:21 +02:00
|
|
|
*
|
|
|
|
* Conky, a system monitor, based on torsmo
|
|
|
|
*
|
|
|
|
* Any original torsmo code is licensed under the BSD license
|
|
|
|
*
|
|
|
|
* All code written since the fork of torsmo is licensed under the GPL
|
|
|
|
*
|
|
|
|
* Please see COPYING for details
|
|
|
|
*
|
|
|
|
* Copyright (c) 2004, Hannu Saransaari and Lauri Hakkarainen
|
2010-01-01 15:46:17 -08:00
|
|
|
* Copyright (c) 2005-2010 Brenden Matthews, Philip Kovacs, et. al.
|
2009-10-15 21:51:21 +02:00
|
|
|
* (see AUTHORS)
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "conky.h"
|
|
|
|
#include "core.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "specials.h"
|
|
|
|
#include "text_object.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
2009-11-07 12:21:48 +01:00
|
|
|
#include <unistd.h>
|
2011-02-26 18:54:48 +01:00
|
|
|
#include <cmath>
|
2010-01-02 23:59:24 -08:00
|
|
|
#include <mutex>
|
2011-02-26 18:54:48 +01:00
|
|
|
#include "update-cb.hh"
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class exec_cb: public conky::callback<std::string, std::string> {
|
|
|
|
typedef conky::callback<std::string, std::string> Base;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual void work();
|
|
|
|
|
|
|
|
public:
|
|
|
|
exec_cb(uint32_t period, bool wait, const std::string &cmd)
|
|
|
|
: Base(period, wait, Base::Tuple(cmd))
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
}
|
2009-10-15 21:51:21 +02:00
|
|
|
|
2009-10-07 00:09:14 +02:00
|
|
|
struct execi_data {
|
|
|
|
float interval;
|
|
|
|
char *cmd;
|
2011-02-26 18:54:48 +01:00
|
|
|
execi_data() : interval(0), cmd(0) {}
|
2009-10-07 00:09:14 +02:00
|
|
|
};
|
|
|
|
|
2009-10-15 21:51:21 +02:00
|
|
|
//our own implementation of popen, the difference : the value of 'childpid' will be filled with
|
|
|
|
//the pid of the running 'command'. This is useful if want to kill it when it hangs while reading
|
|
|
|
//or writing to it. We have to kill it because pclose will wait until the process dies by itself
|
|
|
|
static FILE* pid_popen(const char *command, const char *mode, pid_t *child) {
|
|
|
|
int ends[2];
|
|
|
|
int parentend, childend;
|
|
|
|
|
|
|
|
//by running pipe after the strcmp's we make sure that we don't have to create a pipe
|
|
|
|
//and close the ends if mode is something illegal
|
|
|
|
if(strcmp(mode, "r") == 0) {
|
|
|
|
if(pipe(ends) != 0) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
parentend = ends[0];
|
|
|
|
childend = ends[1];
|
|
|
|
} else if(strcmp(mode, "w") == 0) {
|
|
|
|
if(pipe(ends) != 0) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
parentend = ends[1];
|
|
|
|
childend = ends[0];
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*child = fork();
|
|
|
|
if(*child == -1) {
|
|
|
|
close(parentend);
|
|
|
|
close(childend);
|
|
|
|
return NULL;
|
|
|
|
} else if(*child > 0) {
|
|
|
|
close(childend);
|
|
|
|
waitpid(*child, NULL, 0);
|
|
|
|
} else {
|
|
|
|
//don't read from both stdin and pipe or write to both stdout and pipe
|
|
|
|
if(childend == ends[0]) {
|
|
|
|
close(0);
|
|
|
|
} else {
|
|
|
|
close(1);
|
|
|
|
}
|
2010-12-05 21:16:01 +01:00
|
|
|
close(parentend);
|
2009-12-27 03:41:21 +01:00
|
|
|
|
|
|
|
//by dupping childend, the returned fd will have close-on-exec turned off
|
|
|
|
if (dup(childend) == -1)
|
|
|
|
perror("dup()");
|
2011-02-26 18:54:48 +01:00
|
|
|
close(childend);
|
2009-12-27 03:41:21 +01:00
|
|
|
|
2009-10-15 21:51:21 +02:00
|
|
|
execl("/bin/sh", "sh", "-c", command, (char *) NULL);
|
2009-10-07 00:09:14 +02:00
|
|
|
_exit(EXIT_FAILURE); //child should die here, (normally execl will take care of this but it can fail)
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
return fdopen(parentend, mode);
|
|
|
|
}
|
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
void exec_cb::work()
|
|
|
|
{
|
|
|
|
pid_t childpid;
|
|
|
|
std::string buf;
|
|
|
|
std::shared_ptr<FILE> fp;
|
|
|
|
char b[0x1000];
|
|
|
|
|
|
|
|
if(FILE *t = pid_popen(std::get<0>(tuple).c_str(), "r", &childpid))
|
|
|
|
fp.reset(t, fclose);
|
|
|
|
else
|
|
|
|
return;
|
|
|
|
|
|
|
|
while(!feof(fp.get()) && !ferror(fp.get())) {
|
|
|
|
int length = fread(b, 1, sizeof b, fp.get());
|
|
|
|
buf.append(b, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(*buf.rbegin() == '\n')
|
|
|
|
buf.resize(buf.size()-1);
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> l(result_mutex);
|
|
|
|
result = buf;
|
|
|
|
}
|
2009-10-15 21:51:21 +02:00
|
|
|
//remove backspaced chars, example: "dog^H^H^Hcat" becomes "cat"
|
|
|
|
//string has to end with \0 and it's length should fit in a int
|
|
|
|
#define BACKSPACE 8
|
|
|
|
static void remove_deleted_chars(char *string){
|
|
|
|
int i = 0;
|
|
|
|
while(string[i] != 0){
|
|
|
|
if(string[i] == BACKSPACE){
|
|
|
|
if(i != 0){
|
|
|
|
strcpy( &(string[i-1]), &(string[i+1]) );
|
|
|
|
i--;
|
|
|
|
}else strcpy( &(string[i]), &(string[i+1]) ); //necessary for ^H's at the start of a string
|
|
|
|
}else i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
static inline double get_barnum(const char *buf)
|
2009-10-15 21:51:21 +02:00
|
|
|
{
|
|
|
|
double barnum;
|
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
if (sscanf(buf, "%lf", &barnum) != 1) {
|
2009-10-15 21:51:21 +02:00
|
|
|
NORM_ERR("reading exec value failed (perhaps it's not the "
|
|
|
|
"correct format?)");
|
2009-11-22 22:52:31 +01:00
|
|
|
return 0.0;
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
if (barnum > 100.0 || barnum < 0.0) {
|
|
|
|
NORM_ERR("your exec value is not between 0 and 100, "
|
|
|
|
"therefore it will be ignored");
|
2009-11-22 22:52:31 +01:00
|
|
|
return 0.0;
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
return barnum;
|
|
|
|
}
|
|
|
|
|
|
|
|
void scan_exec_arg(struct text_object *obj, const char *arg)
|
|
|
|
{
|
2009-12-04 02:00:32 +01:00
|
|
|
/* XXX: do real bar parsing here */
|
|
|
|
scan_bar(obj, "", 100);
|
2010-08-29 23:50:32 +02:00
|
|
|
obj->data.s = strndup(arg ? arg : "", text_buffer_size.get(*state));
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void scan_execi_arg(struct text_object *obj, const char *arg)
|
|
|
|
{
|
2009-10-07 00:09:14 +02:00
|
|
|
struct execi_data *ed;
|
2009-10-15 21:51:21 +02:00
|
|
|
int n;
|
|
|
|
|
2010-01-02 23:59:24 -08:00
|
|
|
ed = new execi_data;
|
2009-10-07 00:09:14 +02:00
|
|
|
|
|
|
|
if (sscanf(arg, "%f %n", &ed->interval, &n) <= 0) {
|
2009-10-15 21:51:21 +02:00
|
|
|
NORM_ERR("${execi* <interval> command}");
|
2010-02-11 14:44:15 +01:00
|
|
|
delete ed;
|
2009-10-15 21:51:21 +02:00
|
|
|
return;
|
|
|
|
}
|
2010-08-29 23:50:32 +02:00
|
|
|
ed->cmd = strndup(arg + n, text_buffer_size.get(*state));
|
2009-10-07 00:09:14 +02:00
|
|
|
obj->data.opaque = ed;
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
2009-12-04 02:09:41 +01:00
|
|
|
void scan_execi_bar_arg(struct text_object *obj, const char *arg)
|
|
|
|
{
|
|
|
|
/* XXX: do real bar parsing here */
|
|
|
|
scan_bar(obj, "", 100);
|
|
|
|
scan_execi_arg(obj, arg);
|
|
|
|
}
|
|
|
|
|
2010-11-13 23:59:31 +01:00
|
|
|
#ifdef BUILD_X11
|
2009-12-04 02:09:41 +01:00
|
|
|
void scan_execi_gauge_arg(struct text_object *obj, const char *arg)
|
|
|
|
{
|
|
|
|
/* XXX: do real gauge parsing here */
|
|
|
|
scan_gauge(obj, "", 100);
|
|
|
|
scan_execi_arg(obj, arg);
|
|
|
|
}
|
|
|
|
|
2009-10-29 03:08:08 +01:00
|
|
|
void scan_execgraph_arg(struct text_object *obj, const char *arg)
|
|
|
|
{
|
|
|
|
struct execi_data *ed;
|
|
|
|
char *buf;
|
|
|
|
|
2010-01-02 23:59:24 -08:00
|
|
|
ed = new execi_data;
|
2009-10-29 03:08:08 +01:00
|
|
|
memset(ed, 0, sizeof(struct execi_data));
|
|
|
|
|
2009-12-04 02:00:32 +01:00
|
|
|
buf = scan_graph(obj, arg, 100);
|
2009-10-29 03:08:08 +01:00
|
|
|
if (!buf) {
|
|
|
|
NORM_ERR("missing command argument to execgraph object");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ed->cmd = buf;
|
|
|
|
obj->data.opaque = ed;
|
|
|
|
}
|
2010-01-06 18:38:12 -08:00
|
|
|
#endif /* BUILD_X11 */
|
2009-10-29 03:08:08 +01:00
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
void fill_p(const char *buffer, struct text_object *obj, char *p, int p_max_size) {
|
2010-02-10 19:45:42 +01:00
|
|
|
if(obj->parse == true) {
|
2010-11-20 16:59:32 +01:00
|
|
|
evaluate(buffer, p, p_max_size);
|
2010-02-10 19:45:42 +01:00
|
|
|
} else snprintf(p, p_max_size, "%s", buffer);
|
2010-02-10 23:31:35 +01:00
|
|
|
remove_deleted_chars(p);
|
2010-02-10 19:45:42 +01:00
|
|
|
}
|
|
|
|
|
2010-02-11 01:08:25 +01:00
|
|
|
void print_exec(struct text_object *obj, char *p, int p_max_size)
|
2009-10-15 21:51:21 +02:00
|
|
|
{
|
2011-02-26 18:54:48 +01:00
|
|
|
auto cb = conky::register_cb<exec_cb>(1, true, obj->data.s);
|
|
|
|
fill_p(cb->get_result_copy().c_str(), obj, p, p_max_size);
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void print_execi(struct text_object *obj, char *p, int p_max_size)
|
|
|
|
{
|
2010-01-02 23:59:24 -08:00
|
|
|
struct execi_data *ed = (struct execi_data *)obj->data.opaque;
|
2009-10-07 00:09:14 +02:00
|
|
|
|
|
|
|
if (!ed)
|
|
|
|
return;
|
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
uint32_t period = std::max(std::lround(ed->interval/active_update_interval()), 1l);
|
|
|
|
|
|
|
|
auto cb = conky::register_cb<exec_cb>(period, !obj->thread, ed->cmd);
|
|
|
|
|
|
|
|
fill_p(cb->get_result_copy().c_str(), obj, p, p_max_size);
|
2010-02-07 10:29:43 +01:00
|
|
|
}
|
|
|
|
|
2009-12-04 02:00:32 +01:00
|
|
|
double execbarval(struct text_object *obj)
|
2009-10-15 21:51:21 +02:00
|
|
|
{
|
2011-02-26 18:54:48 +01:00
|
|
|
auto cb = conky::register_cb<exec_cb>(1, true, obj->data.s);
|
|
|
|
return get_barnum(cb->get_result_copy().c_str());
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
2009-12-04 02:09:41 +01:00
|
|
|
double execi_barval(struct text_object *obj)
|
2009-10-15 21:51:21 +02:00
|
|
|
{
|
2010-01-02 23:59:24 -08:00
|
|
|
struct execi_data *ed = (struct execi_data *)obj->data.opaque;
|
2009-10-15 21:51:21 +02:00
|
|
|
|
2009-10-07 00:09:14 +02:00
|
|
|
if (!ed)
|
2009-11-30 23:59:13 +01:00
|
|
|
return 0;
|
2009-10-07 00:09:14 +02:00
|
|
|
|
2011-02-26 18:54:48 +01:00
|
|
|
uint32_t period = std::max(std::lround(ed->interval/active_update_interval()), 1l);
|
|
|
|
|
|
|
|
auto cb = conky::register_cb<exec_cb>(period, !obj->thread, ed->cmd);
|
|
|
|
|
|
|
|
return get_barnum(cb->get_result_copy().c_str());
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void free_exec(struct text_object *obj)
|
|
|
|
{
|
2010-02-23 23:13:11 +01:00
|
|
|
free_and_zero(obj->data.s);
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void free_execi(struct text_object *obj)
|
|
|
|
{
|
2010-01-02 23:59:24 -08:00
|
|
|
struct execi_data *ed = (struct execi_data *)obj->data.opaque;
|
2009-10-07 00:09:14 +02:00
|
|
|
|
|
|
|
if (!ed)
|
|
|
|
return;
|
|
|
|
|
2010-02-23 23:13:11 +01:00
|
|
|
free_and_zero(ed->cmd);
|
2010-01-04 19:20:47 -08:00
|
|
|
delete ed;
|
2009-10-07 00:09:14 +02:00
|
|
|
obj->data.opaque = NULL;
|
2009-10-15 21:51:21 +02:00
|
|
|
}
|