]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/network/wait-online/wait-online.c
wait-online: wait for address family
[thirdparty/systemd.git] / src / network / wait-online / wait-online.c
CommitLineData
db9ecf05 1/* SPDX-License-Identifier: LGPL-2.1-or-later */
020d5900 2
cef8b073 3#include <getopt.h>
ca78ad1d
ZJS
4#include <sys/stat.h>
5#include <sys/types.h>
3a67e927 6
020d5900 7#include "sd-daemon.h"
3f6fd1ba 8
c03a80c4
YW
9#include "daemon-util.h"
10#include "main-func.h"
c5fcf6e4 11#include "manager.h"
294bf0c3 12#include "pretty-print.h"
3f6fd1ba 13#include "signal-util.h"
f023184e 14#include "socket-util.h"
3f6fd1ba 15#include "strv.h"
cef8b073
TG
16
17static bool arg_quiet = false;
e56cdb7a 18static usec_t arg_timeout = 120 * USEC_PER_SEC;
f023184e 19static Hashmap *arg_interfaces = NULL;
79b1f37d 20static char **arg_ignore = NULL;
75cd4a5d 21static LinkOperationalStateRange arg_required_operstate = { _LINK_OPERSTATE_INVALID, _LINK_OPERSTATE_INVALID };
6dc4531d 22static AddressFamily arg_required_family = ADDRESS_FAMILY_NO;
2f9859ba 23static bool arg_any = false;
cef8b073 24
75cd4a5d 25STATIC_DESTRUCTOR_REGISTER(arg_interfaces, hashmap_free_free_freep);
c03a80c4
YW
26STATIC_DESTRUCTOR_REGISTER(arg_ignore, strv_freep);
27
37ec0fdd
LP
28static int help(void) {
29 _cleanup_free_ char *link = NULL;
30 int r;
31
32 r = terminal_urlify_man("systemd-networkd-wait-online.service", "8", &link);
33 if (r < 0)
34 return log_oom();
35
cef8b073
TG
36 printf("%s [OPTIONS...]\n\n"
37 "Block until network is configured.\n\n"
38 " -h --help Show this help\n"
39 " --version Print version string\n"
40 " -q --quiet Do not show status information\n"
75cd4a5d 41 " -i --interface=INTERFACE[:MIN_OPERSTATE[:MAX_OPERSTATE]]\n"
f023184e 42 " Block until at least these interfaces have appeared\n"
79b1f37d 43 " --ignore=INTERFACE Don't take these interfaces into account\n"
75cd4a5d 44 " -o --operational-state=MIN_OPERSTATE[:MAX_OPERSTATE]\n"
f023184e 45 " Required operational state\n"
6dc4531d
L
46 " -4 --ipv4 Requires at least one IPv4 address\n"
47 " -6 --ipv6 Requires at least one IPv6 address\n"
2f9859ba 48 " --any Wait until at least one of the interfaces is online\n"
e56cdb7a 49 " --timeout=SECS Maximum time to wait for network connectivity\n"
bc556335
DDM
50 "\nSee the %s for details.\n",
51 program_invocation_short_name,
52 link);
37ec0fdd
LP
53
54 return 0;
cef8b073
TG
55}
56
75cd4a5d 57static int parse_interface_with_operstate_range(const char *str) {
f023184e 58 _cleanup_free_ char *ifname = NULL;
75cd4a5d 59 _cleanup_free_ LinkOperationalStateRange *range;
f023184e
YW
60 const char *p;
61 int r;
62
63 assert(str);
64
75cd4a5d
DDM
65 range = new(LinkOperationalStateRange, 1);
66 if (!range)
67 return log_oom();
68
f023184e
YW
69 p = strchr(str, ':');
70 if (p) {
75cd4a5d
DDM
71 r = parse_operational_state_range(p + 1, range);
72 if (r < 0)
73 log_error_errno(r, "Invalid operational state range '%s'", p + 1);
f023184e
YW
74
75 ifname = strndup(optarg, p - optarg);
76 } else {
75cd4a5d
DDM
77 range->min = _LINK_OPERSTATE_INVALID;
78 range->max = _LINK_OPERSTATE_INVALID;
f023184e
YW
79 ifname = strdup(str);
80 }
81 if (!ifname)
82 return log_oom();
83
84 if (!ifname_valid(ifname))
85 return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
86 "Invalid interface name '%s'", ifname);
87
4f38ba38
SS
88 r = hashmap_ensure_put(&arg_interfaces, &string_hash_ops, ifname, TAKE_PTR(range));
89 if (r == -ENOMEM)
f023184e 90 return log_oom();
f023184e
YW
91 if (r < 0)
92 return log_error_errno(r, "Failed to store interface name: %m");
93 if (r == 0)
94 return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
95 "Interface name %s is already specified", ifname);
96
97 TAKE_PTR(ifname);
98 return 0;
99}
100
cef8b073
TG
101static int parse_argv(int argc, char *argv[]) {
102
103 enum {
104 ARG_VERSION = 0x100,
79b1f37d 105 ARG_IGNORE,
2f9859ba 106 ARG_ANY,
e56cdb7a 107 ARG_TIMEOUT,
cef8b073
TG
108 };
109
110 static const struct option options[] = {
f023184e
YW
111 { "help", no_argument, NULL, 'h' },
112 { "version", no_argument, NULL, ARG_VERSION },
113 { "quiet", no_argument, NULL, 'q' },
114 { "interface", required_argument, NULL, 'i' },
115 { "ignore", required_argument, NULL, ARG_IGNORE },
116 { "operational-state", required_argument, NULL, 'o' },
6dc4531d
L
117 { "ipv4", no_argument, NULL, '4' },
118 { "ipv6", no_argument, NULL, '6' },
2f9859ba 119 { "any", no_argument, NULL, ARG_ANY },
f023184e 120 { "timeout", required_argument, NULL, ARG_TIMEOUT },
cef8b073
TG
121 {}
122 };
123
e56cdb7a 124 int c, r;
cef8b073
TG
125
126 assert(argc >= 0);
127 assert(argv);
128
6dc4531d 129 while ((c = getopt_long(argc, argv, "hi:qo:46", options, NULL)) >= 0)
cef8b073
TG
130
131 switch (c) {
132
133 case 'h':
601185b4
ZJS
134 help();
135 return 0;
cef8b073
TG
136
137 case 'q':
138 arg_quiet = true;
139 break;
140
141 case ARG_VERSION:
3f6fd1ba 142 return version();
cef8b073
TG
143
144 case 'i':
75cd4a5d 145 r = parse_interface_with_operstate_range(optarg);
f023184e
YW
146 if (r < 0)
147 return r;
cef8b073
TG
148 break;
149
79b1f37d
TG
150 case ARG_IGNORE:
151 if (strv_extend(&arg_ignore, optarg) < 0)
152 return log_oom();
153
154 break;
155
f023184e 156 case 'o': {
75cd4a5d
DDM
157 LinkOperationalStateRange range;
158
159 r = parse_operational_state_range(optarg, &range);
160 if (r < 0)
161 return log_error_errno(r, "Invalid operational state range '%s'", optarg);
f023184e 162
75cd4a5d 163 arg_required_operstate = range;
f023184e 164
f023184e
YW
165 break;
166 }
6dc4531d
L
167
168 case '4':
169 arg_required_family |= ADDRESS_FAMILY_IPV4;
170 break;
171
172 case '6':
173 arg_required_family |= ADDRESS_FAMILY_IPV6;
174 break;
175
2f9859ba
YW
176 case ARG_ANY:
177 arg_any = true;
178 break;
179
e56cdb7a
TG
180 case ARG_TIMEOUT:
181 r = parse_sec(optarg, &arg_timeout);
182 if (r < 0)
183 return r;
e56cdb7a
TG
184 break;
185
cef8b073
TG
186 case '?':
187 return -EINVAL;
188
189 default:
190 assert_not_reached("Unhandled option");
191 }
cef8b073
TG
192
193 return 1;
194}
020d5900 195
c03a80c4 196static int run(int argc, char *argv[]) {
7de12ae7 197 _cleanup_(manager_freep) Manager *m = NULL;
272ac70a 198 _cleanup_(notify_on_cleanup) const char *notify_message = NULL;
7de12ae7 199 int r;
cef8b073 200
d2acb93d 201 log_setup();
020d5900 202
7de12ae7
TG
203 umask(0022);
204
cef8b073
TG
205 r = parse_argv(argc, argv);
206 if (r <= 0)
207 return r;
020d5900 208
cef8b073 209 if (arg_quiet)
f023184e 210 log_set_max_level(LOG_ERR);
020d5900 211
72c0a2c2 212 assert_se(sigprocmask_many(SIG_BLOCK, NULL, SIGTERM, SIGINT, -1) >= 0);
3a67e927 213
6dc4531d 214 r = manager_new(&m, arg_interfaces, arg_ignore, arg_required_operstate, arg_required_family, arg_any, arg_timeout);
c03a80c4
YW
215 if (r < 0)
216 return log_error_errno(r, "Could not create manager: %m");
020d5900 217
2f9859ba 218 if (manager_configured(m))
c03a80c4 219 goto success;
020d5900 220
c03a80c4
YW
221 notify_message = notify_start("READY=1\n"
222 "STATUS=Waiting for network connections...",
223 "STATUS=Failed to wait for network connectivity...");
020d5900 224
3a67e927 225 r = sd_event_loop(m->event);
c03a80c4
YW
226 if (r < 0)
227 return log_error_errno(r, "Event loop failed: %m");
e56cdb7a 228
c03a80c4
YW
229success:
230 notify_message = "STATUS=All interfaces configured...";
e56cdb7a 231
c03a80c4 232 return 0;
020d5900 233}
c03a80c4
YW
234
235DEFINE_MAIN_FUNCTION(run);