82b9ea8a09
Cast before multiplication to avoid overflow. Fixes #743
186 lines
6.8 KiB
C++
186 lines
6.8 KiB
C++
#include <sys/statvfs.h>
|
|
#include <fstream>
|
|
|
|
#include "drawtypes/label.hpp"
|
|
#include "drawtypes/progressbar.hpp"
|
|
#include "drawtypes/ramp.hpp"
|
|
#include "modules/fs.hpp"
|
|
#include "utils/factory.hpp"
|
|
#include "utils/math.hpp"
|
|
#include "utils/string.hpp"
|
|
|
|
#include "modules/meta/base.inl"
|
|
|
|
POLYBAR_NS
|
|
|
|
// Columns in /proc/self/mountinfo
|
|
#define MOUNTINFO_DIR 4
|
|
#define MOUNTINFO_TYPE 7
|
|
#define MOUNTINFO_FSNAME 8
|
|
|
|
namespace modules {
|
|
template class module<fs_module>;
|
|
|
|
/**
|
|
* Bootstrap the module by reading config values and
|
|
* setting up required components
|
|
*/
|
|
fs_module::fs_module(const bar_settings& bar, string name_) : timer_module<fs_module>(bar, move(name_)) {
|
|
m_mountpoints = m_conf.get_list(name(), "mount");
|
|
m_remove_unmounted = m_conf.get(name(), "remove-unmounted", m_remove_unmounted);
|
|
m_fixed = m_conf.get(name(), "fixed-values", m_fixed);
|
|
m_spacing = m_conf.get(name(), "spacing", m_spacing);
|
|
m_interval = m_conf.get<decltype(m_interval)>(name(), "interval", 30s);
|
|
|
|
// Add formats and elements
|
|
m_formatter->add(
|
|
FORMAT_MOUNTED, TAG_LABEL_MOUNTED, {TAG_LABEL_MOUNTED, TAG_BAR_FREE, TAG_BAR_USED, TAG_RAMP_CAPACITY});
|
|
m_formatter->add(FORMAT_UNMOUNTED, TAG_LABEL_UNMOUNTED, {TAG_LABEL_UNMOUNTED});
|
|
|
|
if (m_formatter->has(TAG_LABEL_MOUNTED)) {
|
|
m_labelmounted = load_optional_label(m_conf, name(), TAG_LABEL_MOUNTED, "%mountpoint% %percentage_free%%");
|
|
}
|
|
if (m_formatter->has(TAG_LABEL_UNMOUNTED)) {
|
|
m_labelunmounted = load_optional_label(m_conf, name(), TAG_LABEL_UNMOUNTED, "%mountpoint% is not mounted");
|
|
}
|
|
if (m_formatter->has(TAG_BAR_FREE)) {
|
|
m_barfree = load_progressbar(m_bar, m_conf, name(), TAG_BAR_FREE);
|
|
}
|
|
if (m_formatter->has(TAG_BAR_USED)) {
|
|
m_barused = load_progressbar(m_bar, m_conf, name(), TAG_BAR_USED);
|
|
}
|
|
if (m_formatter->has(TAG_RAMP_CAPACITY)) {
|
|
m_rampcapacity = load_ramp(m_conf, name(), TAG_RAMP_CAPACITY);
|
|
}
|
|
|
|
// Warn about "unreachable" format tag
|
|
if (m_formatter->has(TAG_LABEL_UNMOUNTED) && m_remove_unmounted) {
|
|
m_log.warn("%s: Defined format tag \"%s\" will never be used (reason: `remove-unmounted = true`)", name(),
|
|
TAG_LABEL_UNMOUNTED);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update mountpoints
|
|
*/
|
|
bool fs_module::update() {
|
|
m_mounts.clear();
|
|
|
|
vector<vector<string>> mountinfo;
|
|
std::ifstream filestream("/proc/self/mountinfo");
|
|
string line;
|
|
|
|
// Get details for mounted filesystems
|
|
while (std::getline(filestream, line)) {
|
|
auto cols = string_util::split(line, ' ');
|
|
if (std::find(m_mountpoints.begin(), m_mountpoints.end(), cols[MOUNTINFO_DIR]) != m_mountpoints.end()) {
|
|
mountinfo.emplace_back(move(cols));
|
|
}
|
|
}
|
|
|
|
// Get data for defined mountpoints
|
|
for (auto&& mountpoint : m_mountpoints) {
|
|
auto details = std::find_if(mountinfo.begin(), mountinfo.end(),
|
|
[&](const vector<string>& m) { return m.size() > 4 && m[4] == mountpoint; });
|
|
|
|
m_mounts.emplace_back(new fs_mount{mountpoint, details != mountinfo.end()});
|
|
struct statvfs buffer {};
|
|
|
|
if (!m_mounts.back()->mounted) {
|
|
m_log.warn("%s: Mountpoint %s is not mounted", name(), mountpoint);
|
|
} else if (statvfs(mountpoint.c_str(), &buffer) == -1) {
|
|
m_log.err("%s: Failed to query filesystem (statvfs() error: %s)", name(), strerror(errno));
|
|
} else {
|
|
auto& mount = m_mounts.back();
|
|
mount->mountpoint = details->at(MOUNTINFO_DIR);
|
|
mount->type = details->at(MOUNTINFO_TYPE);
|
|
mount->fsname = details->at(MOUNTINFO_FSNAME);
|
|
|
|
// see: http://en.cppreference.com/w/cpp/filesystem/space
|
|
mount->bytes_total = static_cast<uint64_t>(buffer.f_frsize) * static_cast<uint64_t>(buffer.f_blocks);
|
|
mount->bytes_free = static_cast<uint64_t>(buffer.f_frsize) * static_cast<uint64_t>(buffer.f_bfree);
|
|
mount->bytes_used = mount->bytes_total - mount->bytes_free;
|
|
mount->bytes_avail = static_cast<uint64_t>(buffer.f_frsize) * static_cast<uint64_t>(buffer.f_bavail);
|
|
|
|
mount->percentage_free = math_util::percentage<double>(mount->bytes_avail, mount->bytes_used + mount->bytes_avail);
|
|
mount->percentage_used = math_util::percentage<double>(mount->bytes_used, mount->bytes_used + mount->bytes_avail);
|
|
}
|
|
}
|
|
|
|
if (m_remove_unmounted) {
|
|
for (auto&& mount : m_mounts) {
|
|
if (!mount->mounted) {
|
|
m_log.info("%s: Removing mountpoint \"%s\" (reason: `remove-unmounted = true`)", name(), mount->mountpoint);
|
|
m_mountpoints.erase(
|
|
std::remove(m_mountpoints.begin(), m_mountpoints.end(), mount->mountpoint), m_mountpoints.end());
|
|
m_mounts.erase(std::remove(m_mounts.begin(), m_mounts.end(), mount), m_mounts.end());
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Generate the module output
|
|
*/
|
|
string fs_module::get_output() {
|
|
string output;
|
|
|
|
for (m_index = 0_z; m_index < m_mounts.size(); ++m_index) {
|
|
if (!output.empty()) {
|
|
m_builder->space(m_spacing);
|
|
}
|
|
output += timer_module::get_output();
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
/**
|
|
* Select format based on fs state
|
|
*/
|
|
string fs_module::get_format() const {
|
|
return m_mounts[m_index]->mounted ? FORMAT_MOUNTED : FORMAT_UNMOUNTED;
|
|
}
|
|
|
|
/**
|
|
* Output content using configured format tags
|
|
*/
|
|
bool fs_module::build(builder* builder, const string& tag) const {
|
|
auto& mount = m_mounts[m_index];
|
|
|
|
if (tag == TAG_BAR_FREE) {
|
|
builder->node(m_barfree->output(mount->percentage_free));
|
|
} else if (tag == TAG_BAR_USED) {
|
|
builder->node(m_barused->output(mount->percentage_used));
|
|
} else if (tag == TAG_RAMP_CAPACITY) {
|
|
builder->node(m_rampcapacity->get_by_percentage(mount->percentage_free));
|
|
} else if (tag == TAG_LABEL_MOUNTED) {
|
|
m_labelmounted->reset_tokens();
|
|
m_labelmounted->replace_token("%mountpoint%", mount->mountpoint);
|
|
m_labelmounted->replace_token("%type%", mount->type);
|
|
m_labelmounted->replace_token("%fsname%", mount->fsname);
|
|
m_labelmounted->replace_token("%percentage_free%", to_string(mount->percentage_free));
|
|
m_labelmounted->replace_token("%percentage_used%", to_string(mount->percentage_used));
|
|
m_labelmounted->replace_token(
|
|
"%total%", string_util::filesize(mount->bytes_total, m_fixed ? 2 : 0, m_fixed, m_bar.locale));
|
|
m_labelmounted->replace_token(
|
|
"%free%", string_util::filesize(mount->bytes_avail, m_fixed ? 2 : 0, m_fixed, m_bar.locale));
|
|
m_labelmounted->replace_token(
|
|
"%used%", string_util::filesize(mount->bytes_used, m_fixed ? 2 : 0, m_fixed, m_bar.locale));
|
|
builder->node(m_labelmounted);
|
|
} else if (tag == TAG_LABEL_UNMOUNTED) {
|
|
m_labelunmounted->reset_tokens();
|
|
m_labelunmounted->replace_token("%mountpoint%", mount->mountpoint);
|
|
builder->node(m_labelunmounted);
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
POLYBAR_NS_END
|