feat: add network module
parent
b8917e9973
commit
731a352b41
|
@ -6,6 +6,7 @@
|
||||||
#include "modules/battery.hpp"
|
#include "modules/battery.hpp"
|
||||||
#include "modules/memory.hpp"
|
#include "modules/memory.hpp"
|
||||||
#include "modules/cpu.hpp"
|
#include "modules/cpu.hpp"
|
||||||
|
#include "modules/network.hpp"
|
||||||
|
|
||||||
namespace waybar {
|
namespace waybar {
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <net/if.h>
|
||||||
|
#include <netlink/netlink.h>
|
||||||
|
#include <netlink/genl/genl.h>
|
||||||
|
#include <netlink/genl/ctrl.h>
|
||||||
|
#include <linux/nl80211.h>
|
||||||
|
#include <iwlib.h> // TODO
|
||||||
|
#include <json/json.h>
|
||||||
|
#include <gtkmm.h>
|
||||||
|
#include <fmt/format.h>
|
||||||
|
#include <thread>
|
||||||
|
#include "util/chrono.hpp"
|
||||||
|
#include "IModule.hpp"
|
||||||
|
|
||||||
|
namespace waybar::modules {
|
||||||
|
|
||||||
|
class Network : public IModule {
|
||||||
|
public:
|
||||||
|
Network(Json::Value config);
|
||||||
|
auto update() -> void;
|
||||||
|
operator Gtk::Widget &();
|
||||||
|
typedef struct {
|
||||||
|
int flags;
|
||||||
|
char essid[IW_ESSID_MAX_SIZE + 1];
|
||||||
|
uint8_t bssid[ETH_ALEN];
|
||||||
|
int quality;
|
||||||
|
int quality_max;
|
||||||
|
int quality_average;
|
||||||
|
int signal_level;
|
||||||
|
int signal_level_max;
|
||||||
|
int noise_level;
|
||||||
|
int noise_level_max;
|
||||||
|
int bitrate;
|
||||||
|
double frequency;
|
||||||
|
} wireless_info_t;
|
||||||
|
private:
|
||||||
|
void _parseEssid(struct nlattr** bss);
|
||||||
|
bool _associatedOrJoined(struct nlattr **bss);
|
||||||
|
static int _scanCb(struct nl_msg *msg, void *data);
|
||||||
|
auto _getInfo() -> void;
|
||||||
|
Gtk::Label _label;
|
||||||
|
waybar::util::SleeperThread _thread;
|
||||||
|
Json::Value _config;
|
||||||
|
std::size_t _ifid;
|
||||||
|
std::string _essid;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -26,6 +26,8 @@ wlroots = dependency('wlroots', fallback: ['wlroots', 'wlroots'])
|
||||||
gtkmm = dependency('gtkmm-3.0')
|
gtkmm = dependency('gtkmm-3.0')
|
||||||
jsoncpp = dependency('jsoncpp')
|
jsoncpp = dependency('jsoncpp')
|
||||||
sigcpp = dependency('sigc++-2.0')
|
sigcpp = dependency('sigc++-2.0')
|
||||||
|
libnl = dependency('libnl-3.0')
|
||||||
|
libnlgen = dependency('libnl-genl-3.0')
|
||||||
|
|
||||||
subdir('protocol')
|
subdir('protocol')
|
||||||
|
|
||||||
|
@ -43,6 +45,8 @@ executable(
|
||||||
libinput,
|
libinput,
|
||||||
wayland_cursor,
|
wayland_cursor,
|
||||||
gtkmm,
|
gtkmm,
|
||||||
|
libnl,
|
||||||
|
libnlgen,
|
||||||
],
|
],
|
||||||
include_directories: [include_directories('include')],
|
include_directories: [include_directories('include')],
|
||||||
install: true,
|
install: true,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"modules-left": ["workspaces"],
|
"modules-left": ["workspaces"],
|
||||||
"modules-right": ["cpu", "memory", "battery", "clock"],
|
"modules-right": ["network", "cpu", "memory", "battery", "clock"],
|
||||||
"cpu": {
|
"cpu": {
|
||||||
"format": "{}% "
|
"format": "{}% "
|
||||||
},
|
},
|
||||||
|
@ -9,5 +9,9 @@
|
||||||
},
|
},
|
||||||
"battery": {
|
"battery": {
|
||||||
"format": "{}% "
|
"format": "{}% "
|
||||||
|
},
|
||||||
|
"network": {
|
||||||
|
"interface": "wlp2s0",
|
||||||
|
"format": "{} "
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ window {
|
||||||
border-bottom: 3px solid white;
|
border-bottom: 3px solid white;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clock, .battery, .cpu, .memory {
|
.clock, .battery, .cpu, .memory, .network {
|
||||||
padding: 0 10px;
|
padding: 0 10px;
|
||||||
margin: 0 5px;
|
margin: 0 5px;
|
||||||
}
|
}
|
||||||
|
@ -50,3 +50,7 @@ window {
|
||||||
.memory {
|
.memory {
|
||||||
background: #9b59b6;
|
background: #9b59b6;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.network {
|
||||||
|
background: #2980b9;
|
||||||
|
}
|
||||||
|
|
|
@ -16,5 +16,7 @@ waybar::IModule &waybar::Factory::makeModule(std::string name)
|
||||||
return *new waybar::modules::Cpu(_config[name]);
|
return *new waybar::modules::Cpu(_config[name]);
|
||||||
if (name == "clock")
|
if (name == "clock")
|
||||||
return *new waybar::modules::Clock(_config[name]);
|
return *new waybar::modules::Clock(_config[name]);
|
||||||
|
if (name == "network")
|
||||||
|
return *new waybar::modules::Network(_config[name]);
|
||||||
throw std::runtime_error("Unknown module: " + name);
|
throw std::runtime_error("Unknown module: " + name);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,123 @@
|
||||||
|
#include "modules/network.hpp"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
waybar::modules::Network::Network(Json::Value config)
|
||||||
|
: _config(config), _ifid(if_nametoindex(config["interface"].asString().c_str()))
|
||||||
|
{
|
||||||
|
_label.get_style_context()->add_class("network");
|
||||||
|
_thread = [this] {
|
||||||
|
update();
|
||||||
|
_thread.sleep_for(chrono::minutes(1));
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
auto waybar::modules::Network::update() -> void
|
||||||
|
{
|
||||||
|
_getInfo();
|
||||||
|
auto format = _config["format"] ? _config["format"].asString() : "{}";
|
||||||
|
_label.set_text(fmt::format(format, _essid));
|
||||||
|
}
|
||||||
|
|
||||||
|
int waybar::modules::Network::_scanCb(struct nl_msg *msg, void *data) {
|
||||||
|
auto net = static_cast<waybar::modules::Network *>(data);
|
||||||
|
auto gnlh = static_cast<genlmsghdr *>(nlmsg_data(nlmsg_hdr(msg)));
|
||||||
|
struct nlattr* tb[NL80211_ATTR_MAX + 1];
|
||||||
|
struct nlattr* bss[NL80211_BSS_MAX + 1];
|
||||||
|
struct nla_policy bss_policy[NL80211_BSS_MAX + 1]{};
|
||||||
|
bss_policy[NL80211_BSS_TSF].type = NLA_U64;
|
||||||
|
bss_policy[NL80211_BSS_FREQUENCY].type = NLA_U32;
|
||||||
|
bss_policy[NL80211_BSS_BSSID].type = NLA_UNSPEC;
|
||||||
|
bss_policy[NL80211_BSS_BEACON_INTERVAL].type = NLA_U16;
|
||||||
|
bss_policy[NL80211_BSS_CAPABILITY].type = NLA_U16;
|
||||||
|
bss_policy[NL80211_BSS_INFORMATION_ELEMENTS].type = NLA_UNSPEC;
|
||||||
|
bss_policy[NL80211_BSS_SIGNAL_MBM].type = NLA_U32;
|
||||||
|
bss_policy[NL80211_BSS_SIGNAL_UNSPEC].type = NLA_U8;
|
||||||
|
bss_policy[NL80211_BSS_STATUS].type = NLA_U32;
|
||||||
|
|
||||||
|
if (nla_parse(tb, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0), genlmsg_attrlen(gnlh, 0), nullptr) < 0)
|
||||||
|
return NL_SKIP;
|
||||||
|
if (!tb[NL80211_ATTR_BSS])
|
||||||
|
return NL_SKIP;
|
||||||
|
if (nla_parse_nested(bss, NL80211_BSS_MAX, tb[NL80211_ATTR_BSS], bss_policy))
|
||||||
|
return NL_SKIP;
|
||||||
|
if (!net->_associatedOrJoined(bss))
|
||||||
|
return NL_SKIP;
|
||||||
|
net->_parseEssid(bss);
|
||||||
|
// TODO: parse signal
|
||||||
|
return NL_SKIP;
|
||||||
|
}
|
||||||
|
|
||||||
|
void waybar::modules::Network::_parseEssid(struct nlattr **bss)
|
||||||
|
{
|
||||||
|
_essid.clear();
|
||||||
|
if (bss[NL80211_BSS_INFORMATION_ELEMENTS] != nullptr) {
|
||||||
|
auto ies =
|
||||||
|
static_cast<char*>(nla_data(bss[NL80211_BSS_INFORMATION_ELEMENTS]));
|
||||||
|
auto ies_len = nla_len(bss[NL80211_BSS_INFORMATION_ELEMENTS]);
|
||||||
|
const auto hdr_len = 2;
|
||||||
|
while (ies_len > hdr_len && ies[0] != 0) {
|
||||||
|
ies_len -= ies[1] + hdr_len;
|
||||||
|
ies += ies[1] + hdr_len;
|
||||||
|
}
|
||||||
|
if (ies_len > hdr_len && ies_len > ies[1] + hdr_len) {
|
||||||
|
auto essid_begin = ies + hdr_len;
|
||||||
|
auto essid_end = essid_begin + ies[1];
|
||||||
|
// Only use printable characters of the current locale
|
||||||
|
std::copy_if(essid_begin, essid_end, std::back_inserter(_essid),
|
||||||
|
[](char c) { return isprint(static_cast<unsigned char>(c)); });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool waybar::modules::Network::_associatedOrJoined(struct nlattr** bss)
|
||||||
|
{
|
||||||
|
if (!bss[NL80211_BSS_STATUS])
|
||||||
|
return false;
|
||||||
|
auto status = nla_get_u32(bss[NL80211_BSS_STATUS]);
|
||||||
|
switch (status) {
|
||||||
|
case NL80211_BSS_STATUS_ASSOCIATED:
|
||||||
|
case NL80211_BSS_STATUS_IBSS_JOINED:
|
||||||
|
case NL80211_BSS_STATUS_AUTHENTICATED:
|
||||||
|
return true;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto waybar::modules::Network::_getInfo() -> void
|
||||||
|
{
|
||||||
|
if (_ifid == 0)
|
||||||
|
return;
|
||||||
|
struct nl_sock *sk = nl_socket_alloc();
|
||||||
|
if (genl_connect(sk) != 0) {
|
||||||
|
nl_socket_free(sk);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (nl_socket_modify_cb(sk, NL_CB_VALID, NL_CB_CUSTOM, _scanCb, this) < 0) {
|
||||||
|
nl_socket_free(sk);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const int nl80211_id = genl_ctrl_resolve(sk, "nl80211");
|
||||||
|
if (nl80211_id < 0) {
|
||||||
|
nl_socket_free(sk);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
struct nl_msg *msg = nlmsg_alloc();
|
||||||
|
if (!msg) {
|
||||||
|
nl_socket_free(sk);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!genlmsg_put(msg, NL_AUTO_PORT, NL_AUTO_SEQ, nl80211_id, 0,
|
||||||
|
NLM_F_DUMP, NL80211_CMD_GET_SCAN, 0) ||
|
||||||
|
nla_put_u32(msg, NL80211_ATTR_IFINDEX, _ifid) < 0) {
|
||||||
|
nlmsg_free(msg);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
nl_send_sync(sk, msg);
|
||||||
|
nl_socket_free(sk);
|
||||||
|
}
|
||||||
|
|
||||||
|
waybar::modules::Network::operator Gtk::Widget &() {
|
||||||
|
return _label;
|
||||||
|
}
|
Loading…
Reference in New Issue