2016-11-02 19:22:45 +00:00
|
|
|
#include "modules/network.hpp"
|
|
|
|
|
2016-11-19 14:49:03 +00:00
|
|
|
#include "drawtypes/animation.hpp"
|
|
|
|
#include "drawtypes/label.hpp"
|
|
|
|
#include "drawtypes/ramp.hpp"
|
2016-11-20 22:04:31 +00:00
|
|
|
#include "modules/meta/base.inl"
|
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
namespace modules {
|
2016-11-20 22:04:31 +00:00
|
|
|
template class module<network_module>;
|
|
|
|
|
2016-12-21 07:00:09 +00:00
|
|
|
network_module::network_module(const bar_settings& bar, string name_)
|
|
|
|
: timer_module<network_module>(bar, move(name_)) {
|
2016-11-02 19:22:45 +00:00
|
|
|
// Load configuration values
|
2016-12-30 22:32:05 +00:00
|
|
|
m_interface = m_conf.get(name(), "interface", m_interface);
|
2021-01-03 10:48:15 +00:00
|
|
|
|
|
|
|
if (m_interface.empty()) {
|
|
|
|
std::string type = m_conf.get(name(), "interface-type");
|
|
|
|
if (type == "wired") {
|
|
|
|
m_interface = net::find_wired_interface();
|
|
|
|
if (!m_interface.empty()) {
|
|
|
|
m_log.notice("%s: Discovered wired interface %s", name(), m_interface);
|
|
|
|
}
|
|
|
|
} else if (type == "wireless") {
|
|
|
|
m_interface = net::find_wireless_interface();
|
|
|
|
if (!m_interface.empty()) {
|
|
|
|
m_log.notice("%s: Discovered wireless interface %s", name(), m_interface);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw module_error("Invalid interface type '" + type + "'");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_interface.empty()) {
|
|
|
|
throw module_error("No interface found for type '" + type + "'");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_interface.empty()) {
|
2021-09-09 19:47:23 +00:00
|
|
|
throw module_error("Missing 'interface' or 'interface-type'");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!net::is_interface_valid(m_interface)) {
|
|
|
|
throw module_error("Invalid network interface \"" + m_interface + "\"");
|
2021-01-03 10:48:15 +00:00
|
|
|
}
|
|
|
|
|
2021-10-15 15:36:32 +00:00
|
|
|
auto canonical = net::get_canonical_interface(m_interface);
|
|
|
|
|
|
|
|
if (canonical.second) {
|
|
|
|
m_log.info(
|
|
|
|
"%s: Replacing given interface '%s' with its canonical name '%s'", name(), m_interface, canonical.first);
|
|
|
|
m_interface = canonical.first;
|
|
|
|
}
|
|
|
|
|
2016-12-30 22:32:05 +00:00
|
|
|
m_ping_nth_update = m_conf.get(name(), "ping-interval", m_ping_nth_update);
|
|
|
|
m_udspeed_minwidth = m_conf.get(name(), "udspeed-minwidth", m_udspeed_minwidth);
|
|
|
|
m_accumulate = m_conf.get(name(), "accumulate-stats", m_accumulate);
|
2020-12-05 21:58:38 +00:00
|
|
|
set_interval(1s);
|
2017-11-03 06:05:26 +00:00
|
|
|
m_unknown_up = m_conf.get<bool>(name(), "unknown-as-up", false);
|
2020-11-29 13:15:27 +00:00
|
|
|
m_udspeed_unit = m_conf.get<string>(name(), "speed-unit", m_udspeed_unit);
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-22 02:01:50 +00:00
|
|
|
m_conf.warn_deprecated(name(), "udspeed-minwidth", "%downspeed:min:max% and %upspeed:min:max%");
|
|
|
|
|
2016-11-02 19:22:45 +00:00
|
|
|
// Add formats
|
2016-11-25 07:42:31 +00:00
|
|
|
m_formatter->add(FORMAT_CONNECTED, TAG_LABEL_CONNECTED, {TAG_RAMP_SIGNAL, TAG_RAMP_QUALITY, TAG_LABEL_CONNECTED});
|
2016-11-02 19:22:45 +00:00
|
|
|
m_formatter->add(FORMAT_DISCONNECTED, TAG_LABEL_DISCONNECTED, {TAG_LABEL_DISCONNECTED});
|
|
|
|
|
|
|
|
// Create elements for format-connected
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_formatter->has(TAG_RAMP_SIGNAL, FORMAT_CONNECTED)) {
|
2016-11-02 19:22:45 +00:00
|
|
|
m_ramp_signal = load_ramp(m_conf, name(), TAG_RAMP_SIGNAL);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
|
|
|
if (m_formatter->has(TAG_RAMP_QUALITY, FORMAT_CONNECTED)) {
|
2016-11-02 19:22:45 +00:00
|
|
|
m_ramp_quality = load_ramp(m_conf, name(), TAG_RAMP_QUALITY);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
if (m_formatter->has(TAG_LABEL_CONNECTED, FORMAT_CONNECTED)) {
|
|
|
|
m_label[connection_state::CONNECTED] =
|
|
|
|
load_optional_label(m_conf, name(), TAG_LABEL_CONNECTED, "%ifname% %local_ip%");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create elements for format-disconnected
|
|
|
|
if (m_formatter->has(TAG_LABEL_DISCONNECTED, FORMAT_DISCONNECTED)) {
|
2016-11-25 07:42:31 +00:00
|
|
|
m_label[connection_state::DISCONNECTED] = load_optional_label(m_conf, name(), TAG_LABEL_DISCONNECTED, "");
|
2016-11-02 19:22:45 +00:00
|
|
|
m_label[connection_state::DISCONNECTED]->reset_tokens();
|
|
|
|
m_label[connection_state::DISCONNECTED]->replace_token("%ifname%", m_interface);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create elements for format-packetloss if we are told to test connectivity
|
|
|
|
if (m_ping_nth_update > 0) {
|
|
|
|
m_formatter->add(FORMAT_PACKETLOSS, TAG_LABEL_CONNECTED,
|
|
|
|
{TAG_ANIMATION_PACKETLOSS, TAG_LABEL_PACKETLOSS, TAG_LABEL_CONNECTED});
|
|
|
|
|
|
|
|
if (m_formatter->has(TAG_LABEL_PACKETLOSS, FORMAT_PACKETLOSS)) {
|
2016-11-25 07:42:31 +00:00
|
|
|
m_label[connection_state::PACKETLOSS] = load_optional_label(m_conf, name(), TAG_LABEL_PACKETLOSS, "");
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_formatter->has(TAG_ANIMATION_PACKETLOSS, FORMAT_PACKETLOSS)) {
|
2016-11-02 19:22:45 +00:00
|
|
|
m_animation_packetloss = load_animation(m_conf, name(), TAG_ANIMATION_PACKETLOSS);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get an intstance of the network interface
|
2016-11-25 12:55:15 +00:00
|
|
|
if (net::is_wireless_interface(m_interface)) {
|
2021-09-21 19:15:49 +00:00
|
|
|
m_wireless = std::make_unique<net::wireless_network>(m_interface);
|
2017-11-03 06:05:26 +00:00
|
|
|
m_wireless->set_unknown_up(m_unknown_up);
|
2016-11-25 12:55:15 +00:00
|
|
|
} else {
|
2021-09-21 19:15:49 +00:00
|
|
|
m_wired = std::make_unique<net::wired_network>(m_interface);
|
2017-11-03 06:05:26 +00:00
|
|
|
m_wired->set_unknown_up(m_unknown_up);
|
2016-11-25 12:55:15 +00:00
|
|
|
};
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
// We only need to start the subthread if the packetloss animation is used
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_animation_packetloss) {
|
2016-11-02 19:22:45 +00:00
|
|
|
m_threads.emplace_back(thread(&network_module::subthread_routine, this));
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void network_module::teardown() {
|
|
|
|
m_wireless.reset();
|
|
|
|
m_wired.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool network_module::update() {
|
2016-11-25 07:42:31 +00:00
|
|
|
net::network* network =
|
|
|
|
m_wireless ? static_cast<net::network*>(m_wireless.get()) : static_cast<net::network*>(m_wired.get());
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-14 11:42:58 +00:00
|
|
|
if (!network->query(m_accumulate)) {
|
2016-11-02 19:22:45 +00:00
|
|
|
m_log.warn("%s: Failed to query interface '%s'", name(), m_interface);
|
2018-04-16 20:19:01 +00:00
|
|
|
m_connected = false;
|
2016-11-02 19:22:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (m_wireless) {
|
|
|
|
m_signal = m_wireless->signal();
|
|
|
|
m_quality = m_wireless->quality();
|
|
|
|
}
|
|
|
|
} catch (const net::network_error& err) {
|
|
|
|
m_log.warn("%s: Error getting interface data (%s)", name(), err.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
m_connected = network->connected();
|
|
|
|
|
|
|
|
// Ignore the first run
|
|
|
|
if (m_counter == -1) {
|
|
|
|
m_counter = 0;
|
|
|
|
} else if (m_ping_nth_update > 0 && m_connected && (++m_counter % m_ping_nth_update) == 0) {
|
|
|
|
m_packetloss = !network->ping();
|
|
|
|
m_counter = 0;
|
|
|
|
}
|
|
|
|
|
2020-11-29 13:15:27 +00:00
|
|
|
auto upspeed = network->upspeed(m_udspeed_minwidth, m_udspeed_unit);
|
|
|
|
auto downspeed = network->downspeed(m_udspeed_minwidth, m_udspeed_unit);
|
2022-02-06 20:12:38 +00:00
|
|
|
auto netspeed = network->netspeed(m_udspeed_minwidth, m_udspeed_unit);
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
// Update label contents
|
|
|
|
const auto replace_tokens = [&](label_t& label) {
|
|
|
|
label->reset_tokens();
|
|
|
|
label->replace_token("%ifname%", m_interface);
|
|
|
|
label->replace_token("%local_ip%", network->ip());
|
2022-01-16 01:39:55 +00:00
|
|
|
label->replace_token("%mac%", network->mac());
|
2018-06-10 20:51:43 +00:00
|
|
|
label->replace_token("%local_ip6%", network->ip6());
|
2016-11-02 19:22:45 +00:00
|
|
|
label->replace_token("%upspeed%", upspeed);
|
|
|
|
label->replace_token("%downspeed%", downspeed);
|
2022-02-06 20:12:38 +00:00
|
|
|
label->replace_token("%netspeed%", netspeed);
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
if (m_wired) {
|
|
|
|
label->replace_token("%linkspeed%", m_wired->linkspeed());
|
|
|
|
} else if (m_wireless) {
|
|
|
|
label->replace_token("%essid%", m_wireless->essid());
|
2017-01-13 19:03:08 +00:00
|
|
|
label->replace_token("%signal%", to_string(m_signal));
|
|
|
|
label->replace_token("%quality%", to_string(m_quality));
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_label[connection_state::CONNECTED]) {
|
2016-11-02 19:22:45 +00:00
|
|
|
replace_tokens(m_label[connection_state::CONNECTED]);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
|
|
|
if (m_label[connection_state::PACKETLOSS]) {
|
2016-11-02 19:22:45 +00:00
|
|
|
replace_tokens(m_label[connection_state::PACKETLOSS]);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2019-01-12 14:48:23 +00:00
|
|
|
if (m_label[connection_state::DISCONNECTED]) {
|
|
|
|
replace_tokens(m_label[connection_state::DISCONNECTED]);
|
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
string network_module::get_format() const {
|
2016-11-25 12:55:15 +00:00
|
|
|
if (!m_connected) {
|
2016-11-02 19:22:45 +00:00
|
|
|
return FORMAT_DISCONNECTED;
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (m_packetloss && m_ping_nth_update > 0) {
|
2016-11-02 19:22:45 +00:00
|
|
|
return FORMAT_PACKETLOSS;
|
2016-11-25 12:55:15 +00:00
|
|
|
} else {
|
2016-11-02 19:22:45 +00:00
|
|
|
return FORMAT_CONNECTED;
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
2016-11-25 12:55:15 +00:00
|
|
|
bool network_module::build(builder* builder, const string& tag) const {
|
|
|
|
if (tag == TAG_LABEL_CONNECTED) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_label.at(connection_state::CONNECTED));
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (tag == TAG_LABEL_DISCONNECTED) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_label.at(connection_state::DISCONNECTED));
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (tag == TAG_LABEL_PACKETLOSS) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_label.at(connection_state::PACKETLOSS));
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (tag == TAG_ANIMATION_PACKETLOSS) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_animation_packetloss->get());
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (tag == TAG_RAMP_SIGNAL) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_ramp_signal->get_by_percentage(m_signal));
|
2016-11-25 12:55:15 +00:00
|
|
|
} else if (tag == TAG_RAMP_QUALITY) {
|
2016-11-02 19:22:45 +00:00
|
|
|
builder->node(m_ramp_quality->get_by_percentage(m_quality));
|
2016-11-25 12:55:15 +00:00
|
|
|
} else {
|
2016-11-02 19:22:45 +00:00
|
|
|
return false;
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void network_module::subthread_routine() {
|
|
|
|
const chrono::milliseconds framerate{m_animation_packetloss->framerate()};
|
|
|
|
|
|
|
|
while (running()) {
|
2019-04-05 01:36:33 +00:00
|
|
|
auto now = chrono::steady_clock::now();
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_connected && m_packetloss) {
|
2019-03-09 05:14:39 +00:00
|
|
|
m_animation_packetloss->increment();
|
2016-11-02 19:22:45 +00:00
|
|
|
broadcast();
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2019-03-09 05:14:39 +00:00
|
|
|
|
|
|
|
now += framerate;
|
|
|
|
this_thread::sleep_until(now);
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_log.trace("%s: Reached end of network subthread", name());
|
|
|
|
}
|
2021-01-03 10:48:15 +00:00
|
|
|
} // namespace modules
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS_END
|