103
45
#include <avahi-common/malloc.h>
104
46
#include <avahi-common/error.h>
107
#include <gnutls/gnutls.h> /* All GnuTLS types, constants and
110
init_gnutls_session(),
112
#include <gnutls/openpgp.h>
113
/* gnutls_certificate_set_openpgp_key_file(),
114
GNUTLS_OPENPGP_FMT_BASE64 */
117
#include <gpgme.h> /* All GPGME types, constants and
120
GPGME_PROTOCOL_OpenPGP,
49
#include <sys/types.h> /* socket(), inet_pton() */
50
#include <sys/socket.h> /* socket(), struct sockaddr_in6,
51
struct in6_addr, inet_pton() */
52
#include <gnutls/gnutls.h> /* All GnuTLS stuff */
53
#include <gnutls/openpgp.h> /* GnuTLS with openpgp stuff */
55
#include <unistd.h> /* close() */
56
#include <netinet/in.h>
57
#include <stdbool.h> /* true */
58
#include <string.h> /* memset */
59
#include <arpa/inet.h> /* inet_pton() */
60
#include <iso646.h> /* not */
63
#include <errno.h> /* perror() */
123
69
#define BUFFER_SIZE 256
125
#define PATHDIR "/conf/conf.d/mandos"
126
#define SECKEY "seckey.txt"
127
#define PUBKEY "pubkey.txt"
128
#define HOOKDIR "/lib/mandos/network-hooks.d"
72
static const char *certdir = "/conf/conf.d/mandos";
73
static const char *certfile = "openpgp-client.txt";
74
static const char *certkey = "openpgp-client-key.txt";
130
76
bool debug = false;
131
static const char mandos_protocol_version[] = "1";
132
const char *argp_program_version = "mandos-client " VERSION;
133
const char *argp_program_bug_address = "<mandos@recompile.se>";
134
static const char sys_class_net[] = "/sys/class/net";
135
char *connect_to = NULL;
136
const char *hookdir = HOOKDIR;
138
/* Doubly linked list that need to be circularly linked when used */
139
typedef struct server{
142
AvahiIfIndex if_index;
144
struct timespec last_seen;
149
/* Used for passing in values through the Avahi callback functions */
151
AvahiSimplePoll *simple_poll;
79
gnutls_session_t session;
153
80
gnutls_certificate_credentials_t cred;
154
unsigned int dh_bits;
155
81
gnutls_dh_params_t dh_params;
156
const char *priority;
85
static ssize_t pgp_packet_decrypt (char *packet, size_t packet_size,
88
gpgme_data_t dh_crypto, dh_plain;
158
server *current_server;
161
/* global context so signal handler can reach it*/
162
mandos_context mc = { .simple_poll = NULL, .server = NULL,
163
.dh_bits = 1024, .priority = "SECURE256"
164
":!CTYPE-X.509:+CTYPE-OPENPGP",
165
.current_server = NULL };
167
sig_atomic_t quit_now = 0;
168
int signal_received = 0;
170
/* Function to use when printing errors */
171
void perror_plus(const char *print_text){
172
fprintf(stderr, "Mandos plugin %s: ",
173
program_invocation_short_name);
177
int fprintf_plus(FILE *stream, const char *format, ...){
179
va_start (ap, format);
181
TEMP_FAILURE_RETRY(fprintf(stream, "Mandos plugin %s: ",
182
program_invocation_short_name));
183
return TEMP_FAILURE_RETRY(vfprintf(stream, format, ap));
187
* Make additional room in "buffer" for at least BUFFER_SIZE more
188
* bytes. "buffer_capacity" is how much is currently allocated,
189
* "buffer_length" is how much is already used.
191
size_t incbuffer(char **buffer, size_t buffer_length,
192
size_t buffer_capacity){
193
if(buffer_length + BUFFER_SIZE > buffer_capacity){
194
*buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
198
buffer_capacity += BUFFER_SIZE;
200
return buffer_capacity;
203
/* Add server to set of servers to retry periodically */
204
int add_server(const char *ip, uint16_t port, AvahiIfIndex if_index,
207
server *new_server = malloc(sizeof(server));
208
if(new_server == NULL){
209
perror_plus("malloc");
212
*new_server = (server){ .ip = strdup(ip),
214
.if_index = if_index,
216
if(new_server->ip == NULL){
217
perror_plus("strdup");
220
/* Special case of first server */
221
if (mc.current_server == NULL){
222
new_server->next = new_server;
223
new_server->prev = new_server;
224
mc.current_server = new_server;
225
/* Place the new server last in the list */
227
new_server->next = mc.current_server;
228
new_server->prev = mc.current_server->prev;
229
new_server->prev->next = new_server;
230
mc.current_server->prev = new_server;
232
ret = clock_gettime(CLOCK_MONOTONIC, &mc.current_server->last_seen);
234
perror_plus("clock_gettime");
243
static bool init_gpgme(const char *seckey, const char *pubkey,
244
const char *tempdir){
92
ssize_t new_packet_capacity = 0;
93
ssize_t new_packet_length = 0;
246
94
gpgme_engine_info_t engine_info;
250
* Helper function to insert pub and seckey to the engine keyring.
252
bool import_key(const char *filename){
255
gpgme_data_t pgp_data;
257
fd = (int)TEMP_FAILURE_RETRY(open(filename, O_RDONLY));
263
rc = gpgme_data_new_from_fd(&pgp_data, fd);
264
if(rc != GPG_ERR_NO_ERROR){
265
fprintf_plus(stderr, "bad gpgme_data_new_from_fd: %s: %s\n",
266
gpgme_strsource(rc), gpgme_strerror(rc));
270
rc = gpgme_op_import(mc.ctx, pgp_data);
271
if(rc != GPG_ERR_NO_ERROR){
272
fprintf_plus(stderr, "bad gpgme_op_import: %s: %s\n",
273
gpgme_strsource(rc), gpgme_strerror(rc));
277
ret = (int)TEMP_FAILURE_RETRY(close(fd));
279
perror_plus("close");
281
gpgme_data_release(pgp_data);
286
fprintf_plus(stderr, "Initializing GPGME\n");
97
fprintf(stderr, "Trying to decrypt OpenPGP packet\n");
290
101
gpgme_check_version(NULL);
291
102
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
292
if(rc != GPG_ERR_NO_ERROR){
293
fprintf_plus(stderr, "bad gpgme_engine_check_version: %s: %s\n",
294
gpgme_strsource(rc), gpgme_strerror(rc));
103
if (rc != GPG_ERR_NO_ERROR){
104
fprintf(stderr, "bad gpgme_engine_check_version: %s: %s\n",
105
gpgme_strsource(rc), gpgme_strerror(rc));
298
/* Set GPGME home directory for the OpenPGP engine only */
299
rc = gpgme_get_engine_info(&engine_info);
300
if(rc != GPG_ERR_NO_ERROR){
301
fprintf_plus(stderr, "bad gpgme_get_engine_info: %s: %s\n",
302
gpgme_strsource(rc), gpgme_strerror(rc));
109
/* Set GPGME home directory */
110
rc = gpgme_get_engine_info (&engine_info);
111
if (rc != GPG_ERR_NO_ERROR){
112
fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
113
gpgme_strsource(rc), gpgme_strerror(rc));
305
116
while(engine_info != NULL){
306
117
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
307
118
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
308
engine_info->file_name, tempdir);
119
engine_info->file_name, homedir);
311
122
engine_info = engine_info->next;
313
124
if(engine_info == NULL){
314
fprintf_plus(stderr, "Could not set GPGME home dir to %s\n",
319
/* Create new GPGME "context" */
320
rc = gpgme_new(&(mc.ctx));
321
if(rc != GPG_ERR_NO_ERROR){
322
fprintf_plus(stderr, "Mandos plugin mandos-client: "
323
"bad gpgme_new: %s: %s\n", gpgme_strsource(rc),
328
if(not import_key(pubkey) or not import_key(seckey)){
336
* Decrypt OpenPGP data.
337
* Returns -1 on error
339
static ssize_t pgp_packet_decrypt(const char *cryptotext,
342
gpgme_data_t dh_crypto, dh_plain;
345
size_t plaintext_capacity = 0;
346
ssize_t plaintext_length = 0;
349
fprintf_plus(stderr, "Trying to decrypt OpenPGP data\n");
352
/* Create new GPGME data buffer from memory cryptotext */
353
rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
355
if(rc != GPG_ERR_NO_ERROR){
356
fprintf_plus(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
357
gpgme_strsource(rc), gpgme_strerror(rc));
125
fprintf(stderr, "Could not set home dir to %s\n", homedir);
129
/* Create new GPGME data buffer from packet buffer */
130
rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
131
if (rc != GPG_ERR_NO_ERROR){
132
fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
133
gpgme_strsource(rc), gpgme_strerror(rc));
361
137
/* Create new empty GPGME data buffer for the plaintext */
362
138
rc = gpgme_data_new(&dh_plain);
363
if(rc != GPG_ERR_NO_ERROR){
364
fprintf_plus(stderr, "Mandos plugin mandos-client: "
365
"bad gpgme_data_new: %s: %s\n",
366
gpgme_strsource(rc), gpgme_strerror(rc));
367
gpgme_data_release(dh_crypto);
371
/* Decrypt data from the cryptotext data buffer to the plaintext
373
rc = gpgme_op_decrypt(mc.ctx, dh_crypto, dh_plain);
374
if(rc != GPG_ERR_NO_ERROR){
375
fprintf_plus(stderr, "bad gpgme_op_decrypt: %s: %s\n",
376
gpgme_strsource(rc), gpgme_strerror(rc));
377
plaintext_length = -1;
379
gpgme_decrypt_result_t result;
380
result = gpgme_op_decrypt_result(mc.ctx);
382
fprintf_plus(stderr, "gpgme_op_decrypt_result failed\n");
384
fprintf_plus(stderr, "Unsupported algorithm: %s\n",
385
result->unsupported_algorithm);
386
fprintf_plus(stderr, "Wrong key usage: %u\n",
387
result->wrong_key_usage);
388
if(result->file_name != NULL){
389
fprintf_plus(stderr, "File name: %s\n", result->file_name);
391
gpgme_recipient_t recipient;
392
recipient = result->recipients;
139
if (rc != GPG_ERR_NO_ERROR){
140
fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
141
gpgme_strsource(rc), gpgme_strerror(rc));
145
/* Create new GPGME "context" */
146
rc = gpgme_new(&ctx);
147
if (rc != GPG_ERR_NO_ERROR){
148
fprintf(stderr, "bad gpgme_new: %s: %s\n",
149
gpgme_strsource(rc), gpgme_strerror(rc));
153
/* Decrypt data from the FILE pointer to the plaintext data
155
rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
156
if (rc != GPG_ERR_NO_ERROR){
157
fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
158
gpgme_strsource(rc), gpgme_strerror(rc));
163
fprintf(stderr, "Decryption of OpenPGP packet succeeded\n");
167
gpgme_decrypt_result_t result;
168
result = gpgme_op_decrypt_result(ctx);
170
fprintf(stderr, "gpgme_op_decrypt_result failed\n");
172
fprintf(stderr, "Unsupported algorithm: %s\n",
173
result->unsupported_algorithm);
174
fprintf(stderr, "Wrong key usage: %d\n",
175
result->wrong_key_usage);
176
if(result->file_name != NULL){
177
fprintf(stderr, "File name: %s\n", result->file_name);
179
gpgme_recipient_t recipient;
180
recipient = result->recipients;
393
182
while(recipient != NULL){
394
fprintf_plus(stderr, "Public key algorithm: %s\n",
395
gpgme_pubkey_algo_name
396
(recipient->pubkey_algo));
397
fprintf_plus(stderr, "Key ID: %s\n", recipient->keyid);
398
fprintf_plus(stderr, "Secret key available: %s\n",
399
recipient->status == GPG_ERR_NO_SECKEY
183
fprintf(stderr, "Public key algorithm: %s\n",
184
gpgme_pubkey_algo_name(recipient->pubkey_algo));
185
fprintf(stderr, "Key ID: %s\n", recipient->keyid);
186
fprintf(stderr, "Secret key available: %s\n",
187
recipient->status == GPG_ERR_NO_SECKEY
401
189
recipient = recipient->next;
409
fprintf_plus(stderr, "Decryption of OpenPGP data succeeded\n");
195
/* Delete the GPGME FILE pointer cryptotext data buffer */
196
gpgme_data_release(dh_crypto);
412
198
/* Seek back to the beginning of the GPGME plaintext data buffer */
413
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
414
perror_plus("gpgme_data_seek");
415
plaintext_length = -1;
199
if (gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET) == -1){
200
perror("pgpme_data_seek");
421
plaintext_capacity = incbuffer(plaintext,
422
(size_t)plaintext_length,
424
if(plaintext_capacity == 0){
425
perror_plus("incbuffer");
426
plaintext_length = -1;
205
if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
206
*new_packet = realloc(*new_packet,
207
(unsigned int)new_packet_capacity
209
if (*new_packet == NULL){
213
new_packet_capacity += BUFFER_SIZE;
430
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
216
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length,
432
218
/* Print the data, if any */
438
perror_plus("gpgme_data_read");
439
plaintext_length = -1;
442
plaintext_length += ret;
446
fprintf_plus(stderr, "Decrypted password is: ");
447
for(ssize_t i = 0; i < plaintext_length; i++){
448
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
450
fprintf(stderr, "\n");
455
/* Delete the GPGME cryptotext data buffer */
456
gpgme_data_release(dh_crypto);
223
perror("gpgme_data_read");
226
new_packet_length += ret;
229
/* FIXME: check characters before printing to screen so to not print
230
terminal control characters */
232
/* fprintf(stderr, "decrypted password is: "); */
233
/* fwrite(*new_packet, 1, new_packet_length, stderr); */
234
/* fprintf(stderr, "\n"); */
458
237
/* Delete the GPGME plaintext data buffer */
459
238
gpgme_data_release(dh_plain);
460
return plaintext_length;
239
return new_packet_length;
463
static const char * safer_gnutls_strerror(int value){
464
const char *ret = gnutls_strerror(value); /* Spurious warning from
465
-Wunreachable-code */
242
static const char * safer_gnutls_strerror (int value) {
243
const char *ret = gnutls_strerror (value);
467
245
ret = "(unknown)";
471
/* GnuTLS log function callback */
472
249
static void debuggnutls(__attribute__((unused)) int level,
473
250
const char* string){
474
fprintf_plus(stderr, "GnuTLS: %s", string);
251
fprintf(stderr, "%s", string);
477
static int init_gnutls_global(const char *pubkeyfilename,
478
const char *seckeyfilename){
254
static int initgnutls(encrypted_session *es){
482
fprintf_plus(stderr, "Initializing GnuTLS\n");
259
fprintf(stderr, "Initializing GnuTLS\n");
485
ret = gnutls_global_init();
486
if(ret != GNUTLS_E_SUCCESS){
487
fprintf_plus(stderr, "GnuTLS global_init: %s\n",
488
safer_gnutls_strerror(ret));
262
if ((ret = gnutls_global_init ())
263
!= GNUTLS_E_SUCCESS) {
264
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
493
/* "Use a log level over 10 to enable all debugging options."
496
269
gnutls_global_set_log_level(11);
497
270
gnutls_global_set_log_function(debuggnutls);
500
/* OpenPGP credentials */
501
ret = gnutls_certificate_allocate_credentials(&mc.cred);
502
if(ret != GNUTLS_E_SUCCESS){
503
fprintf_plus(stderr, "GnuTLS memory error: %s\n",
504
safer_gnutls_strerror(ret));
505
gnutls_global_deinit();
273
/* openpgp credentials */
274
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
275
!= GNUTLS_E_SUCCESS) {
276
fprintf (stderr, "memory error: %s\n",
277
safer_gnutls_strerror(ret));
510
fprintf_plus(stderr, "Attempting to use OpenPGP public key %s and"
511
" secret key %s as GnuTLS credentials\n",
282
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
283
" and keyfile %s as GnuTLS credentials\n", certfile,
516
287
ret = gnutls_certificate_set_openpgp_key_file
517
(mc.cred, pubkeyfilename, seckeyfilename,
518
GNUTLS_OPENPGP_FMT_BASE64);
519
if(ret != GNUTLS_E_SUCCESS){
521
"Error[%d] while reading the OpenPGP key pair ('%s',"
522
" '%s')\n", ret, pubkeyfilename, seckeyfilename);
523
fprintf_plus(stderr, "The GnuTLS error is: %s\n",
524
safer_gnutls_strerror(ret));
528
/* GnuTLS server initialization */
529
ret = gnutls_dh_params_init(&mc.dh_params);
530
if(ret != GNUTLS_E_SUCCESS){
531
fprintf_plus(stderr, "Error in GnuTLS DH parameter"
532
" initialization: %s\n",
533
safer_gnutls_strerror(ret));
536
ret = gnutls_dh_params_generate2(mc.dh_params, mc.dh_bits);
537
if(ret != GNUTLS_E_SUCCESS){
538
fprintf_plus(stderr, "Error in GnuTLS prime generation: %s\n",
539
safer_gnutls_strerror(ret));
543
gnutls_certificate_set_dh_params(mc.cred, mc.dh_params);
549
gnutls_certificate_free_credentials(mc.cred);
550
gnutls_global_deinit();
551
gnutls_dh_params_deinit(mc.dh_params);
555
static int init_gnutls_session(gnutls_session_t *session){
557
/* GnuTLS session creation */
559
ret = gnutls_init(session, GNUTLS_SERVER);
563
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
564
if(ret != GNUTLS_E_SUCCESS){
566
"Error in GnuTLS session initialization: %s\n",
567
safer_gnutls_strerror(ret));
573
ret = gnutls_priority_set_direct(*session, mc.priority, &err);
575
gnutls_deinit(*session);
578
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
579
if(ret != GNUTLS_E_SUCCESS){
580
fprintf_plus(stderr, "Syntax error at: %s\n", err);
581
fprintf_plus(stderr, "GnuTLS error: %s\n",
582
safer_gnutls_strerror(ret));
583
gnutls_deinit(*session);
589
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
592
gnutls_deinit(*session);
595
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
596
if(ret != GNUTLS_E_SUCCESS){
597
fprintf_plus(stderr, "Error setting GnuTLS credentials: %s\n",
598
safer_gnutls_strerror(ret));
599
gnutls_deinit(*session);
288
(es->cred, certfile, certkey, GNUTLS_OPENPGP_FMT_BASE64);
289
if (ret != GNUTLS_E_SUCCESS) {
291
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s',"
293
ret, certfile, certkey);
294
fprintf(stdout, "The Error is: %s\n",
295
safer_gnutls_strerror(ret));
299
//GnuTLS server initialization
300
if ((ret = gnutls_dh_params_init (&es->dh_params))
301
!= GNUTLS_E_SUCCESS) {
302
fprintf (stderr, "Error in dh parameter initialization: %s\n",
303
safer_gnutls_strerror(ret));
307
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
308
!= GNUTLS_E_SUCCESS) {
309
fprintf (stderr, "Error in prime generation: %s\n",
310
safer_gnutls_strerror(ret));
314
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
316
// GnuTLS session creation
317
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
318
!= GNUTLS_E_SUCCESS){
319
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
320
safer_gnutls_strerror(ret));
323
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
324
!= GNUTLS_E_SUCCESS) {
325
fprintf(stderr, "Syntax error at: %s\n", err);
326
fprintf(stderr, "GnuTLS error: %s\n",
327
safer_gnutls_strerror(ret));
331
if ((ret = gnutls_credentials_set
332
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
333
!= GNUTLS_E_SUCCESS) {
334
fprintf(stderr, "Error setting a credentials set: %s\n",
335
safer_gnutls_strerror(ret));
603
339
/* ignore client certificate if any. */
604
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
340
gnutls_certificate_server_set_request (es->session,
606
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
343
gnutls_dh_set_prime_bits (es->session, DH_BITS);
611
/* Avahi log function callback */
612
348
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
613
349
__attribute__((unused)) const char *txt){}
615
/* Called when a Mandos server is found */
616
351
static int start_mandos_communication(const char *ip, uint16_t port,
617
AvahiIfIndex if_index,
619
int ret, tcp_sd = -1;
622
struct sockaddr_in in;
623
struct sockaddr_in6 in6;
352
AvahiIfIndex if_index){
354
struct sockaddr_in6 to;
355
encrypted_session es;
625
356
char *buffer = NULL;
626
char *decrypted_buffer = NULL;
357
char *decrypted_buffer;
627
358
size_t buffer_length = 0;
628
359
size_t buffer_capacity = 0;
631
gnutls_session_t session;
632
int pf; /* Protocol family */
649
fprintf_plus(stderr, "Bad address family: %d\n", af);
654
ret = init_gnutls_session(&session);
660
fprintf_plus(stderr, "Setting up a TCP connection to %s, port %"
661
PRIu16 "\n", ip, port);
664
tcp_sd = socket(pf, SOCK_STREAM, 0);
667
perror_plus("socket");
677
memset(&to, 0, sizeof(to));
679
to.in6.sin6_family = (sa_family_t)af;
680
ret = inet_pton(af, ip, &to.in6.sin6_addr);
682
to.in.sin_family = (sa_family_t)af;
683
ret = inet_pton(af, ip, &to.in.sin_addr);
687
perror_plus("inet_pton");
360
ssize_t decrypted_buffer_size;
363
char interface[IF_NAMESIZE];
366
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
370
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
376
if(if_indextoname((unsigned int)if_index, interface) == NULL){
378
perror("if_indextoname");
384
fprintf(stderr, "Binding to interface %s\n", interface);
387
memset(&to,0,sizeof(to)); /* Spurious warning */
388
to.sin6_family = AF_INET6;
389
ret = inet_pton(AF_INET6, ip, &to.sin6_addr);
693
fprintf_plus(stderr, "Bad address: %s\n", ip);
698
to.in6.sin6_port = htons(port); /* Spurious warnings from
700
-Wunreachable-code */
702
if(IN6_IS_ADDR_LINKLOCAL /* Spurious warnings from */
703
(&to.in6.sin6_addr)){ /* -Wstrict-aliasing=2 or lower and
705
if(if_index == AVAHI_IF_UNSPEC){
706
fprintf_plus(stderr, "An IPv6 link-local address is"
707
" incomplete without a network interface\n");
711
/* Set the network interface number as scope */
712
to.in6.sin6_scope_id = (uint32_t)if_index;
715
to.in.sin_port = htons(port); /* Spurious warnings from
717
-Wunreachable-code */
726
if(af == AF_INET6 and if_index != AVAHI_IF_UNSPEC){
727
char interface[IF_NAMESIZE];
728
if(if_indextoname((unsigned int)if_index, interface) == NULL){
729
perror_plus("if_indextoname");
731
fprintf_plus(stderr, "Connection to: %s%%%s, port %" PRIu16
732
"\n", ip, interface, port);
735
fprintf_plus(stderr, "Connection to: %s, port %" PRIu16 "\n",
738
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
739
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
742
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
745
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
749
perror_plus("inet_ntop");
751
if(strcmp(addrstr, ip) != 0){
752
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
763
ret = connect(tcp_sd, &to.in6, sizeof(to));
765
ret = connect(tcp_sd, &to.in, sizeof(to)); /* IPv4 */
768
if ((errno != ECONNREFUSED and errno != ENETUNREACH) or debug){
770
perror_plus("connect");
781
const char *out = mandos_protocol_version;
784
size_t out_size = strlen(out);
785
ret = (int)TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
786
out_size - written));
789
perror_plus("write");
793
written += (size_t)ret;
794
if(written < out_size){
797
if(out == mandos_protocol_version){
812
fprintf_plus(stderr, "Establishing TLS session with %s\n", ip);
820
/* Spurious warning from -Wint-to-pointer-cast */
821
gnutls_transport_set_ptr(session, (gnutls_transport_ptr_t) tcp_sd);
829
ret = gnutls_handshake(session);
834
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
836
if(ret != GNUTLS_E_SUCCESS){
395
fprintf(stderr, "Bad address: %s\n", ip);
398
to.sin6_port = htons(port); /* Spurious warning */
400
to.sin6_scope_id = (uint32_t)if_index;
403
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
404
/* char addrstr[INET6_ADDRSTRLEN]; */
405
/* if(inet_ntop(to.sin6_family, &(to.sin6_addr), addrstr, */
406
/* sizeof(addrstr)) == NULL){ */
407
/* perror("inet_ntop"); */
409
/* fprintf(stderr, "Really connecting to: %s, port %d\n", */
410
/* addrstr, ntohs(to.sin6_port)); */
414
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
420
ret = initgnutls (&es);
426
gnutls_transport_set_ptr (es.session,
427
(gnutls_transport_ptr_t) tcp_sd);
430
fprintf(stderr, "Establishing TLS session with %s\n", ip);
433
ret = gnutls_handshake (es.session);
435
if (ret != GNUTLS_E_SUCCESS){
838
fprintf_plus(stderr, "*** GnuTLS Handshake failed ***\n");
437
fprintf(stderr, "\n*** Handshake failed ***\n");
845
/* Read OpenPGP packet that contains the wanted password */
444
//Retrieve OpenPGP packet that contains the wanted password
848
fprintf_plus(stderr, "Retrieving OpenPGP encrypted password from"
447
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
859
buffer_capacity = incbuffer(&buffer, buffer_length,
861
if(buffer_capacity == 0){
863
perror_plus("incbuffer");
873
sret = gnutls_record_recv(session, buffer+buffer_length,
452
if (buffer_length + BUFFER_SIZE > buffer_capacity){
453
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
458
buffer_capacity += BUFFER_SIZE;
461
ret = gnutls_record_recv
462
(es.session, buffer+buffer_length, BUFFER_SIZE);
880
468
case GNUTLS_E_INTERRUPTED:
881
469
case GNUTLS_E_AGAIN:
883
471
case GNUTLS_E_REHANDSHAKE:
885
ret = gnutls_handshake(session);
891
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
893
fprintf_plus(stderr, "*** GnuTLS Re-handshake failed "
472
ret = gnutls_handshake (es.session);
474
fprintf(stderr, "\n*** Handshake failed ***\n");
901
fprintf_plus(stderr, "Unknown error while reading data from"
902
" encrypted session with Mandos server\n");
903
gnutls_bye(session, GNUTLS_SHUT_RDWR);
481
fprintf(stderr, "Unknown error while reading data from"
482
" encrypted session with mandos server\n");
484
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
908
buffer_length += (size_t) sret;
913
fprintf_plus(stderr, "Closing TLS session\n");
922
ret = gnutls_bye(session, GNUTLS_SHUT_RDWR);
927
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
929
if(buffer_length > 0){
930
ssize_t decrypted_buffer_size;
931
decrypted_buffer_size = pgp_packet_decrypt(buffer, buffer_length,
933
if(decrypted_buffer_size >= 0){
488
buffer_length += (size_t) ret;
492
if (buffer_length > 0){
493
decrypted_buffer_size = pgp_packet_decrypt(buffer,
497
if (decrypted_buffer_size >= 0){
936
498
while(written < (size_t) decrypted_buffer_size){
942
ret = (int)fwrite(decrypted_buffer + written, 1,
943
(size_t)decrypted_buffer_size - written,
499
ret = (int)fwrite (decrypted_buffer + written, 1,
500
(size_t)decrypted_buffer_size - written,
945
502
if(ret == 0 and ferror(stdout)){
948
fprintf_plus(stderr, "Error writing encrypted data: %s\n",
504
fprintf(stderr, "Error writing encrypted data: %s\n",
954
510
written += (size_t)ret;
960
/* Shutdown procedure */
965
free(decrypted_buffer);
968
ret = (int)TEMP_FAILURE_RETRY(close(tcp_sd));
974
perror_plus("close");
976
gnutls_deinit(session);
512
free(decrypted_buffer);
521
fprintf(stderr, "Closing TLS session\n");
525
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
528
gnutls_deinit (es.session);
529
gnutls_certificate_free_credentials (es.cred);
530
gnutls_global_deinit ();
986
static void resolve_callback(AvahiSServiceResolver *r,
987
AvahiIfIndex interface,
989
AvahiResolverEvent event,
993
const char *host_name,
994
const AvahiAddress *address,
996
AVAHI_GCC_UNUSED AvahiStringList *txt,
997
AVAHI_GCC_UNUSED AvahiLookupResultFlags
999
AVAHI_GCC_UNUSED void* userdata){
534
static AvahiSimplePoll *simple_poll = NULL;
535
static AvahiServer *server = NULL;
537
static void resolve_callback(
538
AvahiSServiceResolver *r,
539
AvahiIfIndex interface,
540
AVAHI_GCC_UNUSED AvahiProtocol protocol,
541
AvahiResolverEvent event,
545
const char *host_name,
546
const AvahiAddress *address,
548
AVAHI_GCC_UNUSED AvahiStringList *txt,
549
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
550
AVAHI_GCC_UNUSED void* userdata) {
552
assert(r); /* Spurious warning */
1002
554
/* Called whenever a service has been resolved successfully or
1011
559
case AVAHI_RESOLVER_FAILURE:
1012
fprintf_plus(stderr, "(Avahi Resolver) Failed to resolve service "
1013
"'%s' of type '%s' in domain '%s': %s\n", name, type,
1015
avahi_strerror(avahi_server_errno(mc.server)));
560
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of"
561
" type '%s' in domain '%s': %s\n", name, type, domain,
562
avahi_strerror(avahi_server_errno(server)));
1018
565
case AVAHI_RESOLVER_FOUND:
1020
567
char ip[AVAHI_ADDRESS_STR_MAX];
1021
568
avahi_address_snprint(ip, sizeof(ip), address);
1023
fprintf_plus(stderr, "Mandos server \"%s\" found on %s (%s, %"
1024
PRIdMAX ") on port %" PRIu16 "\n", name,
1025
host_name, ip, (intmax_t)interface, port);
570
fprintf(stderr, "Mandos server \"%s\" found on %s (%s) on"
571
" port %d\n", name, host_name, ip, port);
1027
int ret = start_mandos_communication(ip, port, interface,
1028
avahi_proto_to_af(proto));
1030
avahi_simple_poll_quit(mc.simple_poll);
1032
ret = add_server(ip, port, interface,
1033
avahi_proto_to_af(proto));
573
int ret = start_mandos_communication(ip, port, interface);
1037
579
avahi_s_service_resolver_free(r);
1040
static void browse_callback(AvahiSServiceBrowser *b,
1041
AvahiIfIndex interface,
1042
AvahiProtocol protocol,
1043
AvahiBrowserEvent event,
1047
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1049
AVAHI_GCC_UNUSED void* userdata){
1052
/* Called whenever a new services becomes available on the LAN or
1053
is removed from the LAN */
1061
case AVAHI_BROWSER_FAILURE:
1063
fprintf_plus(stderr, "(Avahi browser) %s\n",
1064
avahi_strerror(avahi_server_errno(mc.server)));
1065
avahi_simple_poll_quit(mc.simple_poll);
1068
case AVAHI_BROWSER_NEW:
1069
/* We ignore the returned Avahi resolver object. In the callback
1070
function we free it. If the Avahi server is terminated before
1071
the callback function is called the Avahi server will free the
1074
if(avahi_s_service_resolver_new(mc.server, interface, protocol,
1075
name, type, domain, protocol, 0,
1076
resolve_callback, NULL) == NULL)
1077
fprintf_plus(stderr, "Avahi: Failed to resolve service '%s':"
1079
avahi_strerror(avahi_server_errno(mc.server)));
1082
case AVAHI_BROWSER_REMOVE:
1085
case AVAHI_BROWSER_ALL_FOR_NOW:
1086
case AVAHI_BROWSER_CACHE_EXHAUSTED:
1088
fprintf_plus(stderr, "No Mandos server found, still"
1095
/* Signal handler that stops main loop after SIGTERM */
1096
static void handle_sigterm(int sig){
1101
signal_received = sig;
1102
int old_errno = errno;
1103
/* set main loop to exit */
1104
if(mc.simple_poll != NULL){
1105
avahi_simple_poll_quit(mc.simple_poll);
1110
bool get_flags(const char *ifname, struct ifreq *ifr){
1113
int s = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1115
perror_plus("socket");
1118
strcpy(ifr->ifr_name, ifname);
1119
ret = ioctl(s, SIOCGIFFLAGS, ifr);
1122
perror_plus("ioctl SIOCGIFFLAGS");
1129
bool good_flags(const char *ifname, const struct ifreq *ifr){
1131
/* Reject the loopback device */
1132
if(ifr->ifr_flags & IFF_LOOPBACK){
1134
fprintf_plus(stderr, "Rejecting loopback interface \"%s\"\n",
1139
/* Accept point-to-point devices only if connect_to is specified */
1140
if(connect_to != NULL and (ifr->ifr_flags & IFF_POINTOPOINT)){
1142
fprintf_plus(stderr, "Accepting point-to-point interface"
1143
" \"%s\"\n", ifname);
1147
/* Otherwise, reject non-broadcast-capable devices */
1148
if(not (ifr->ifr_flags & IFF_BROADCAST)){
1150
fprintf_plus(stderr, "Rejecting non-broadcast interface"
1151
" \"%s\"\n", ifname);
1155
/* Reject non-ARP interfaces (including dummy interfaces) */
1156
if(ifr->ifr_flags & IFF_NOARP){
1158
fprintf_plus(stderr, "Rejecting non-ARP interface \"%s\"\n",
1164
/* Accept this device */
1166
fprintf_plus(stderr, "Interface \"%s\" is good\n", ifname);
1172
* This function determines if a directory entry in /sys/class/net
1173
* corresponds to an acceptable network device.
1174
* (This function is passed to scandir(3) as a filter function.)
1176
int good_interface(const struct dirent *if_entry){
1177
if(if_entry->d_name[0] == '.'){
1182
if(not get_flags(if_entry->d_name, &ifr)){
1184
fprintf_plus(stderr, "Failed to get flags for interface "
1185
"\"%s\"\n", if_entry->d_name);
1190
if(not good_flags(if_entry->d_name, &ifr)){
1197
* This function determines if a directory entry in /sys/class/net
1198
* corresponds to an acceptable network device which is up.
1199
* (This function is passed to scandir(3) as a filter function.)
1201
int up_interface(const struct dirent *if_entry){
1202
if(if_entry->d_name[0] == '.'){
1207
if(not get_flags(if_entry->d_name, &ifr)){
1209
fprintf_plus(stderr, "Failed to get flags for interface "
1210
"\"%s\"\n", if_entry->d_name);
1215
/* Reject down interfaces */
1216
if(not (ifr.ifr_flags & IFF_UP)){
1218
fprintf_plus(stderr, "Rejecting down interface \"%s\"\n",
1224
/* Reject non-running interfaces */
1225
if(not (ifr.ifr_flags & IFF_RUNNING)){
1227
fprintf_plus(stderr, "Rejecting non-running interface \"%s\"\n",
1233
if(not good_flags(if_entry->d_name, &ifr)){
1239
int notdotentries(const struct dirent *direntry){
1240
/* Skip "." and ".." */
1241
if(direntry->d_name[0] == '.'
1242
and (direntry->d_name[1] == '\0'
1243
or (direntry->d_name[1] == '.'
1244
and direntry->d_name[2] == '\0'))){
1250
/* Is this directory entry a runnable program? */
1251
int runnable_hook(const struct dirent *direntry){
1256
if((direntry->d_name)[0] == '\0'){
1261
sret = strspn(direntry->d_name, "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1262
"abcdefghijklmnopqrstuvwxyz"
1265
if((direntry->d_name)[sret] != '\0'){
1266
/* Contains non-allowed characters */
1268
fprintf_plus(stderr, "Ignoring hook \"%s\" with bad name\n",
1274
char *fullname = NULL;
1275
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1277
perror_plus("asprintf");
1281
ret = stat(fullname, &st);
1284
perror_plus("Could not stat hook");
1288
if(not (S_ISREG(st.st_mode))){
1289
/* Not a regular file */
1291
fprintf_plus(stderr, "Ignoring hook \"%s\" - not a file\n",
1296
if(not (st.st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))){
1297
/* Not executable */
1299
fprintf_plus(stderr, "Ignoring hook \"%s\" - not executable\n",
1307
int avahi_loop_with_timeout(AvahiSimplePoll *s, int retry_interval){
1309
struct timespec now;
1310
struct timespec waited_time;
1311
intmax_t block_time;
1314
if(mc.current_server == NULL){
1316
fprintf_plus(stderr, "Wait until first server is found."
1319
ret = avahi_simple_poll_iterate(s, -1);
1322
fprintf_plus(stderr, "Check current_server if we should run"
1325
/* the current time */
1326
ret = clock_gettime(CLOCK_MONOTONIC, &now);
1328
perror_plus("clock_gettime");
1331
/* Calculating in ms how long time between now and server
1332
who we visted longest time ago. Now - last seen. */
1333
waited_time.tv_sec = (now.tv_sec
1334
- mc.current_server->last_seen.tv_sec);
1335
waited_time.tv_nsec = (now.tv_nsec
1336
- mc.current_server->last_seen.tv_nsec);
1337
/* total time is 10s/10,000ms.
1338
Converting to s from ms by dividing by 1,000,
1339
and ns to ms by dividing by 1,000,000. */
1340
block_time = ((retry_interval
1341
- ((intmax_t)waited_time.tv_sec * 1000))
1342
- ((intmax_t)waited_time.tv_nsec / 1000000));
1345
fprintf_plus(stderr, "Blocking for %" PRIdMAX " ms\n",
1349
if(block_time <= 0){
1350
ret = start_mandos_communication(mc.current_server->ip,
1351
mc.current_server->port,
1352
mc.current_server->if_index,
1353
mc.current_server->af);
1355
avahi_simple_poll_quit(mc.simple_poll);
1358
ret = clock_gettime(CLOCK_MONOTONIC,
1359
&mc.current_server->last_seen);
1361
perror_plus("clock_gettime");
1364
mc.current_server = mc.current_server->next;
1365
block_time = 0; /* Call avahi to find new Mandos
1366
servers, but don't block */
1369
ret = avahi_simple_poll_iterate(s, (int)block_time);
1372
if (ret > 0 or errno != EINTR){
1373
return (ret != 1) ? ret : 0;
1379
bool run_network_hooks(const char *mode, const char *interface,
1381
struct dirent **direntries;
1382
struct dirent *direntry;
1384
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1387
perror_plus("scandir");
1389
int devnull = open("/dev/null", O_RDONLY);
1390
for(int i = 0; i < numhooks; i++){
1391
direntry = direntries[i];
1392
char *fullname = NULL;
1393
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1395
perror_plus("asprintf");
1399
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1402
pid_t hook_pid = fork();
1405
dup2(devnull, STDIN_FILENO);
1407
dup2(STDERR_FILENO, STDOUT_FILENO);
1408
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1410
perror_plus("setenv");
1413
ret = setenv("DEVICE", interface, 1);
1415
perror_plus("setenv");
1418
ret = setenv("VERBOSE", debug ? "1" : "0", 1);
1420
perror_plus("setenv");
1423
ret = setenv("MODE", mode, 1);
1425
perror_plus("setenv");
1429
ret = asprintf(&delaystring, "%f", delay);
1431
perror_plus("asprintf");
1434
ret = setenv("DELAY", delaystring, 1);
1437
perror_plus("setenv");
1441
ret = execl(fullname, direntry->d_name, mode, NULL);
1442
perror_plus("execl");
1445
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1446
perror_plus("waitpid");
1450
if(WIFEXITED(status)){
1451
if(WEXITSTATUS(status) != 0){
1452
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1453
" with status %d\n", direntry->d_name,
1454
WEXITSTATUS(status));
1458
} else if(WIFSIGNALED(status)){
1459
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1460
" signal %d\n", direntry->d_name,
1465
fprintf_plus(stderr, "Warning: network hook \"%s\""
1466
" crashed\n", direntry->d_name);
1481
int main(int argc, char *argv[]){
1482
AvahiSServiceBrowser *sb = NULL;
1487
int exitcode = EXIT_SUCCESS;
1488
const char *interface = "";
1489
struct ifreq network;
1491
bool take_down_interface = false;
1494
char tempdir[] = "/tmp/mandosXXXXXX";
1495
bool tempdir_created = false;
1496
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
1497
const char *seckey = PATHDIR "/" SECKEY;
1498
const char *pubkey = PATHDIR "/" PUBKEY;
1500
bool gnutls_initialized = false;
1501
bool gpgme_initialized = false;
1503
double retry_interval = 10; /* 10s between trying a server and
1504
retrying the same server again */
1506
struct sigaction old_sigterm_action = { .sa_handler = SIG_DFL };
1507
struct sigaction sigterm_action = { .sa_handler = handle_sigterm };
1512
/* Lower any group privileges we might have, just to be safe */
1516
perror_plus("setgid");
1519
/* Lower user privileges (temporarily) */
1523
perror_plus("seteuid");
1531
struct argp_option options[] = {
1532
{ .name = "debug", .key = 128,
1533
.doc = "Debug mode", .group = 3 },
1534
{ .name = "connect", .key = 'c',
1535
.arg = "ADDRESS:PORT",
1536
.doc = "Connect directly to a specific Mandos server",
1538
{ .name = "interface", .key = 'i',
1540
.doc = "Network interface that will be used to search for"
1543
{ .name = "seckey", .key = 's',
1545
.doc = "OpenPGP secret key file base name",
1547
{ .name = "pubkey", .key = 'p',
1549
.doc = "OpenPGP public key file base name",
1551
{ .name = "dh-bits", .key = 129,
1553
.doc = "Bit length of the prime number used in the"
1554
" Diffie-Hellman key exchange",
1556
{ .name = "priority", .key = 130,
1558
.doc = "GnuTLS priority string for the TLS handshake",
1560
{ .name = "delay", .key = 131,
1562
.doc = "Maximum delay to wait for interface startup",
1564
{ .name = "retry", .key = 132,
1566
.doc = "Retry interval used when denied by the mandos server",
1568
{ .name = "network-hook-dir", .key = 133,
1570
.doc = "Directory where network hooks are located",
1573
* These reproduce what we would get without ARGP_NO_HELP
1575
{ .name = "help", .key = '?',
1576
.doc = "Give this help list", .group = -1 },
1577
{ .name = "usage", .key = -3,
1578
.doc = "Give a short usage message", .group = -1 },
1579
{ .name = "version", .key = 'V',
1580
.doc = "Print program version", .group = -1 },
1584
error_t parse_opt(int key, char *arg,
1585
struct argp_state *state){
1588
case 128: /* --debug */
1591
case 'c': /* --connect */
1594
case 'i': /* --interface */
1597
case 's': /* --seckey */
1600
case 'p': /* --pubkey */
1603
case 129: /* --dh-bits */
1605
tmpmax = strtoimax(arg, &tmp, 10);
1606
if(errno != 0 or tmp == arg or *tmp != '\0'
1607
or tmpmax != (typeof(mc.dh_bits))tmpmax){
1608
argp_error(state, "Bad number of DH bits");
1610
mc.dh_bits = (typeof(mc.dh_bits))tmpmax;
1612
case 130: /* --priority */
1615
case 131: /* --delay */
1617
delay = strtof(arg, &tmp);
1618
if(errno != 0 or tmp == arg or *tmp != '\0'){
1619
argp_error(state, "Bad delay");
1621
case 132: /* --retry */
1623
retry_interval = strtod(arg, &tmp);
1624
if(errno != 0 or tmp == arg or *tmp != '\0'
1625
or (retry_interval * 1000) > INT_MAX
1626
or retry_interval < 0){
1627
argp_error(state, "Bad retry interval");
1630
case 133: /* --network-hook-dir */
1634
* These reproduce what we would get without ARGP_NO_HELP
1636
case '?': /* --help */
1637
argp_state_help(state, state->out_stream,
1638
(ARGP_HELP_STD_HELP | ARGP_HELP_EXIT_ERR)
1639
& ~(unsigned int)ARGP_HELP_EXIT_OK);
1640
case -3: /* --usage */
1641
argp_state_help(state, state->out_stream,
1642
ARGP_HELP_USAGE | ARGP_HELP_EXIT_ERR);
1643
case 'V': /* --version */
1644
fprintf_plus(state->out_stream,
1645
"Mandos plugin mandos-client: ");
1646
fprintf_plus(state->out_stream, "%s\n", argp_program_version);
1647
exit(argp_err_exit_status);
1650
return ARGP_ERR_UNKNOWN;
1655
struct argp argp = { .options = options, .parser = parse_opt,
1657
.doc = "Mandos client -- Get and decrypt"
1658
" passwords from a Mandos server" };
1659
ret = argp_parse(&argp, argc, argv,
1660
ARGP_IN_ORDER | ARGP_NO_HELP, 0, NULL);
582
static void browse_callback(
583
AvahiSServiceBrowser *b,
584
AvahiIfIndex interface,
585
AvahiProtocol protocol,
586
AvahiBrowserEvent event,
590
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
593
AvahiServer *s = userdata;
594
assert(b); /* Spurious warning */
596
/* Called whenever a new services becomes available on the LAN or
597
is removed from the LAN */
1667
perror_plus("argp_parse");
1668
exitcode = EX_OSERR;
1671
exitcode = EX_USAGE;
1677
/* Work around Debian bug #633582:
1678
<http://bugs.debian.org/633582> */
1681
/* Re-raise priviliges */
1685
perror_plus("seteuid");
1688
if(strcmp(seckey, PATHDIR "/" SECKEY) == 0){
1689
int seckey_fd = open(seckey, O_RDONLY);
1690
if(seckey_fd == -1){
1691
perror_plus("open");
1693
ret = (int)TEMP_FAILURE_RETRY(fstat(seckey_fd, &st));
1695
perror_plus("fstat");
1697
if(S_ISREG(st.st_mode)
1698
and st.st_uid == 0 and st.st_gid == 0){
1699
ret = fchown(seckey_fd, uid, gid);
1701
perror_plus("fchown");
1705
TEMP_FAILURE_RETRY(close(seckey_fd));
1709
if(strcmp(pubkey, PATHDIR "/" PUBKEY) == 0){
1710
int pubkey_fd = open(pubkey, O_RDONLY);
1711
if(pubkey_fd == -1){
1712
perror_plus("open");
1714
ret = (int)TEMP_FAILURE_RETRY(fstat(pubkey_fd, &st));
1716
perror_plus("fstat");
1718
if(S_ISREG(st.st_mode)
1719
and st.st_uid == 0 and st.st_gid == 0){
1720
ret = fchown(pubkey_fd, uid, gid);
1722
perror_plus("fchown");
1726
TEMP_FAILURE_RETRY(close(pubkey_fd));
1730
/* Lower privileges */
1734
perror_plus("seteuid");
1738
/* Run network hooks */
1741
/* Re-raise priviliges */
1745
perror_plus("seteuid");
1748
if(not run_network_hooks("start", interface, delay)){
1752
/* Lower privileges */
1756
perror_plus("seteuid");
1762
avahi_set_log_function(empty_log);
1765
if(interface[0] == '\0'){
1766
struct dirent **direntries;
1767
/* First look for interfaces that are up */
1768
ret = scandir(sys_class_net, &direntries, up_interface,
1771
/* No up interfaces, look for any good interfaces */
1773
ret = scandir(sys_class_net, &direntries, good_interface,
1777
/* Pick the first interface returned */
1778
interface = strdup(direntries[0]->d_name);
1780
fprintf_plus(stderr, "Using interface \"%s\"\n", interface);
1782
if(interface == NULL){
1783
perror_plus("malloc");
1785
exitcode = EXIT_FAILURE;
1791
fprintf_plus(stderr, "Could not find a network interface\n");
1792
exitcode = EXIT_FAILURE;
1797
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
1798
from the signal handler */
1799
/* Initialize the pseudo-RNG for Avahi */
1800
srand((unsigned int) time(NULL));
1801
mc.simple_poll = avahi_simple_poll_new();
1802
if(mc.simple_poll == NULL){
1803
fprintf_plus(stderr,
1804
"Avahi: Failed to create simple poll object.\n");
1805
exitcode = EX_UNAVAILABLE;
1809
sigemptyset(&sigterm_action.sa_mask);
1810
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
1812
perror_plus("sigaddset");
1813
exitcode = EX_OSERR;
1816
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
1818
perror_plus("sigaddset");
1819
exitcode = EX_OSERR;
1822
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
1824
perror_plus("sigaddset");
1825
exitcode = EX_OSERR;
1828
/* Need to check if the handler is SIG_IGN before handling:
1829
| [[info:libc:Initial Signal Actions]] |
1830
| [[info:libc:Basic Signal Handling]] |
1832
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
1834
perror_plus("sigaction");
1837
if(old_sigterm_action.sa_handler != SIG_IGN){
1838
ret = sigaction(SIGINT, &sigterm_action, NULL);
1840
perror_plus("sigaction");
1841
exitcode = EX_OSERR;
1845
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
1847
perror_plus("sigaction");
1850
if(old_sigterm_action.sa_handler != SIG_IGN){
1851
ret = sigaction(SIGHUP, &sigterm_action, NULL);
1853
perror_plus("sigaction");
1854
exitcode = EX_OSERR;
1858
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
1860
perror_plus("sigaction");
1863
if(old_sigterm_action.sa_handler != SIG_IGN){
1864
ret = sigaction(SIGTERM, &sigterm_action, NULL);
1866
perror_plus("sigaction");
1867
exitcode = EX_OSERR;
1872
/* If the interface is down, bring it up */
1873
if(strcmp(interface, "none") != 0){
1874
if_index = (AvahiIfIndex) if_nametoindex(interface);
1876
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1877
exitcode = EX_UNAVAILABLE;
1885
/* Re-raise priviliges */
1889
perror_plus("seteuid");
1893
/* Lower kernel loglevel to KERN_NOTICE to avoid KERN_INFO
1894
messages about the network interface to mess up the prompt */
1895
ret = klogctl(8, NULL, 5);
1896
bool restore_loglevel = true;
1898
restore_loglevel = false;
1899
perror_plus("klogctl");
1901
#endif /* __linux__ */
1903
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1905
perror_plus("socket");
1906
exitcode = EX_OSERR;
1908
if(restore_loglevel){
1909
ret = klogctl(7, NULL, 0);
1911
perror_plus("klogctl");
1914
#endif /* __linux__ */
1915
/* Lower privileges */
1919
perror_plus("seteuid");
1923
strcpy(network.ifr_name, interface);
1924
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1926
perror_plus("ioctl SIOCGIFFLAGS");
1928
if(restore_loglevel){
1929
ret = klogctl(7, NULL, 0);
1931
perror_plus("klogctl");
1934
#endif /* __linux__ */
1935
exitcode = EX_OSERR;
1936
/* Lower privileges */
1940
perror_plus("seteuid");
1944
if((network.ifr_flags & IFF_UP) == 0){
1945
network.ifr_flags |= IFF_UP;
1946
take_down_interface = true;
1947
ret = ioctl(sd, SIOCSIFFLAGS, &network);
1949
take_down_interface = false;
1950
perror_plus("ioctl SIOCSIFFLAGS +IFF_UP");
1951
exitcode = EX_OSERR;
1953
if(restore_loglevel){
1954
ret = klogctl(7, NULL, 0);
1956
perror_plus("klogctl");
1959
#endif /* __linux__ */
1960
/* Lower privileges */
1964
perror_plus("seteuid");
1969
/* Sleep checking until interface is running.
1970
Check every 0.25s, up to total time of delay */
1971
for(int i=0; i < delay * 4; i++){
1972
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1974
perror_plus("ioctl SIOCGIFFLAGS");
1975
} else if(network.ifr_flags & IFF_RUNNING){
1978
struct timespec sleeptime = { .tv_nsec = 250000000 };
1979
ret = nanosleep(&sleeptime, NULL);
1980
if(ret == -1 and errno != EINTR){
1981
perror_plus("nanosleep");
1984
if(not take_down_interface){
1985
/* We won't need the socket anymore */
1986
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1988
perror_plus("close");
1992
if(restore_loglevel){
1993
/* Restores kernel loglevel to default */
1994
ret = klogctl(7, NULL, 0);
1996
perror_plus("klogctl");
1999
#endif /* __linux__ */
2000
/* Lower privileges */
2002
/* Lower privileges */
2005
perror_plus("seteuid");
2013
ret = init_gnutls_global(pubkey, seckey);
2015
fprintf_plus(stderr, "init_gnutls_global failed\n");
2016
exitcode = EX_UNAVAILABLE;
2019
gnutls_initialized = true;
2026
if(mkdtemp(tempdir) == NULL){
2027
perror_plus("mkdtemp");
2030
tempdir_created = true;
2036
if(not init_gpgme(pubkey, seckey, tempdir)){
2037
fprintf_plus(stderr, "init_gpgme failed\n");
2038
exitcode = EX_UNAVAILABLE;
2041
gpgme_initialized = true;
2048
if(connect_to != NULL){
2049
/* Connect directly, do not use Zeroconf */
2050
/* (Mainly meant for debugging) */
2051
char *address = strrchr(connect_to, ':');
2052
if(address == NULL){
2053
fprintf_plus(stderr, "No colon in address\n");
2054
exitcode = EX_USAGE;
2064
tmpmax = strtoimax(address+1, &tmp, 10);
2065
if(errno != 0 or tmp == address+1 or *tmp != '\0'
2066
or tmpmax != (uint16_t)tmpmax){
2067
fprintf_plus(stderr, "Bad port number\n");
2068
exitcode = EX_USAGE;
2076
port = (uint16_t)tmpmax;
2078
/* Colon in address indicates IPv6 */
2080
if(strchr(connect_to, ':') != NULL){
2082
/* Accept [] around IPv6 address - see RFC 5952 */
2083
if(connect_to[0] == '[' and address[-1] == ']')
2091
address = connect_to;
2097
while(not quit_now){
2098
ret = start_mandos_communication(address, port, if_index, af);
2099
if(quit_now or ret == 0){
2103
fprintf_plus(stderr, "Retrying in %d seconds\n",
2104
(int)retry_interval);
2106
sleep((int)retry_interval);
2110
exitcode = EXIT_SUCCESS;
601
case AVAHI_BROWSER_FAILURE:
603
fprintf(stderr, "(Browser) %s\n",
604
avahi_strerror(avahi_server_errno(server)));
605
avahi_simple_poll_quit(simple_poll);
608
case AVAHI_BROWSER_NEW:
609
/* We ignore the returned resolver object. In the callback
610
function we free it. If the server is terminated before
611
the callback function is called the server will free
612
the resolver for us. */
614
if (!(avahi_s_service_resolver_new(s, interface, protocol, name,
616
AVAHI_PROTO_INET6, 0,
617
resolve_callback, s)))
618
fprintf(stderr, "Failed to resolve service '%s': %s\n", name,
619
avahi_strerror(avahi_server_errno(s)));
622
case AVAHI_BROWSER_REMOVE:
625
case AVAHI_BROWSER_ALL_FOR_NOW:
626
case AVAHI_BROWSER_CACHE_EXHAUSTED:
631
/* Combines file name and path and returns the malloced new
632
string. some sane checks could/should be added */
633
static const char *combinepath(const char *first, const char *second){
634
size_t f_len = strlen(first);
635
size_t s_len = strlen(second);
636
char *tmp = malloc(f_len + s_len + 2);
641
memcpy(tmp, first, f_len);
645
memcpy(tmp + f_len + 1, second, s_len);
647
tmp[f_len + 1 + s_len] = '\0';
652
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
2121
653
AvahiServerConfig config;
2122
/* Do not publish any local Zeroconf records */
654
AvahiSServiceBrowser *sb = NULL;
657
int returncode = EXIT_SUCCESS;
658
const char *interface = NULL;
659
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
660
char *connect_to = NULL;
663
static struct option long_options[] = {
664
{"debug", no_argument, (int *)&debug, 1},
665
{"connect", required_argument, 0, 'C'},
666
{"interface", required_argument, 0, 'i'},
667
{"certdir", required_argument, 0, 'd'},
668
{"certkey", required_argument, 0, 'c'},
669
{"certfile", required_argument, 0, 'k'},
672
int option_index = 0;
673
ret = getopt_long (argc, argv, "i:", long_options,
703
certfile = combinepath(certdir, certfile);
704
if (certfile == NULL){
705
perror("combinepath");
709
if(interface != NULL){
710
if_index = (AvahiIfIndex) if_nametoindex(interface);
712
fprintf(stderr, "No such interface: \"%s\"\n", interface);
717
if(connect_to != NULL){
718
/* Connect directly, do not use Zeroconf */
719
/* (Mainly meant for debugging) */
720
char *address = strrchr(connect_to, ':');
722
fprintf(stderr, "No colon in address\n");
726
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
728
perror("Bad port number");
732
address = connect_to;
733
ret = start_mandos_communication(address, port, if_index);
741
certkey = combinepath(certdir, certkey);
742
if (certkey == NULL){
743
perror("combinepath");
748
avahi_set_log_function(empty_log);
751
/* Initialize the psuedo-RNG */
752
srand((unsigned int) time(NULL));
754
/* Allocate main loop object */
755
if (!(simple_poll = avahi_simple_poll_new())) {
756
fprintf(stderr, "Failed to create simple poll object.\n");
761
/* Do not publish any local records */
2123
762
avahi_server_config_init(&config);
2124
763
config.publish_hinfo = 0;
2125
764
config.publish_addresses = 0;
2126
765
config.publish_workstation = 0;
2127
766
config.publish_domain = 0;
2129
768
/* Allocate a new server */
2130
mc.server = avahi_server_new(avahi_simple_poll_get
2131
(mc.simple_poll), &config, NULL,
2134
/* Free the Avahi configuration data */
769
server = avahi_server_new(avahi_simple_poll_get(simple_poll),
770
&config, NULL, NULL, &error);
772
/* Free the configuration data */
2135
773
avahi_server_config_free(&config);
2138
/* Check if creating the Avahi server object succeeded */
2139
if(mc.server == NULL){
2140
fprintf_plus(stderr, "Failed to create Avahi server: %s\n",
2141
avahi_strerror(error));
2142
exitcode = EX_UNAVAILABLE;
2150
/* Create the Avahi service browser */
2151
sb = avahi_s_service_browser_new(mc.server, if_index,
2152
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
2153
NULL, 0, browse_callback, NULL);
2155
fprintf_plus(stderr, "Failed to create service browser: %s\n",
2156
avahi_strerror(avahi_server_errno(mc.server)));
2157
exitcode = EX_UNAVAILABLE;
2165
/* Run the main loop */
2168
fprintf_plus(stderr, "Starting Avahi loop search\n");
2171
ret = avahi_loop_with_timeout(mc.simple_poll,
2172
(int)(retry_interval * 1000));
2174
fprintf_plus(stderr, "avahi_loop_with_timeout exited %s\n",
2175
(ret == 0) ? "successfully" : "with error");
2181
fprintf_plus(stderr, "%s exiting\n", argv[0]);
2184
/* Cleanup things */
2186
avahi_s_service_browser_free(sb);
2188
if(mc.server != NULL)
2189
avahi_server_free(mc.server);
2191
if(mc.simple_poll != NULL)
2192
avahi_simple_poll_free(mc.simple_poll);
2194
if(gnutls_initialized){
2195
gnutls_certificate_free_credentials(mc.cred);
2196
gnutls_global_deinit();
2197
gnutls_dh_params_deinit(mc.dh_params);
2200
if(gpgme_initialized){
2201
gpgme_release(mc.ctx);
2204
/* Cleans up the circular linked list of Mandos servers the client
2206
if(mc.current_server != NULL){
2207
mc.current_server->prev->next = NULL;
2208
while(mc.current_server != NULL){
2209
server *next = mc.current_server->next;
2210
free(mc.current_server);
2211
mc.current_server = next;
2215
/* Re-raise priviliges */
2221
perror_plus("seteuid");
2224
/* Run network hooks */
2225
if(not run_network_hooks("stop", interface, delay)){
2229
/* Take down the network interface */
2230
if(take_down_interface and geteuid() == 0){
2231
ret = ioctl(sd, SIOCGIFFLAGS, &network);
2233
perror_plus("ioctl SIOCGIFFLAGS");
2234
} else if(network.ifr_flags & IFF_UP){
2235
network.ifr_flags &= ~(short)IFF_UP; /* clear flag */
2236
ret = ioctl(sd, SIOCSIFFLAGS, &network);
2238
perror_plus("ioctl SIOCSIFFLAGS -IFF_UP");
2241
ret = (int)TEMP_FAILURE_RETRY(close(sd));
2243
perror_plus("close");
2248
/* Lower privileges permanently */
2252
perror_plus("setuid");
2256
/* Removes the GPGME temp directory and all files inside */
2257
if(tempdir_created){
2258
struct dirent **direntries = NULL;
2259
struct dirent *direntry = NULL;
2260
int numentries = scandir(tempdir, &direntries, notdotentries,
2262
if (numentries > 0){
2263
for(int i = 0; i < numentries; i++){
2264
direntry = direntries[i];
2265
char *fullname = NULL;
2266
ret = asprintf(&fullname, "%s/%s", tempdir,
2269
perror_plus("asprintf");
2272
ret = remove(fullname);
2274
fprintf_plus(stderr, "remove(\"%s\"): %s\n", fullname,
2281
/* need to clean even if 0 because man page doesn't specify */
2283
if (numentries == -1){
2284
perror_plus("scandir");
2286
ret = rmdir(tempdir);
2287
if(ret == -1 and errno != ENOENT){
2288
perror_plus("rmdir");
2293
sigemptyset(&old_sigterm_action.sa_mask);
2294
old_sigterm_action.sa_handler = SIG_DFL;
2295
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
2296
&old_sigterm_action,
2299
perror_plus("sigaction");
2302
ret = raise(signal_received);
2303
} while(ret != 0 and errno == EINTR);
2305
perror_plus("raise");
2308
TEMP_FAILURE_RETRY(pause());
775
/* Check if creating the server object succeeded */
777
fprintf(stderr, "Failed to create server: %s\n",
778
avahi_strerror(error));
779
returncode = EXIT_FAILURE;
783
/* Create the service browser */
784
sb = avahi_s_service_browser_new(server, if_index,
786
"_mandos._tcp", NULL, 0,
787
browse_callback, server);
789
fprintf(stderr, "Failed to create service browser: %s\n",
790
avahi_strerror(avahi_server_errno(server)));
791
returncode = EXIT_FAILURE;
795
/* Run the main loop */
798
fprintf(stderr, "Starting avahi loop search\n");
801
avahi_simple_poll_loop(simple_poll);
806
fprintf(stderr, "%s exiting\n", argv[0]);
811
avahi_s_service_browser_free(sb);
814
avahi_server_free(server);
817
avahi_simple_poll_free(simple_poll);