3 * Copyright 2009 Roy Marples <roy@marples.name>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 /* TODO: Animate the icon from carrier -> address
28 * maybe use network-idle -> network-transmit ->
29 * network-receive -> network-transmit-receive */
35 #include <libnotify/notify.h>
37 #include "dhcpcd-gtk.h"
40 DBusGProxy *dbus = NULL;
41 GList *interfaces = NULL;
43 static GtkStatusIcon *status_icon;
44 static gint ani_timer;
45 static int ani_counter;
46 static gboolean online;
47 static gboolean carrier;
48 static char **interface_order;
49 static NotifyNotification *nn;
51 const char *const up_reasons[] = {
62 const char *const down_reasons[] = {
72 ignore_if_msg(const struct if_msg *ifm)
74 if (g_strcmp0(ifm->reason, "STOP") == 0 ||
75 g_strcmp0(ifm->reason, "RELEASE") == 0)
80 static struct if_msg *
81 find_if_msg(const char *iface)
86 for (gl = interfaces; gl; gl = gl->next) {
87 ifm = (struct if_msg *)gl->data;
88 if (g_strcmp0(ifm->name, iface) == 0)
95 free_if_ap(struct if_ap *ifa)
104 free_if_msg(struct if_msg *ifm)
111 for (gl = ifm->scan_results; gl; gl = gl->next)
112 free_if_ap((struct if_ap *)gl->data);
113 g_slist_free(ifm->scan_results);
118 error_exit(const char *msg, GError *error)
123 g_critical("%s: %s", msg, error->message);
124 dialog = gtk_message_dialog_new(NULL,
132 g_critical("%s", msg);
133 dialog = gtk_message_dialog_new(NULL,
140 gtk_dialog_run(GTK_DIALOG(dialog));
141 gtk_widget_destroy(dialog);
142 if (gtk_main_level())
149 get_scan_results(struct if_msg *ifm)
160 otype = dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE);
161 otype = dbus_g_type_get_collection("GPtrArray", otype);
163 if (!dbus_g_proxy_call(dbus, "GetScanResults", &error,
164 G_TYPE_STRING, ifm->name, G_TYPE_INVALID,
165 otype, &array, G_TYPE_INVALID))
166 error_exit(_("GetScanResults"), error);
168 for (i = 0; i < array->len; i++) {
169 config = g_ptr_array_index(array, i);
170 val = g_hash_table_lookup(config, "BSSID");
173 ifa = g_malloc0(sizeof(*ifa));
174 ifa->bssid = g_strdup(g_value_get_string(val));
175 val = g_hash_table_lookup(config, "Frequency");
177 ifa->frequency = g_value_get_int(val);
178 val = g_hash_table_lookup(config, "Quality");
180 ifa->quality = g_value_get_int(val);
181 val = g_hash_table_lookup(config, "Noise");
183 ifa->noise = g_value_get_int(val);
184 val = g_hash_table_lookup(config, "Level");
186 ifa->level = g_value_get_int(val);
187 val = g_hash_table_lookup(config, "Flags");
189 ifa->flags = g_strdup(g_value_get_string(val));
190 val = g_hash_table_lookup(config, "SSID");
192 ifa->ssid = g_strdup(g_value_get_string(val));
193 list = g_slist_append(list, ifa);
198 static struct if_msg *
199 make_if_msg(GHashTable *config)
204 val = g_hash_table_lookup(config, "Interface");
207 ifm = g_malloc0(sizeof(*ifm));
208 ifm->name = g_strdup(g_value_get_string(val));
209 val = g_hash_table_lookup(config, "Reason");
211 ifm->reason = g_strdup(g_value_get_string(val));
212 val = g_hash_table_lookup(config, "Wireless");
214 ifm->wireless = g_value_get_boolean(val);
216 val = g_hash_table_lookup(config, "SSID");
218 ifm->ssid = g_strdup(g_value_get_string(val));
220 val = g_hash_table_lookup(config, "IPAddress");
222 ifm->ip.s_addr = g_value_get_uint(val);
223 val = g_hash_table_lookup(config, "SubnetCIDR");
225 ifm->cidr = g_value_get_uchar(val);
226 val = g_hash_table_lookup(config, "InterfaceOrder");
228 g_strfreev(interface_order);
229 interface_order = g_strsplit(g_value_get_string(val), " ", 0);
235 if_up(const struct if_msg *ifm)
237 const char *const *r;
239 for (r = up_reasons; *r; r++)
240 if (g_strcmp0(*r, ifm->reason) == 0)
246 print_if_msg(const struct if_msg *ifm)
249 const char *reason = NULL;
251 gboolean showip, showssid;
256 reason = N_("Acquired address");
258 if (g_strcmp0(ifm->reason, "EXPIRE") == 0)
259 reason = N_("Failed to renew");
260 else if (g_strcmp0(ifm->reason, "CARRIER") == 0) {
262 reason = N_("Asssociated with");
263 if (ifm->ssid != NULL)
266 reason = N_("Cable plugged in");
268 } else if (g_strcmp0(ifm->reason, "NOCARRIER") == 0) {
270 if (ifm->ssid != NULL || ifm->ip.s_addr != 0) {
271 reason = N_("Disassociated from");
274 reason = N_("Not associated");
276 reason = N_("Cable unplugged");
281 reason = ifm->reason;
283 len = strlen(ifm->name) + 3;
284 len += strlen(reason) + 1;
285 if (ifm->ip.s_addr != 0) {
286 len += 16; /* 000. * 4 */
291 len += strlen(ifm->ssid) + 1;
292 msg = p = g_malloc(len);
293 p += g_snprintf(msg, len, "%s: %s", ifm->name, reason);
295 p += g_snprintf(p, len - (p - msg), " %s", ifm->ssid);
296 if (ifm->ip.s_addr != 0 && showip) {
297 p += g_snprintf(p, len - (p - msg), " %s", inet_ntoa(ifm->ip));
299 g_snprintf(p, len - (p - msg), "/%d", ifm->cidr);
305 if_msg_comparer(gconstpointer a, gconstpointer b)
307 const struct if_msg *ifa, *ifb;
308 const char *const *order;
310 ifa = (const struct if_msg *)a;
311 ifb = (const struct if_msg *)b;
312 for (order = (const char *const *)interface_order; *order; order++) {
313 if (g_strcmp0(*order, ifa->name) == 0)
315 if (g_strcmp0(*order, ifb->name) == 0)
322 animate_carrier(_unused gpointer data)
327 switch(ani_counter++) {
329 gtk_status_icon_set_from_icon_name(status_icon, "network-transmit");
332 gtk_status_icon_set_from_icon_name(status_icon, "network-receive");
335 gtk_status_icon_set_from_icon_name(status_icon, "network-idle");
343 animate_online(_unused gpointer data)
348 if (ani_counter++ > 6) {
354 if (ani_counter % 2 == 0)
355 gtk_status_icon_set_from_icon_name(status_icon, "network-idle");
357 gtk_status_icon_set_from_icon_name(status_icon, "network-transmit-receive");
362 update_online(char **buffer)
364 gboolean ison, iscarrier;
365 char *msg, *msgs, *tmp;
367 const struct if_msg *ifm;
369 ison = iscarrier = FALSE;
371 for (gl = interfaces; gl; gl = gl->next) {
372 ifm = (const struct if_msg *)gl->data;
374 ison = iscarrier = TRUE;
375 if (!iscarrier && g_strcmp0(ifm->reason, "CARRIER") == 0)
377 msg = print_if_msg(ifm);
379 tmp = g_strconcat(msgs, "\n", msg, NULL);
387 if (online != ison || carrier != iscarrier) {
389 if (ani_timer != 0) {
390 g_source_remove(ani_timer);
395 animate_online(NULL);
396 ani_timer = g_timeout_add(300, animate_online, NULL);
397 } else if (iscarrier) {
398 animate_carrier(NULL);
399 ani_timer = g_timeout_add(500, animate_carrier, NULL);
401 gtk_status_icon_set_from_icon_name(status_icon,
405 gtk_status_icon_set_tooltip(status_icon, msgs);
416 notify_notification_close(nn, NULL);
426 notify(const char *title, const char *msg, const char *icon)
430 msgs = g_strsplit(msg, "\n", 0);
431 for (m = msgs; *m; m++)
435 notify_notification_close(nn, NULL);
436 if (gtk_status_icon_get_visible(status_icon))
437 nn = notify_notification_new_with_status_icon(title,
442 nn = notify_notification_new(title, msg, icon, NULL);
443 notify_notification_set_timeout(nn, 5000);
444 g_signal_connect(nn, "closed", G_CALLBACK(notify_closed), NULL);
445 notify_notification_show(nn, NULL);
449 dhcpcd_event(_unused DBusGProxy *proxy, GHashTable *config, _unused void *data)
451 struct if_msg *ifm, *ifp;
455 const char *act, *net;
456 const char *const *r;
459 ifm = make_if_msg(config);
463 rem = ignore_if_msg(ifm);
465 for (gl = interfaces; gl; gl = gl->next) {
466 ifp = (struct if_msg *)gl->data;
467 if (g_strcmp0(ifp->name, ifm->name) == 0) {
468 ifm->scan_results = ifp->scan_results;
469 ifp->scan_results = NULL;
472 interfaces = g_list_delete_link(interfaces, gl);
478 if (ifp == NULL && !rem)
479 interfaces = g_list_prepend(interfaces, ifm);
480 interfaces = g_list_sort(interfaces, if_msg_comparer);
483 /* We should ignore renew and stop so we don't annoy the user */
484 if (g_strcmp0(ifm->reason, "RENEW") == 0 ||
485 g_strcmp0(ifm->reason, "STOP") == 0)
488 msg = print_if_msg(ifm);
491 act = N_("Connected to ");
494 for (r = down_reasons; *r; r++) {
495 if (g_strcmp0(*r, ifm->reason) == 0) {
496 act = N_("Disconnected from ");
500 if (act && ifm->ip.s_addr) {
501 ipn = htonl(ifm->ip.s_addr);
502 if (IN_LINKLOCAL(ipn))
503 net = N_("private network");
504 else if (IN_PRIVATE(ipn))
507 net = N_("internet");
508 title = g_strconcat(act, net, NULL);
512 notify(title, msg, GTK_STOCK_NETWORK);
515 notify(N_("Interface event"), msg, GTK_STOCK_NETWORK);
520 foreach_make_ifm(_unused gpointer key, gpointer value, _unused gpointer data)
524 ifm = make_if_msg((GHashTable *)value);
525 if (ignore_if_msg(ifm))
528 interfaces = g_list_prepend(interfaces, ifm);
532 dhcpcd_get_interfaces()
535 GError *error = NULL;
543 otype = dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE);
544 otype = dbus_g_type_get_map("GHashTable", G_TYPE_STRING, otype);
545 if (!dbus_g_proxy_call(dbus, "GetInterfaces", &error,
547 otype, &ifs, G_TYPE_INVALID))
548 error_exit("GetInterfaces", error);
549 g_hash_table_foreach(ifs, foreach_make_ifm, NULL);
550 g_hash_table_unref(ifs);
552 /* Each interface config only remembers the last order when
553 * that interface was configured, so get the real order now. */
554 g_strfreev(interface_order);
555 interface_order = NULL;
556 if (!dbus_g_proxy_call(dbus, "ListInterfaces", &error,
558 G_TYPE_STRV, &interface_order, G_TYPE_INVALID))
559 error_exit("ListInterfaces", error);
560 interfaces = g_list_sort(interfaces, if_msg_comparer);
562 otype = dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE);
563 otype = dbus_g_type_get_collection("GPtrArray", otype);
564 for (gl = interfaces; gl; gl = gl->next) {
565 ifm = (struct if_msg *)gl->data;
568 if (!dbus_g_proxy_call(dbus, "GetScanResults", &error,
569 G_TYPE_STRING, ifm->name, G_TYPE_INVALID,
570 otype, &array, G_TYPE_INVALID))
572 g_message("GetScanResults: %s", error->message);
573 g_clear_error(&error);
576 for (gsl = ifm->scan_results; gsl; gsl = gsl->next)
578 g_slist_free(ifm->scan_results);
579 ifm->scan_results = get_scan_results(ifm);
584 // GTK+ 2.16 msg = gtk_status_icon_get_tooltip_text(status_icon);
586 notify(N_("Interface status"), msg, GTK_STOCK_NETWORK);
592 check_status(const char *status)
594 static char *last = NULL;
599 GError *error = NULL;
601 g_message("Status changed to %s", status);
602 if (g_strcmp0(status, "down") == 0) {
603 for (gl = interfaces; gl; gl = gl->next)
604 free_if_msg((struct if_msg *)gl->data);
605 g_list_free(interfaces);
608 msg = N_(last? "Connection to dhcpcd lost" : "dhcpcd not running");
609 gtk_status_icon_set_tooltip(status_icon, msg);
610 notify(_("No network"), msg, GTK_STOCK_NETWORK);
615 if (g_strcmp0(status, "down") != 0)
618 if (g_strcmp0(status, last) == 0)
620 if (g_strcmp0(last, "down") == 0)
624 last = g_strdup(status);
628 if (!dbus_g_proxy_call(dbus, "GetDhcpcdVersion", &error,
630 G_TYPE_STRING, &version, G_TYPE_INVALID))
631 error_exit(_("GetDhcpcdVersion"), error);
632 g_message(_("Connected to %s-%s"), "dhcpcd", version);
634 dhcpcd_get_interfaces();
638 dhcpcd_status(_unused DBusGProxy *proxy, const char *status, _unused void *data)
640 check_status(status);
644 dhcpcd_scan_results(_unused DBusGProxy *proxy, const char *iface, _unused void *data)
647 struct if_ap *ifa, *ifa2;
648 GSList *gl, *aps, *l;
651 ifm = find_if_msg(iface);
654 g_message(_("%s: Received scan results"), ifm->name);
655 aps = get_scan_results(ifm);
657 for (gl = aps; gl; gl = gl->next) {
658 ifa = (struct if_ap *)gl->data;
659 for (l = ifm->scan_results; l; l = l->next) {
660 ifa2 = (struct if_ap *)l->data;
661 if (g_strcmp0(ifa->ssid, ifa2->ssid) == 0)
666 txt = g_strdup(ifa->ssid);
668 ntxt = g_strconcat(txt, "\n", ifa->ssid, NULL);
674 for (gl = ifm->scan_results; gl; gl = gl->next)
675 free_if_ap((struct if_ap *)gl->data);
676 g_slist_free(ifm->scan_results);
677 ifm->scan_results = aps;
679 notify(N_("Found new AP"), txt, GTK_STOCK_NETWORK);
685 main(int argc, char *argv[])
687 DBusGConnection *bus;
688 GError *error = NULL;
689 char *version = NULL;
694 setlocale(LC_ALL, "");
695 bindtextdomain(PACKAGE, NULL);
696 bind_textdomain_codeset(PACKAGE, "UTF-8");
699 gtk_init(&argc, &argv);
700 g_set_application_name("dhcpcd Monitor");
701 status_icon = gtk_status_icon_new_from_icon_name("network-offline");
702 if (status_icon == NULL)
703 status_icon = gtk_status_icon_new_from_stock(GTK_STOCK_DISCONNECT);
704 //network_offline = gtk_status_icon_get_pixbuf(status_icon);
706 gtk_status_icon_set_tooltip(status_icon, _("Connecting to dhcpcd ..."));
707 gtk_status_icon_set_visible(status_icon, TRUE);
709 notify_init(PACKAGE);
711 g_message(_("Connecting to dbus ..."));
712 bus = dbus_g_bus_get(DBUS_BUS_SYSTEM, &error);
713 if (bus == NULL || error != NULL)
714 error_exit(_("Could not connect to system bus"), error);
715 dbus = dbus_g_proxy_new_for_name(bus,
720 g_message(_("Connecting to dhcpcd-dbus ..."));
721 while (--tries > 0) {
722 g_clear_error(&error);
723 if (dbus_g_proxy_call_with_timeout(dbus,
734 error_exit(_("GetVersion"), error);
735 g_message(_("Connected to %s-%s"), "dhcpcd-dbus", version);
738 gtk_status_icon_set_tooltip(status_icon, _("Triggering dhcpcd ..."));
740 menu_init(status_icon);
742 if (!dbus_g_proxy_call(dbus, "GetStatus", &error,
744 G_TYPE_STRING, &version, G_TYPE_INVALID))
745 error_exit(_("GetStatus"), error);
746 check_status(version);
749 otype = dbus_g_type_get_map("GHashTable", G_TYPE_STRING, G_TYPE_VALUE);
750 dbus_g_proxy_add_signal(dbus, "Event",
751 otype, G_TYPE_INVALID);
752 dbus_g_proxy_connect_signal(dbus, "Event",
753 G_CALLBACK(dhcpcd_event),
755 dbus_g_proxy_add_signal(dbus, "StatusChanged",
756 G_TYPE_STRING, G_TYPE_INVALID);
757 dbus_g_proxy_connect_signal(dbus, "StatusChanged",
758 G_CALLBACK(dhcpcd_status),
760 dbus_g_proxy_add_signal(dbus, "ScanResults",
761 G_TYPE_STRING, G_TYPE_INVALID);
762 dbus_g_proxy_connect_signal(dbus, "ScanResults",
763 G_CALLBACK(dhcpcd_scan_results),