/*
*
* 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
* Copyright (c) 2005-2021 Brenden Matthews, Philip Kovacs, et. al.
* (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 .
*
*/
#include
#include
#include
#include "conky.h"
#define BUFLEN 512
static void user_name(char *ptr) {
const struct utmp *usr = 0;
setutent();
while ((usr = getutent()) != nullptr) {
if (usr->ut_type == USER_PROCESS) {
if (strlen(ptr) + strnlen(usr->ut_name, UT_NAMESIZE - 1) + 1 <= BUFLEN) {
strncat(ptr, usr->ut_name, UT_NAMESIZE);
}
}
}
}
static void user_num(int *ptr) {
const struct utmp *usr;
int users_num = 0;
setutent();
while ((usr = getutent()) != nullptr) {
if (usr->ut_type == USER_PROCESS) { ++users_num; }
}
*ptr = users_num;
}
static void user_term(char *ptr) {
const struct utmp *usr;
setutent();
while ((usr = getutent()) != nullptr) {
if (usr->ut_type == USER_PROCESS) {
if (strlen(ptr) + strnlen(usr->ut_line, UT_LINESIZE) + 1 <= BUFLEN) {
strncat(ptr, usr->ut_line, UT_LINESIZE);
}
}
}
}
static void user_time(char *ptr) {
const struct utmp *usr;
time_t log_in, real, diff;
char buf[BUFLEN] = "";
setutent();
while ((usr = getutent()) != nullptr) {
if (usr->ut_type == USER_PROCESS) {
log_in = usr->ut_time;
time(&real);
diff = difftime(real, log_in);
format_seconds(buf, BUFLEN, diff);
if (strlen(ptr) + strlen(buf) + 1 <= BUFLEN) {
strncat(ptr, buf, BUFLEN - strlen(ptr) - 1);
}
}
}
}
static void tty_user_time(char *ptr, char *tty) {
time_t real, diff, log_in;
char buf[BUFLEN] = "";
struct utmp *usr, line;
setutent();
strncpy(line.ut_line, tty, UT_LINESIZE);
usr = getutline(&line);
if (usr == nullptr) { return; }
log_in = usr->ut_time;
time(&real);
diff = difftime(real, log_in);
format_seconds(buf, BUFLEN, diff);
strncpy(ptr, buf, BUFLEN - 1);
}
static void users_alloc(struct information *ptr) {
if (ptr->users.names == nullptr) {
ptr->users.names = (char *)malloc(text_buffer_size.get(*state));
}
if (ptr->users.terms == nullptr) {
ptr->users.terms = (char *)malloc(text_buffer_size.get(*state));
}
if (ptr->users.times == nullptr) {
ptr->users.times = (char *)malloc(text_buffer_size.get(*state));
}
}
static void update_user_time(char *tty) {
struct information *current_info = &info;
char temp[BUFLEN] = "";
if (current_info->users.ctime == nullptr) {
current_info->users.ctime = (char *)malloc(text_buffer_size.get(*state));
}
tty_user_time(temp, tty);
if (*temp != 0) {
free_and_zero(current_info->users.ctime);
current_info->users.ctime = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.ctime, temp, text_buffer_size.get(*state));
} else {
free_and_zero(current_info->users.ctime);
current_info->users.ctime = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.ctime, "broken", text_buffer_size.get(*state));
}
}
int update_users(void) {
struct information *current_info = &info;
char temp[BUFLEN] = "";
int t;
users_alloc(current_info);
user_name(temp);
if (*temp != 0) {
free_and_zero(current_info->users.names);
current_info->users.names = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.names, temp, text_buffer_size.get(*state));
} else {
free_and_zero(current_info->users.names);
current_info->users.names = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.names, "broken", text_buffer_size.get(*state));
}
user_num(&t);
if (t != 0) {
if (current_info->users.number) { current_info->users.number = 0; }
current_info->users.number = t;
} else {
current_info->users.number = 0;
}
temp[0] = 0;
user_term(temp);
if (*temp != 0) {
free_and_zero(current_info->users.terms);
current_info->users.terms = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.terms, temp, text_buffer_size.get(*state));
} else {
free_and_zero(current_info->users.terms);
current_info->users.terms = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.terms, "broken", text_buffer_size.get(*state));
}
user_time(temp);
if (*temp != 0) {
free_and_zero(current_info->users.times);
current_info->users.times = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.times, temp, text_buffer_size.get(*state));
} else {
free_and_zero(current_info->users.times);
current_info->users.times = (char *)malloc(text_buffer_size.get(*state));
strncpy(current_info->users.times, "broken", text_buffer_size.get(*state));
}
return 0;
}
void print_user_names(struct text_object *obj, char *p,
unsigned int p_max_size) {
(void)obj;
snprintf(p, p_max_size, "%s", info.users.names);
}
void print_user_terms(struct text_object *obj, char *p,
unsigned int p_max_size) {
(void)obj;
snprintf(p, p_max_size, "%s", info.users.terms);
}
void print_user_times(struct text_object *obj, char *p,
unsigned int p_max_size) {
(void)obj;
snprintf(p, p_max_size, "%s", info.users.times);
}
void print_user_time(struct text_object *obj, char *p,
unsigned int p_max_size) {
update_user_time(obj->data.s);
snprintf(p, p_max_size, "%s", info.users.ctime);
}
void print_user_number(struct text_object *obj, char *p,
unsigned int p_max_size) {
(void)obj;
snprintf(p, p_max_size, "%d", info.users.number);
}
void free_user_names(struct text_object *obj) {
(void)obj;
free_and_zero(info.users.names);
}
void free_user_terms(struct text_object *obj) {
(void)obj;
free_and_zero(info.users.terms);
}
void free_user_times(struct text_object *obj) {
(void)obj;
free_and_zero(info.users.times);
}
void free_user_time(struct text_object *obj) {
free_and_zero(info.users.ctime);
free_and_zero(obj->data.s);
}