mirror of
https://github.com/Dasharo/systemd.git
synced 2026-03-06 15:02:31 -08:00
networkd: add minimal client tool "networkd" to query network status
In the long run this should become a full fledged client to networkd (but not before networkd learns bus support). For now, just pull interesting data out of networkd, udev, and rtnl and present it to the user, in a simple but useful output.
This commit is contained in:
1
.gitignore
vendored
1
.gitignore
vendored
@@ -43,6 +43,7 @@
|
||||
/loginctl
|
||||
/machinectl
|
||||
/mtd_probe
|
||||
/networkctl
|
||||
/scsi_id
|
||||
/systemadm
|
||||
/systemctl
|
||||
|
||||
36
Makefile.am
36
Makefile.am
@@ -833,6 +833,8 @@ libsystemd_shared_la_SOURCES = \
|
||||
src/shared/errno-list.h \
|
||||
src/shared/af-list.c \
|
||||
src/shared/af-list.h \
|
||||
src/shared/arphrd-list.c \
|
||||
src/shared/arphrd-list.h \
|
||||
src/shared/audit.c \
|
||||
src/shared/audit.h \
|
||||
src/shared/xml.c \
|
||||
@@ -864,7 +866,9 @@ nodist_libsystemd_shared_la_SOURCES = \
|
||||
src/shared/errno-from-name.h \
|
||||
src/shared/errno-to-name.h \
|
||||
src/shared/af-from-name.h \
|
||||
src/shared/af-to-name.h
|
||||
src/shared/af-to-name.h \
|
||||
src/shared/arphrd-from-name.h \
|
||||
src/shared/arphrd-to-name.h
|
||||
|
||||
libsystemd_shared_la_CFLAGS = \
|
||||
$(AM_CFLAGS) \
|
||||
@@ -1150,6 +1154,8 @@ CLEANFILES += \
|
||||
src/shared/errno-from-name.gperf \
|
||||
src/shared/af-list.txt \
|
||||
src/shared/af-from-name.gperf \
|
||||
src/shared/arphrd-list.txt \
|
||||
src/shared/arphrd-from-name.gperf \
|
||||
src/shared/dns_type-list.txt \
|
||||
src/shared/dns_type-from-name.gperf
|
||||
|
||||
@@ -1158,6 +1164,8 @@ BUILT_SOURCES += \
|
||||
src/shared/errno-to-name.h \
|
||||
src/shared/af-from-name.h \
|
||||
src/shared/af-to-name.h \
|
||||
src/shared/arphrd-from-name.h \
|
||||
src/shared/arphrd-to-name.h \
|
||||
src/resolve/dns_type-from-name.h \
|
||||
src/resolve/dns_type-to-name.h
|
||||
|
||||
@@ -1170,7 +1178,7 @@ BUILT_SOURCES += \
|
||||
|
||||
src/shared/errno-list.txt:
|
||||
$(AM_V_at)$(MKDIR_P) $(dir $@)
|
||||
$(AM_V_GEN)$(CPP) $(CFLAGS) $(AM_CPPFLAGS) $(CPPFLAGS) -dM -include errno.h - </dev/null | $(AWK) '/^#define[ \t]+E[^ _]+[ \t]+/ { print $$2; }' >$@
|
||||
$(AM_V_GEN)$(CPP) $(CFLAGS) $(AM_CPPFLAGS) $(CPPFLAGS) -dM -include errno.h - </dev/null | $(AWK) '/^#define[ \t]+E[^ _]+[ \t]+/ { print $$2; }' >$@
|
||||
|
||||
src/shared/errno-to-name.h: src/shared/errno-list.txt
|
||||
$(AM_V_GEN)$(AWK) 'BEGIN{ print "static const char* const errno_names[] = { "} !/EDEADLOCK/ && !/EWOULDBLOCK/ && !/ENOTSUP/ { printf "[%s] = \"%s\",\n", $$1, $$1 } END{print "};"}' <$< >$@
|
||||
@@ -1178,11 +1186,21 @@ src/shared/errno-to-name.h: src/shared/errno-list.txt
|
||||
|
||||
src/shared/af-list.txt:
|
||||
$(AM_V_at)$(MKDIR_P) $(dir $@)
|
||||
$(AM_V_GEN)$(CPP) $(CFLAGS) $(AM_CPPFLAGS) $(CPPFLAGS) -dM -include sys/socket.h - </dev/null | grep -v AF_UNSPEC | grep -v AF_MAX | $(AWK) '/^#define[ \t]+AF_[^ \t]+[ \t]+PF_[^ \t]/ { print $$2; }' >$@
|
||||
$(AM_V_GEN)$(CPP) $(CFLAGS) $(AM_CPPFLAGS) $(CPPFLAGS) -dM -include sys/socket.h - </dev/null | grep -v AF_UNSPEC | grep -v AF_MAX | $(AWK) '/^#define[ \t]+AF_[^ \t]+[ \t]+PF_[^ \t]/ { print $$2; }' >$@
|
||||
|
||||
src/shared/af-to-name.h: src/shared/af-list.txt
|
||||
$(AM_V_GEN)$(AWK) 'BEGIN{ print "static const char* const af_names[] = { "} !/AF_FILE/ && !/AF_ROUTE/ && !/AF_LOCAL/ { printf "[%s] = \"%s\",\n", $$1, $$1 } END{print "};"}' <$< >$@
|
||||
|
||||
src/shared/arphrd-list.txt:
|
||||
$(AM_V_at)$(MKDIR_P) $(dir $@)
|
||||
$(AM_V_GEN)$(CPP) $(CFLAGS) $(AM_CPPFLAGS) $(CPPFLAGS) -dM -include net/if_arp.h - </dev/null | $(AWK) '/^#define[ \t]+ARPHRD_[^ \t]+[ \t]+[^ \t]/ { print $$2; }' | sed -e 's/ARPHRD_//' >$@
|
||||
|
||||
src/shared/arphrd-to-name.h: src/shared/arphrd-list.txt
|
||||
$(AM_V_GEN)$(AWK) 'BEGIN{ print "static const char* const arphrd_names[] = { "} !/CISCO/ { printf "[ARPHRD_%s] = \"%s\",\n", $$1, $$1 } END{print "};"}' <$< >$@
|
||||
|
||||
src/shared/arphrd-from-name.gperf: src/shared/arphrd-list.txt
|
||||
$(AM_V_GEN)$(AWK) 'BEGIN{ print "struct arphrd_name { const char* name; int id; };"; print "%null-strings"; print "%%";} { printf "%s, ARPHRD_%s\n", $$1, $$1 }' <$< >$@
|
||||
|
||||
|
||||
src/resolve/dns_type-list.txt: src/resolve/dns-type.h
|
||||
$(AM_V_at)$(MKDIR_P) $(dir $@)
|
||||
@@ -4974,6 +4992,18 @@ systemd_networkd_wait_online_LDADD = \
|
||||
libsystemd-internal.la \
|
||||
libsystemd-shared.la
|
||||
|
||||
rootbin_PROGRAMS += \
|
||||
networkctl
|
||||
|
||||
networkctl_SOURCES = \
|
||||
src/network/networkctl.c
|
||||
|
||||
networkctl_LDADD = \
|
||||
libsystemd-internal.la \
|
||||
libudev-internal.la \
|
||||
libsystemd-shared.la \
|
||||
libsystemd-network.la
|
||||
|
||||
test_network_SOURCES = \
|
||||
src/network/test-network.c
|
||||
|
||||
|
||||
@@ -1186,12 +1186,12 @@ int bus_property_get_ulong(
|
||||
#endif
|
||||
|
||||
int bus_log_parse_error(int r) {
|
||||
log_error("Failed to parse message: %s", strerror(-r));
|
||||
log_error("Failed to parse bus message: %s", strerror(-r));
|
||||
return r;
|
||||
}
|
||||
|
||||
int bus_log_create_error(int r) {
|
||||
log_error("Failed to create message: %s", strerror(-r));
|
||||
log_error("Failed to create bus message: %s", strerror(-r));
|
||||
return r;
|
||||
}
|
||||
|
||||
|
||||
@@ -48,7 +48,7 @@ static int address_compare(const void *_a, const void *_b) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int local_addresses(struct local_address **ret) {
|
||||
int local_addresses(sd_rtnl *context, int ifindex, struct local_address **ret) {
|
||||
_cleanup_rtnl_message_unref_ sd_rtnl_message *req = NULL, *reply = NULL;
|
||||
_cleanup_rtnl_unref_ sd_rtnl *rtnl = NULL;
|
||||
_cleanup_free_ struct local_address *list = NULL;
|
||||
@@ -58,9 +58,13 @@ int local_addresses(struct local_address **ret) {
|
||||
|
||||
assert(ret);
|
||||
|
||||
r = sd_rtnl_open(&rtnl, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
if (context)
|
||||
rtnl = sd_rtnl_ref(context);
|
||||
else {
|
||||
r = sd_rtnl_open(&rtnl, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
}
|
||||
|
||||
r = sd_rtnl_message_new_addr(rtnl, &req, RTM_GETADDR, 0, AF_UNSPEC);
|
||||
if (r < 0)
|
||||
@@ -74,6 +78,7 @@ int local_addresses(struct local_address **ret) {
|
||||
struct local_address *a;
|
||||
unsigned char flags;
|
||||
uint16_t type;
|
||||
int ifi;
|
||||
|
||||
r = sd_rtnl_message_get_errno(m);
|
||||
if (r < 0)
|
||||
@@ -86,6 +91,13 @@ int local_addresses(struct local_address **ret) {
|
||||
if (type != RTM_NEWADDR)
|
||||
continue;
|
||||
|
||||
r = sd_rtnl_message_addr_get_ifindex(m, &ifi);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (ifindex != 0 && ifi != ifindex)
|
||||
continue;
|
||||
|
||||
r = sd_rtnl_message_addr_get_flags(m, &flags);
|
||||
if (r < 0)
|
||||
return r;
|
||||
@@ -102,7 +114,7 @@ int local_addresses(struct local_address **ret) {
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (a->scope == RT_SCOPE_HOST || a->scope == RT_SCOPE_NOWHERE)
|
||||
if (ifindex == 0 && (a->scope == RT_SCOPE_HOST || a->scope == RT_SCOPE_NOWHERE))
|
||||
continue;
|
||||
|
||||
r = sd_rtnl_message_addr_get_family(m, &a->family);
|
||||
@@ -133,9 +145,7 @@ int local_addresses(struct local_address **ret) {
|
||||
continue;
|
||||
}
|
||||
|
||||
r = sd_rtnl_message_addr_get_ifindex(m, &a->ifindex);
|
||||
if (r < 0)
|
||||
return r;
|
||||
a->ifindex = ifi;
|
||||
|
||||
n_list++;
|
||||
};
|
||||
|
||||
@@ -26,6 +26,7 @@
|
||||
#include <assert.h>
|
||||
#include <sys/socket.h>
|
||||
|
||||
#include "sd-rtnl.h"
|
||||
#include "in-addr-util.h"
|
||||
|
||||
struct local_address {
|
||||
@@ -34,4 +35,4 @@ struct local_address {
|
||||
union in_addr_union address;
|
||||
};
|
||||
|
||||
int local_addresses(struct local_address **ret);
|
||||
int local_addresses(sd_rtnl *rtnl, int ifindex, struct local_address **ret);
|
||||
|
||||
@@ -465,6 +465,21 @@ int sd_rtnl_message_link_get_flags(sd_rtnl_message *m, unsigned *flags) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int sd_rtnl_message_link_get_type(sd_rtnl_message *m, unsigned *type) {
|
||||
struct ifinfomsg *ifi;
|
||||
|
||||
assert_return(m, -EINVAL);
|
||||
assert_return(m->hdr, -EINVAL);
|
||||
assert_return(rtnl_message_type_is_link(m->hdr->nlmsg_type), -EINVAL);
|
||||
assert_return(type, -EINVAL);
|
||||
|
||||
ifi = NLMSG_DATA(m->hdr);
|
||||
|
||||
*type = ifi->ifi_type;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* If successful the updated message will be correctly aligned, if
|
||||
unsuccessful the old message is untouched. */
|
||||
static int add_rtattr(sd_rtnl_message *m, unsigned short type, const void *data, size_t data_length) {
|
||||
|
||||
@@ -153,3 +153,13 @@ bool rtnl_message_type_is_addr(uint16_t type) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
int rtnl_log_parse_error(int r) {
|
||||
log_error("Failed to parse netlink message: %s", strerror(-r));
|
||||
return r;
|
||||
}
|
||||
|
||||
int rtnl_log_create_error(int r) {
|
||||
log_error("Failed to create netlink message: %s", strerror(-r));
|
||||
return r;
|
||||
}
|
||||
|
||||
@@ -37,6 +37,9 @@ bool rtnl_message_type_is_route(uint16_t type);
|
||||
int rtnl_set_link_name(sd_rtnl *rtnl, int ifindex, const char *name);
|
||||
int rtnl_set_link_properties(sd_rtnl *rtnl, int ifindex, const char *alias, const struct ether_addr *mac, unsigned mtu);
|
||||
|
||||
int rtnl_log_parse_error(int r);
|
||||
int rtnl_log_create_error(int r);
|
||||
|
||||
DEFINE_TRIVIAL_CLEANUP_FUNC(sd_rtnl*, sd_rtnl_unref);
|
||||
DEFINE_TRIVIAL_CLEANUP_FUNC(sd_rtnl_message*, sd_rtnl_message_unref);
|
||||
|
||||
|
||||
@@ -207,7 +207,7 @@ int bus_machine_method_get_addresses(sd_bus *bus, sd_bus_message *message, void
|
||||
if (r < 0)
|
||||
_exit(EXIT_FAILURE);
|
||||
|
||||
n = local_addresses(&addresses);
|
||||
n = local_addresses(NULL, 0, &addresses);
|
||||
if (n < 0)
|
||||
_exit(EXIT_FAILURE);
|
||||
|
||||
|
||||
532
src/network/networkctl.c
Normal file
532
src/network/networkctl.c
Normal file
File diff suppressed because it is too large
Load Diff
@@ -82,7 +82,7 @@ static int parse_argv(int argc, char *argv[]) {
|
||||
assert(argc >= 0);
|
||||
assert(argv);
|
||||
|
||||
while ((c = getopt_long(argc, argv, "h", options, NULL)) >= 0)
|
||||
while ((c = getopt_long(argc, argv, "h", options, NULL)) >= 0) {
|
||||
|
||||
switch (c) {
|
||||
|
||||
@@ -129,6 +129,7 @@ static int parse_argv(int argc, char *argv[]) {
|
||||
default:
|
||||
assert_not_reached("Unhandled option");
|
||||
}
|
||||
}
|
||||
|
||||
if (optind >= argc &&
|
||||
!arg_ready &&
|
||||
|
||||
@@ -92,7 +92,7 @@ enum nss_status _nss_myhostname_gethostbyname4_r(
|
||||
return NSS_STATUS_NOTFOUND;
|
||||
}
|
||||
|
||||
n_addresses = local_addresses(&addresses);
|
||||
n_addresses = local_addresses(NULL, 0, &addresses);
|
||||
if (n_addresses < 0)
|
||||
n_addresses = 0;
|
||||
|
||||
@@ -349,7 +349,7 @@ enum nss_status _nss_myhostname_gethostbyname3_r(
|
||||
return NSS_STATUS_NOTFOUND;
|
||||
}
|
||||
|
||||
n_addresses = local_addresses(&addresses);
|
||||
n_addresses = local_addresses(NULL, 0, &addresses);
|
||||
if (n_addresses < 0)
|
||||
n_addresses = 0;
|
||||
|
||||
@@ -425,7 +425,7 @@ enum nss_status _nss_myhostname_gethostbyaddr2_r(
|
||||
|
||||
}
|
||||
|
||||
n_addresses = local_addresses(&addresses);
|
||||
n_addresses = local_addresses(NULL, 0, &addresses);
|
||||
if (n_addresses < 0)
|
||||
n_addresses = 0;
|
||||
|
||||
|
||||
4
src/shared/.gitignore
vendored
4
src/shared/.gitignore
vendored
@@ -6,3 +6,7 @@
|
||||
/af-from-name.h
|
||||
/af-list.txt
|
||||
/af-to-name.h
|
||||
/arphrd-from-name.gperf
|
||||
/arphrd-from-name.h
|
||||
/arphrd-list.txt
|
||||
/arphrd-to-name.h
|
||||
|
||||
59
src/shared/arphrd-list.c
Normal file
59
src/shared/arphrd-list.c
Normal file
@@ -0,0 +1,59 @@
|
||||
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
|
||||
|
||||
/***
|
||||
This file is part of systemd.
|
||||
|
||||
Copyright 2014 Lennart Poettering
|
||||
|
||||
systemd is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU Lesser General Public License as published by
|
||||
the Free Software Foundation; either version 2.1 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
systemd 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
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public License
|
||||
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
||||
***/
|
||||
|
||||
#include <net/if_arp.h>
|
||||
#include <sys/socket.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "util.h"
|
||||
#include "arphrd-list.h"
|
||||
|
||||
static const struct arphrd_name* lookup_arphrd(register const char *str, register unsigned int len);
|
||||
|
||||
#include "arphrd-to-name.h"
|
||||
#include "arphrd-from-name.h"
|
||||
|
||||
const char *arphrd_to_name(int id) {
|
||||
|
||||
if (id <= 0)
|
||||
return NULL;
|
||||
|
||||
if (id >= (int) ELEMENTSOF(arphrd_names))
|
||||
return NULL;
|
||||
|
||||
return arphrd_names[id];
|
||||
}
|
||||
|
||||
int arphrd_from_name(const char *name) {
|
||||
const struct arphrd_name *sc;
|
||||
|
||||
assert(name);
|
||||
|
||||
sc = lookup_arphrd(name, strlen(name));
|
||||
if (!sc)
|
||||
return 0;
|
||||
|
||||
return sc->id;
|
||||
}
|
||||
|
||||
int arphrd_max(void) {
|
||||
return ELEMENTSOF(arphrd_names);
|
||||
}
|
||||
27
src/shared/arphrd-list.h
Normal file
27
src/shared/arphrd-list.h
Normal file
@@ -0,0 +1,27 @@
|
||||
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
|
||||
|
||||
#pragma once
|
||||
|
||||
/***
|
||||
This file is part of systemd.
|
||||
|
||||
Copyright 2014 Lennart Poettering
|
||||
|
||||
systemd is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU Lesser General Public License as published by
|
||||
the Free Software Foundation; either version 2.1 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
systemd 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
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public License
|
||||
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
||||
***/
|
||||
|
||||
const char *arphrd_to_name(int id);
|
||||
int arphrd_from_name(const char *name);
|
||||
|
||||
int arphrd_max(void);
|
||||
@@ -95,6 +95,7 @@ int sd_rtnl_message_link_set_flags(sd_rtnl_message *m, unsigned flags, unsigned
|
||||
int sd_rtnl_message_link_set_type(sd_rtnl_message *m, unsigned type);
|
||||
int sd_rtnl_message_link_get_ifindex(sd_rtnl_message *m, int *ifindex);
|
||||
int sd_rtnl_message_link_get_flags(sd_rtnl_message *m, unsigned *flags);
|
||||
int sd_rtnl_message_link_get_type(sd_rtnl_message *m, unsigned *type);
|
||||
|
||||
int sd_rtnl_message_route_set_dst_prefixlen(sd_rtnl_message *m, unsigned char prefixlen);
|
||||
int sd_rtnl_message_route_set_scope(sd_rtnl_message *m, unsigned char scope);
|
||||
|
||||
Reference in New Issue
Block a user