47
107
#include <avahi-common/malloc.h>
48
108
#include <avahi-common/error.h>
51
#include <sys/types.h> /* socket(), inet_pton() */
52
#include <sys/socket.h> /* socket(), struct sockaddr_in6,
53
struct in6_addr, inet_pton() */
54
#include <gnutls/gnutls.h> /* All GnuTLS stuff */
55
#include <gnutls/openpgp.h> /* GnuTLS with openpgp stuff */
57
#include <unistd.h> /* close() */
58
#include <netinet/in.h>
59
#include <stdbool.h> /* true */
60
#include <string.h> /* memset */
61
#include <arpa/inet.h> /* inet_pton() */
62
#include <iso646.h> /* not */
65
#include <errno.h> /* perror() */
111
#include <gnutls/gnutls.h> /* All GnuTLS types, constants and
114
init_gnutls_session(),
116
#include <gnutls/openpgp.h>
117
/* gnutls_certificate_set_openpgp_key_file(),
118
GNUTLS_OPENPGP_FMT_BASE64 */
121
#include <gpgme.h> /* All GPGME types, constants and
124
GPGME_PROTOCOL_OpenPGP,
71
127
#define BUFFER_SIZE 256
74
static const char *certdir = "/conf/conf.d/mandos";
75
static const char *certfile = "openpgp-client.txt";
76
static const char *certkey = "openpgp-client-key.txt";
129
#define PATHDIR "/conf/conf.d/mandos"
130
#define SECKEY "seckey.txt"
131
#define PUBKEY "pubkey.txt"
132
#define HOOKDIR "/lib/mandos/network-hooks.d"
78
134
bool debug = false;
135
static const char mandos_protocol_version[] = "1";
136
const char *argp_program_version = "mandos-client " VERSION;
137
const char *argp_program_bug_address = "<mandos@recompile.se>";
138
static const char sys_class_net[] = "/sys/class/net";
139
char *connect_to = NULL;
140
const char *hookdir = HOOKDIR;
144
/* Doubly linked list that need to be circularly linked when used */
145
typedef struct server{
148
AvahiIfIndex if_index;
150
struct timespec last_seen;
155
/* Used for passing in values through the Avahi callback functions */
81
gnutls_session_t session;
157
AvahiSimplePoll *simple_poll;
82
159
gnutls_certificate_credentials_t cred;
160
unsigned int dh_bits;
83
161
gnutls_dh_params_t dh_params;
87
static ssize_t pgp_packet_decrypt (char *packet, size_t packet_size,
90
gpgme_data_t dh_crypto, dh_plain;
162
const char *priority;
164
server *current_server;
167
/* global context so signal handler can reach it*/
168
mandos_context mc = { .simple_poll = NULL, .server = NULL,
169
.dh_bits = 1024, .priority = "SECURE256"
170
":!CTYPE-X.509:+CTYPE-OPENPGP",
171
.current_server = NULL };
173
sig_atomic_t quit_now = 0;
174
int signal_received = 0;
176
/* Function to use when printing errors */
177
void perror_plus(const char *print_text){
179
fprintf(stderr, "Mandos plugin %s: ",
180
program_invocation_short_name);
185
__attribute__((format (gnu_printf, 2, 3)))
186
int fprintf_plus(FILE *stream, const char *format, ...){
188
va_start (ap, format);
190
TEMP_FAILURE_RETRY(fprintf(stream, "Mandos plugin %s: ",
191
program_invocation_short_name));
192
return TEMP_FAILURE_RETRY(vfprintf(stream, format, ap));
196
* Make additional room in "buffer" for at least BUFFER_SIZE more
197
* bytes. "buffer_capacity" is how much is currently allocated,
198
* "buffer_length" is how much is already used.
200
size_t incbuffer(char **buffer, size_t buffer_length,
201
size_t buffer_capacity){
202
if(buffer_length + BUFFER_SIZE > buffer_capacity){
203
*buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
207
buffer_capacity += BUFFER_SIZE;
209
return buffer_capacity;
212
/* Add server to set of servers to retry periodically */
213
bool add_server(const char *ip, in_port_t port, AvahiIfIndex if_index,
216
server *new_server = malloc(sizeof(server));
217
if(new_server == NULL){
218
perror_plus("malloc");
221
*new_server = (server){ .ip = strdup(ip),
223
.if_index = if_index,
225
if(new_server->ip == NULL){
226
perror_plus("strdup");
229
/* Special case of first server */
230
if (mc.current_server == NULL){
231
new_server->next = new_server;
232
new_server->prev = new_server;
233
mc.current_server = new_server;
234
/* Place the new server last in the list */
236
new_server->next = mc.current_server;
237
new_server->prev = mc.current_server->prev;
238
new_server->prev->next = new_server;
239
mc.current_server->prev = new_server;
241
ret = clock_gettime(CLOCK_MONOTONIC, &mc.current_server->last_seen);
243
perror_plus("clock_gettime");
252
static bool init_gpgme(const char *seckey, const char *pubkey,
253
const char *tempdir){
94
ssize_t new_packet_capacity = 0;
95
ssize_t new_packet_length = 0;
96
255
gpgme_engine_info_t engine_info;
99
fprintf(stderr, "Trying to decrypt OpenPGP packet\n");
259
* Helper function to insert pub and seckey to the engine keyring.
261
bool import_key(const char *filename){
264
gpgme_data_t pgp_data;
266
fd = (int)TEMP_FAILURE_RETRY(open(filename, O_RDONLY));
272
rc = gpgme_data_new_from_fd(&pgp_data, fd);
273
if(rc != GPG_ERR_NO_ERROR){
274
fprintf_plus(stderr, "bad gpgme_data_new_from_fd: %s: %s\n",
275
gpgme_strsource(rc), gpgme_strerror(rc));
279
rc = gpgme_op_import(mc.ctx, pgp_data);
280
if(rc != GPG_ERR_NO_ERROR){
281
fprintf_plus(stderr, "bad gpgme_op_import: %s: %s\n",
282
gpgme_strsource(rc), gpgme_strerror(rc));
286
ret = (int)TEMP_FAILURE_RETRY(close(fd));
288
perror_plus("close");
290
gpgme_data_release(pgp_data);
295
fprintf_plus(stderr, "Initializing GPGME\n");
103
299
gpgme_check_version(NULL);
104
300
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
105
if (rc != GPG_ERR_NO_ERROR){
106
fprintf(stderr, "bad gpgme_engine_check_version: %s: %s\n",
107
gpgme_strsource(rc), gpgme_strerror(rc));
301
if(rc != GPG_ERR_NO_ERROR){
302
fprintf_plus(stderr, "bad gpgme_engine_check_version: %s: %s\n",
303
gpgme_strsource(rc), gpgme_strerror(rc));
111
/* Set GPGME home directory */
112
rc = gpgme_get_engine_info (&engine_info);
113
if (rc != GPG_ERR_NO_ERROR){
114
fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
115
gpgme_strsource(rc), gpgme_strerror(rc));
307
/* Set GPGME home directory for the OpenPGP engine only */
308
rc = gpgme_get_engine_info(&engine_info);
309
if(rc != GPG_ERR_NO_ERROR){
310
fprintf_plus(stderr, "bad gpgme_get_engine_info: %s: %s\n",
311
gpgme_strsource(rc), gpgme_strerror(rc));
118
314
while(engine_info != NULL){
119
315
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
120
316
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
121
engine_info->file_name, homedir);
317
engine_info->file_name, tempdir);
124
320
engine_info = engine_info->next;
126
322
if(engine_info == NULL){
127
fprintf(stderr, "Could not set home dir to %s\n", homedir);
131
/* Create new GPGME data buffer from packet buffer */
132
rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
133
if (rc != GPG_ERR_NO_ERROR){
134
fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
135
gpgme_strsource(rc), gpgme_strerror(rc));
323
fprintf_plus(stderr, "Could not set GPGME home dir to %s\n",
328
/* Create new GPGME "context" */
329
rc = gpgme_new(&(mc.ctx));
330
if(rc != GPG_ERR_NO_ERROR){
331
fprintf_plus(stderr, "Mandos plugin mandos-client: "
332
"bad gpgme_new: %s: %s\n", gpgme_strsource(rc),
337
if(not import_key(pubkey) or not import_key(seckey)){
345
* Decrypt OpenPGP data.
346
* Returns -1 on error
348
static ssize_t pgp_packet_decrypt(const char *cryptotext,
351
gpgme_data_t dh_crypto, dh_plain;
354
size_t plaintext_capacity = 0;
355
ssize_t plaintext_length = 0;
358
fprintf_plus(stderr, "Trying to decrypt OpenPGP data\n");
361
/* Create new GPGME data buffer from memory cryptotext */
362
rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
364
if(rc != GPG_ERR_NO_ERROR){
365
fprintf_plus(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
366
gpgme_strsource(rc), gpgme_strerror(rc));
139
370
/* Create new empty GPGME data buffer for the plaintext */
140
371
rc = gpgme_data_new(&dh_plain);
141
if (rc != GPG_ERR_NO_ERROR){
142
fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
143
gpgme_strsource(rc), gpgme_strerror(rc));
147
/* Create new GPGME "context" */
148
rc = gpgme_new(&ctx);
149
if (rc != GPG_ERR_NO_ERROR){
150
fprintf(stderr, "bad gpgme_new: %s: %s\n",
151
gpgme_strsource(rc), gpgme_strerror(rc));
155
/* Decrypt data from the FILE pointer to the plaintext data
157
rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
158
if (rc != GPG_ERR_NO_ERROR){
159
fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
160
gpgme_strsource(rc), gpgme_strerror(rc));
165
fprintf(stderr, "Decryption of OpenPGP packet succeeded\n");
169
gpgme_decrypt_result_t result;
170
result = gpgme_op_decrypt_result(ctx);
172
fprintf(stderr, "gpgme_op_decrypt_result failed\n");
174
fprintf(stderr, "Unsupported algorithm: %s\n",
175
result->unsupported_algorithm);
176
fprintf(stderr, "Wrong key usage: %d\n",
177
result->wrong_key_usage);
178
if(result->file_name != NULL){
179
fprintf(stderr, "File name: %s\n", result->file_name);
181
gpgme_recipient_t recipient;
182
recipient = result->recipients;
372
if(rc != GPG_ERR_NO_ERROR){
373
fprintf_plus(stderr, "Mandos plugin mandos-client: "
374
"bad gpgme_data_new: %s: %s\n",
375
gpgme_strsource(rc), gpgme_strerror(rc));
376
gpgme_data_release(dh_crypto);
380
/* Decrypt data from the cryptotext data buffer to the plaintext
382
rc = gpgme_op_decrypt(mc.ctx, dh_crypto, dh_plain);
383
if(rc != GPG_ERR_NO_ERROR){
384
fprintf_plus(stderr, "bad gpgme_op_decrypt: %s: %s\n",
385
gpgme_strsource(rc), gpgme_strerror(rc));
386
plaintext_length = -1;
388
gpgme_decrypt_result_t result;
389
result = gpgme_op_decrypt_result(mc.ctx);
391
fprintf_plus(stderr, "gpgme_op_decrypt_result failed\n");
393
fprintf_plus(stderr, "Unsupported algorithm: %s\n",
394
result->unsupported_algorithm);
395
fprintf_plus(stderr, "Wrong key usage: %u\n",
396
result->wrong_key_usage);
397
if(result->file_name != NULL){
398
fprintf_plus(stderr, "File name: %s\n", result->file_name);
400
gpgme_recipient_t recipient;
401
recipient = result->recipients;
184
402
while(recipient != NULL){
185
fprintf(stderr, "Public key algorithm: %s\n",
186
gpgme_pubkey_algo_name(recipient->pubkey_algo));
187
fprintf(stderr, "Key ID: %s\n", recipient->keyid);
188
fprintf(stderr, "Secret key available: %s\n",
189
recipient->status == GPG_ERR_NO_SECKEY
403
fprintf_plus(stderr, "Public key algorithm: %s\n",
404
gpgme_pubkey_algo_name
405
(recipient->pubkey_algo));
406
fprintf_plus(stderr, "Key ID: %s\n", recipient->keyid);
407
fprintf_plus(stderr, "Secret key available: %s\n",
408
recipient->status == GPG_ERR_NO_SECKEY
191
410
recipient = recipient->next;
197
/* Delete the GPGME FILE pointer cryptotext data buffer */
198
gpgme_data_release(dh_crypto);
418
fprintf_plus(stderr, "Decryption of OpenPGP data succeeded\n");
200
421
/* Seek back to the beginning of the GPGME plaintext data buffer */
201
if (gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET) == -1){
202
perror("pgpme_data_seek");
422
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
423
perror_plus("gpgme_data_seek");
424
plaintext_length = -1;
207
if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
208
*new_packet = realloc(*new_packet,
209
(unsigned int)new_packet_capacity
211
if (*new_packet == NULL){
215
new_packet_capacity += BUFFER_SIZE;
430
plaintext_capacity = incbuffer(plaintext,
431
(size_t)plaintext_length,
433
if(plaintext_capacity == 0){
434
perror_plus("incbuffer");
435
plaintext_length = -1;
218
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length,
439
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
220
441
/* Print the data, if any */
225
perror("gpgme_data_read");
228
new_packet_length += ret;
231
/* FIXME: check characters before printing to screen so to not print
232
terminal control characters */
234
/* fprintf(stderr, "decrypted password is: "); */
235
/* fwrite(*new_packet, 1, new_packet_length, stderr); */
236
/* fprintf(stderr, "\n"); */
447
perror_plus("gpgme_data_read");
448
plaintext_length = -1;
451
plaintext_length += ret;
455
fprintf_plus(stderr, "Decrypted password is: ");
456
for(ssize_t i = 0; i < plaintext_length; i++){
457
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
459
fprintf(stderr, "\n");
464
/* Delete the GPGME cryptotext data buffer */
465
gpgme_data_release(dh_crypto);
239
467
/* Delete the GPGME plaintext data buffer */
240
468
gpgme_data_release(dh_plain);
241
return new_packet_length;
469
return plaintext_length;
244
static const char * safer_gnutls_strerror (int value) {
245
const char *ret = gnutls_strerror (value);
472
static const char * safer_gnutls_strerror(int value){
473
const char *ret = gnutls_strerror(value);
247
475
ret = "(unknown)";
479
/* GnuTLS log function callback */
251
480
static void debuggnutls(__attribute__((unused)) int level,
252
481
const char* string){
253
fprintf(stderr, "%s", string);
482
fprintf_plus(stderr, "GnuTLS: %s", string);
256
static int initgnutls(encrypted_session *es){
485
static int init_gnutls_global(const char *pubkeyfilename,
486
const char *seckeyfilename){
261
fprintf(stderr, "Initializing GnuTLS\n");
490
fprintf_plus(stderr, "Initializing GnuTLS\n");
264
if ((ret = gnutls_global_init ())
265
!= GNUTLS_E_SUCCESS) {
266
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
493
ret = gnutls_global_init();
494
if(ret != GNUTLS_E_SUCCESS){
495
fprintf_plus(stderr, "GnuTLS global_init: %s\n",
496
safer_gnutls_strerror(ret));
501
/* "Use a log level over 10 to enable all debugging options."
271
504
gnutls_global_set_log_level(11);
272
505
gnutls_global_set_log_function(debuggnutls);
275
/* openpgp credentials */
276
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
277
!= GNUTLS_E_SUCCESS) {
278
fprintf (stderr, "memory error: %s\n",
279
safer_gnutls_strerror(ret));
508
/* OpenPGP credentials */
509
ret = gnutls_certificate_allocate_credentials(&mc.cred);
510
if(ret != GNUTLS_E_SUCCESS){
511
fprintf_plus(stderr, "GnuTLS memory error: %s\n",
512
safer_gnutls_strerror(ret));
513
gnutls_global_deinit();
284
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
285
" and keyfile %s as GnuTLS credentials\n", certfile,
518
fprintf_plus(stderr, "Attempting to use OpenPGP public key %s and"
519
" secret key %s as GnuTLS credentials\n",
289
524
ret = gnutls_certificate_set_openpgp_key_file
290
(es->cred, certfile, certkey, GNUTLS_OPENPGP_FMT_BASE64);
291
if (ret != GNUTLS_E_SUCCESS) {
293
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s',"
295
ret, certfile, certkey);
296
fprintf(stdout, "The Error is: %s\n",
297
safer_gnutls_strerror(ret));
301
//GnuTLS server initialization
302
if ((ret = gnutls_dh_params_init (&es->dh_params))
303
!= GNUTLS_E_SUCCESS) {
304
fprintf (stderr, "Error in dh parameter initialization: %s\n",
305
safer_gnutls_strerror(ret));
309
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
310
!= GNUTLS_E_SUCCESS) {
311
fprintf (stderr, "Error in prime generation: %s\n",
312
safer_gnutls_strerror(ret));
316
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
318
// GnuTLS session creation
319
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
320
!= GNUTLS_E_SUCCESS){
321
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
322
safer_gnutls_strerror(ret));
325
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
326
!= GNUTLS_E_SUCCESS) {
327
fprintf(stderr, "Syntax error at: %s\n", err);
328
fprintf(stderr, "GnuTLS error: %s\n",
329
safer_gnutls_strerror(ret));
333
if ((ret = gnutls_credentials_set
334
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
335
!= GNUTLS_E_SUCCESS) {
336
fprintf(stderr, "Error setting a credentials set: %s\n",
337
safer_gnutls_strerror(ret));
525
(mc.cred, pubkeyfilename, seckeyfilename,
526
GNUTLS_OPENPGP_FMT_BASE64);
527
if(ret != GNUTLS_E_SUCCESS){
529
"Error[%d] while reading the OpenPGP key pair ('%s',"
530
" '%s')\n", ret, pubkeyfilename, seckeyfilename);
531
fprintf_plus(stderr, "The GnuTLS error is: %s\n",
532
safer_gnutls_strerror(ret));
536
/* GnuTLS server initialization */
537
ret = gnutls_dh_params_init(&mc.dh_params);
538
if(ret != GNUTLS_E_SUCCESS){
539
fprintf_plus(stderr, "Error in GnuTLS DH parameter"
540
" initialization: %s\n",
541
safer_gnutls_strerror(ret));
544
ret = gnutls_dh_params_generate2(mc.dh_params, mc.dh_bits);
545
if(ret != GNUTLS_E_SUCCESS){
546
fprintf_plus(stderr, "Error in GnuTLS prime generation: %s\n",
547
safer_gnutls_strerror(ret));
551
gnutls_certificate_set_dh_params(mc.cred, mc.dh_params);
557
gnutls_certificate_free_credentials(mc.cred);
558
gnutls_global_deinit();
559
gnutls_dh_params_deinit(mc.dh_params);
563
static int init_gnutls_session(gnutls_session_t *session){
565
/* GnuTLS session creation */
567
ret = gnutls_init(session, GNUTLS_SERVER);
571
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
572
if(ret != GNUTLS_E_SUCCESS){
574
"Error in GnuTLS session initialization: %s\n",
575
safer_gnutls_strerror(ret));
581
ret = gnutls_priority_set_direct(*session, mc.priority, &err);
583
gnutls_deinit(*session);
586
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
587
if(ret != GNUTLS_E_SUCCESS){
588
fprintf_plus(stderr, "Syntax error at: %s\n", err);
589
fprintf_plus(stderr, "GnuTLS error: %s\n",
590
safer_gnutls_strerror(ret));
591
gnutls_deinit(*session);
597
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
600
gnutls_deinit(*session);
603
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
604
if(ret != GNUTLS_E_SUCCESS){
605
fprintf_plus(stderr, "Error setting GnuTLS credentials: %s\n",
606
safer_gnutls_strerror(ret));
607
gnutls_deinit(*session);
341
611
/* ignore client certificate if any. */
342
gnutls_certificate_server_set_request (es->session,
612
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
345
gnutls_dh_set_prime_bits (es->session, DH_BITS);
614
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
619
/* Avahi log function callback */
350
620
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
351
621
__attribute__((unused)) const char *txt){}
353
static int start_mandos_communication(const char *ip, uint16_t port,
354
AvahiIfIndex if_index){
356
struct sockaddr_in6 to;
357
encrypted_session es;
623
/* Called when a Mandos server is found */
624
static int start_mandos_communication(const char *ip, in_port_t port,
625
AvahiIfIndex if_index,
627
int ret, tcp_sd = -1;
630
struct sockaddr_in in;
631
struct sockaddr_in6 in6;
358
633
char *buffer = NULL;
359
char *decrypted_buffer;
634
char *decrypted_buffer = NULL;
360
635
size_t buffer_length = 0;
361
636
size_t buffer_capacity = 0;
362
ssize_t decrypted_buffer_size;
365
char interface[IF_NAMESIZE];
368
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
372
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
379
if(if_indextoname((unsigned int)if_index, interface) == NULL){
381
perror("if_indextoname");
386
fprintf(stderr, "Binding to interface %s\n", interface);
389
memset(&to,0,sizeof(to)); /* Spurious warning */
390
to.sin6_family = AF_INET6;
391
ret = inet_pton(AF_INET6, ip, &to.sin6_addr);
639
gnutls_session_t session;
640
int pf; /* Protocol family */
657
fprintf_plus(stderr, "Bad address family: %d\n", af);
662
ret = init_gnutls_session(&session);
668
fprintf_plus(stderr, "Setting up a TCP connection to %s, port %"
669
PRIuMAX "\n", ip, (uintmax_t)port);
672
tcp_sd = socket(pf, SOCK_STREAM, 0);
675
perror_plus("socket");
685
memset(&to, 0, sizeof(to));
687
to.in6.sin6_family = (sa_family_t)af;
688
ret = inet_pton(af, ip, &to.in6.sin6_addr);
690
to.in.sin_family = (sa_family_t)af;
691
ret = inet_pton(af, ip, &to.in.sin_addr);
695
perror_plus("inet_pton");
397
fprintf(stderr, "Bad address: %s\n", ip);
400
to.sin6_port = htons(port); /* Spurious warning */
402
to.sin6_scope_id = (uint32_t)if_index;
405
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
406
/* char addrstr[INET6_ADDRSTRLEN]; */
407
/* if(inet_ntop(to.sin6_family, &(to.sin6_addr), addrstr, */
408
/* sizeof(addrstr)) == NULL){ */
409
/* perror("inet_ntop"); */
411
/* fprintf(stderr, "Really connecting to: %s, port %d\n", */
412
/* addrstr, ntohs(to.sin6_port)); */
416
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
422
ret = initgnutls (&es);
428
gnutls_transport_set_ptr (es.session,
429
(gnutls_transport_ptr_t) tcp_sd);
432
fprintf(stderr, "Establishing TLS session with %s\n", ip);
435
ret = gnutls_handshake (es.session);
437
if (ret != GNUTLS_E_SUCCESS){
701
fprintf_plus(stderr, "Bad address: %s\n", ip);
706
to.in6.sin6_port = htons(port);
707
if(IN6_IS_ADDR_LINKLOCAL /* Spurious warnings from */
708
(&to.in6.sin6_addr)){ /* -Wstrict-aliasing=2 or lower and
710
if(if_index == AVAHI_IF_UNSPEC){
711
fprintf_plus(stderr, "An IPv6 link-local address is"
712
" incomplete without a network interface\n");
716
/* Set the network interface number as scope */
717
to.in6.sin6_scope_id = (uint32_t)if_index;
720
to.in.sin_port = htons(port); /* Spurious warnings from
722
-Wunreachable-code */
731
if(af == AF_INET6 and if_index != AVAHI_IF_UNSPEC){
732
char interface[IF_NAMESIZE];
733
if(if_indextoname((unsigned int)if_index, interface) == NULL){
734
perror_plus("if_indextoname");
736
fprintf_plus(stderr, "Connection to: %s%%%s, port %" PRIuMAX
737
"\n", ip, interface, (uintmax_t)port);
740
fprintf_plus(stderr, "Connection to: %s, port %" PRIuMAX "\n",
741
ip, (uintmax_t)port);
743
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
744
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
747
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
750
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
754
perror_plus("inet_ntop");
756
if(strcmp(addrstr, ip) != 0){
757
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
768
ret = connect(tcp_sd, &to.in6, sizeof(to));
770
ret = connect(tcp_sd, &to.in, sizeof(to)); /* IPv4 */
773
if ((errno != ECONNREFUSED and errno != ENETUNREACH) or debug){
775
perror_plus("connect");
786
const char *out = mandos_protocol_version;
789
size_t out_size = strlen(out);
790
ret = (int)TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
791
out_size - written));
794
perror_plus("write");
798
written += (size_t)ret;
799
if(written < out_size){
802
if(out == mandos_protocol_version){
817
fprintf_plus(stderr, "Establishing TLS session with %s\n", ip);
825
/* This casting via intptr_t is to eliminate warning about casting
826
an int to a pointer type. This is exactly how the GnuTLS Guile
827
function "set-session-transport-fd!" does it. */
828
gnutls_transport_set_ptr(session,
829
(gnutls_transport_ptr_t)(intptr_t)tcp_sd);
837
ret = gnutls_handshake(session);
842
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
844
if(ret != GNUTLS_E_SUCCESS){
439
fprintf(stderr, "\n*** Handshake failed ***\n");
846
fprintf_plus(stderr, "*** GnuTLS Handshake failed ***\n");
446
//Retrieve OpenPGP packet that contains the wanted password
853
/* Read OpenPGP packet that contains the wanted password */
449
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
856
fprintf_plus(stderr, "Retrieving OpenPGP encrypted password from"
454
if (buffer_length + BUFFER_SIZE > buffer_capacity){
455
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
460
buffer_capacity += BUFFER_SIZE;
463
ret = gnutls_record_recv
464
(es.session, buffer+buffer_length, BUFFER_SIZE);
867
buffer_capacity = incbuffer(&buffer, buffer_length,
869
if(buffer_capacity == 0){
871
perror_plus("incbuffer");
881
sret = gnutls_record_recv(session, buffer+buffer_length,
470
888
case GNUTLS_E_INTERRUPTED:
471
889
case GNUTLS_E_AGAIN:
473
891
case GNUTLS_E_REHANDSHAKE:
474
ret = gnutls_handshake (es.session);
476
fprintf(stderr, "\n*** Handshake failed ***\n");
893
ret = gnutls_handshake(session);
899
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
901
fprintf_plus(stderr, "*** GnuTLS Re-handshake failed "
483
fprintf(stderr, "Unknown error while reading data from"
484
" encrypted session with mandos server\n");
486
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
909
fprintf_plus(stderr, "Unknown error while reading data from"
910
" encrypted session with Mandos server\n");
911
gnutls_bye(session, GNUTLS_SHUT_RDWR);
490
buffer_length += (size_t) ret;
494
if (buffer_length > 0){
495
decrypted_buffer_size = pgp_packet_decrypt(buffer,
499
if (decrypted_buffer_size >= 0){
916
buffer_length += (size_t) sret;
921
fprintf_plus(stderr, "Closing TLS session\n");
930
ret = gnutls_bye(session, GNUTLS_SHUT_RDWR);
935
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
937
if(buffer_length > 0){
938
ssize_t decrypted_buffer_size;
939
decrypted_buffer_size = pgp_packet_decrypt(buffer, buffer_length,
941
if(decrypted_buffer_size >= 0){
500
944
while(written < (size_t) decrypted_buffer_size){
501
ret = (int)fwrite (decrypted_buffer + written, 1,
502
(size_t)decrypted_buffer_size - written,
950
ret = (int)fwrite(decrypted_buffer + written, 1,
951
(size_t)decrypted_buffer_size - written,
504
953
if(ret == 0 and ferror(stdout)){
506
fprintf(stderr, "Error writing encrypted data: %s\n",
956
fprintf_plus(stderr, "Error writing encrypted data: %s\n",
512
962
written += (size_t)ret;
514
free(decrypted_buffer);
968
/* Shutdown procedure */
973
free(decrypted_buffer);
976
ret = (int)TEMP_FAILURE_RETRY(close(tcp_sd));
982
perror_plus("close");
984
gnutls_deinit(session);
523
fprintf(stderr, "Closing TLS session\n");
527
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
530
gnutls_deinit (es.session);
531
gnutls_certificate_free_credentials (es.cred);
532
gnutls_global_deinit ();
536
static AvahiSimplePoll *simple_poll = NULL;
537
static AvahiServer *server = NULL;
539
static void resolve_callback(
540
AvahiSServiceResolver *r,
541
AvahiIfIndex interface,
542
AVAHI_GCC_UNUSED AvahiProtocol protocol,
543
AvahiResolverEvent event,
547
const char *host_name,
548
const AvahiAddress *address,
550
AVAHI_GCC_UNUSED AvahiStringList *txt,
551
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
552
AVAHI_GCC_UNUSED void* userdata) {
554
assert(r); /* Spurious warning */
994
static void resolve_callback(AvahiSServiceResolver *r,
995
AvahiIfIndex interface,
997
AvahiResolverEvent event,
1001
const char *host_name,
1002
const AvahiAddress *address,
1004
AVAHI_GCC_UNUSED AvahiStringList *txt,
1005
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1007
AVAHI_GCC_UNUSED void* userdata){
556
1012
/* Called whenever a service has been resolved successfully or
561
1021
case AVAHI_RESOLVER_FAILURE:
562
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of"
563
" type '%s' in domain '%s': %s\n", name, type, domain,
564
avahi_strerror(avahi_server_errno(server)));
1022
fprintf_plus(stderr, "(Avahi Resolver) Failed to resolve service "
1023
"'%s' of type '%s' in domain '%s': %s\n", name, type,
1025
avahi_strerror(avahi_server_errno(mc.server)));
567
1028
case AVAHI_RESOLVER_FOUND:
569
1030
char ip[AVAHI_ADDRESS_STR_MAX];
570
1031
avahi_address_snprint(ip, sizeof(ip), address);
572
fprintf(stderr, "Mandos server \"%s\" found on %s (%s) on"
573
" port %d\n", name, host_name, ip, port);
1033
fprintf_plus(stderr, "Mandos server \"%s\" found on %s (%s, %"
1034
PRIdMAX ") on port %" PRIu16 "\n", name,
1035
host_name, ip, (intmax_t)interface, port);
575
int ret = start_mandos_communication(ip, port, interface);
1037
int ret = start_mandos_communication(ip, (in_port_t)port,
1039
avahi_proto_to_af(proto));
1041
avahi_simple_poll_quit(mc.simple_poll);
1043
if(not add_server(ip, (in_port_t)port, interface,
1044
avahi_proto_to_af(proto))){
1045
fprintf_plus(stderr, "Failed to add server \"%s\" to server"
581
1051
avahi_s_service_resolver_free(r);
584
static void browse_callback(
585
AvahiSServiceBrowser *b,
586
AvahiIfIndex interface,
587
AvahiProtocol protocol,
588
AvahiBrowserEvent event,
592
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
595
AvahiServer *s = userdata;
596
assert(b); /* Spurious warning */
598
/* Called whenever a new services becomes available on the LAN or
599
is removed from the LAN */
1054
static void browse_callback(AvahiSServiceBrowser *b,
1055
AvahiIfIndex interface,
1056
AvahiProtocol protocol,
1057
AvahiBrowserEvent event,
1061
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1063
AVAHI_GCC_UNUSED void* userdata){
1068
/* Called whenever a new services becomes available on the LAN or
1069
is removed from the LAN */
1077
case AVAHI_BROWSER_FAILURE:
1079
fprintf_plus(stderr, "(Avahi browser) %s\n",
1080
avahi_strerror(avahi_server_errno(mc.server)));
1081
avahi_simple_poll_quit(mc.simple_poll);
1084
case AVAHI_BROWSER_NEW:
1085
/* We ignore the returned Avahi resolver object. In the callback
1086
function we free it. If the Avahi server is terminated before
1087
the callback function is called the Avahi server will free the
1090
if(avahi_s_service_resolver_new(mc.server, interface, protocol,
1091
name, type, domain, protocol, 0,
1092
resolve_callback, NULL) == NULL)
1093
fprintf_plus(stderr, "Avahi: Failed to resolve service '%s':"
1095
avahi_strerror(avahi_server_errno(mc.server)));
1098
case AVAHI_BROWSER_REMOVE:
1101
case AVAHI_BROWSER_ALL_FOR_NOW:
1102
case AVAHI_BROWSER_CACHE_EXHAUSTED:
1104
fprintf_plus(stderr, "No Mandos server found, still"
1111
/* Signal handler that stops main loop after SIGTERM */
1112
static void handle_sigterm(int sig){
1117
signal_received = sig;
1118
int old_errno = errno;
1119
/* set main loop to exit */
1120
if(mc.simple_poll != NULL){
1121
avahi_simple_poll_quit(mc.simple_poll);
1126
bool get_flags(const char *ifname, struct ifreq *ifr){
1130
int s = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1133
perror_plus("socket");
1137
strcpy(ifr->ifr_name, ifname);
1138
ret = ioctl(s, SIOCGIFFLAGS, ifr);
1142
perror_plus("ioctl SIOCGIFFLAGS");
1150
bool good_flags(const char *ifname, const struct ifreq *ifr){
1152
/* Reject the loopback device */
1153
if(ifr->ifr_flags & IFF_LOOPBACK){
1155
fprintf_plus(stderr, "Rejecting loopback interface \"%s\"\n",
1160
/* Accept point-to-point devices only if connect_to is specified */
1161
if(connect_to != NULL and (ifr->ifr_flags & IFF_POINTOPOINT)){
1163
fprintf_plus(stderr, "Accepting point-to-point interface"
1164
" \"%s\"\n", ifname);
1168
/* Otherwise, reject non-broadcast-capable devices */
1169
if(not (ifr->ifr_flags & IFF_BROADCAST)){
1171
fprintf_plus(stderr, "Rejecting non-broadcast interface"
1172
" \"%s\"\n", ifname);
1176
/* Reject non-ARP interfaces (including dummy interfaces) */
1177
if(ifr->ifr_flags & IFF_NOARP){
1179
fprintf_plus(stderr, "Rejecting non-ARP interface \"%s\"\n",
1185
/* Accept this device */
1187
fprintf_plus(stderr, "Interface \"%s\" is good\n", ifname);
1193
* This function determines if a directory entry in /sys/class/net
1194
* corresponds to an acceptable network device.
1195
* (This function is passed to scandir(3) as a filter function.)
1197
int good_interface(const struct dirent *if_entry){
1198
if(if_entry->d_name[0] == '.'){
1203
if(not get_flags(if_entry->d_name, &ifr)){
1205
fprintf_plus(stderr, "Failed to get flags for interface "
1206
"\"%s\"\n", if_entry->d_name);
1211
if(not good_flags(if_entry->d_name, &ifr)){
1218
* This function determines if a network interface is up.
1220
bool interface_is_up(const char *interface){
1222
if(not get_flags(interface, &ifr)){
1224
fprintf_plus(stderr, "Failed to get flags for interface "
1225
"\"%s\"\n", interface);
1230
return (bool)(ifr.ifr_flags & IFF_UP);
1234
* This function determines if a network interface is running
1236
bool interface_is_running(const char *interface){
1238
if(not get_flags(interface, &ifr)){
1240
fprintf_plus(stderr, "Failed to get flags for interface "
1241
"\"%s\"\n", interface);
1246
return (bool)(ifr.ifr_flags & IFF_RUNNING);
1249
int notdotentries(const struct dirent *direntry){
1250
/* Skip "." and ".." */
1251
if(direntry->d_name[0] == '.'
1252
and (direntry->d_name[1] == '\0'
1253
or (direntry->d_name[1] == '.'
1254
and direntry->d_name[2] == '\0'))){
1260
/* Is this directory entry a runnable program? */
1261
int runnable_hook(const struct dirent *direntry){
1266
if((direntry->d_name)[0] == '\0'){
1271
sret = strspn(direntry->d_name, "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1272
"abcdefghijklmnopqrstuvwxyz"
1275
if((direntry->d_name)[sret] != '\0'){
1276
/* Contains non-allowed characters */
1278
fprintf_plus(stderr, "Ignoring hook \"%s\" with bad name\n",
1284
char *fullname = NULL;
1285
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1287
perror_plus("asprintf");
1291
ret = stat(fullname, &st);
1294
perror_plus("Could not stat hook");
1298
if(not (S_ISREG(st.st_mode))){
1299
/* Not a regular file */
1301
fprintf_plus(stderr, "Ignoring hook \"%s\" - not a file\n",
1306
if(not (st.st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))){
1307
/* Not executable */
1309
fprintf_plus(stderr, "Ignoring hook \"%s\" - not executable\n",
1315
fprintf_plus(stderr, "Hook \"%s\" is acceptable\n",
1321
int avahi_loop_with_timeout(AvahiSimplePoll *s, int retry_interval){
1323
struct timespec now;
1324
struct timespec waited_time;
1325
intmax_t block_time;
1328
if(mc.current_server == NULL){
1330
fprintf_plus(stderr, "Wait until first server is found."
1333
ret = avahi_simple_poll_iterate(s, -1);
1336
fprintf_plus(stderr, "Check current_server if we should run"
1339
/* the current time */
1340
ret = clock_gettime(CLOCK_MONOTONIC, &now);
1342
perror_plus("clock_gettime");
1345
/* Calculating in ms how long time between now and server
1346
who we visted longest time ago. Now - last seen. */
1347
waited_time.tv_sec = (now.tv_sec
1348
- mc.current_server->last_seen.tv_sec);
1349
waited_time.tv_nsec = (now.tv_nsec
1350
- mc.current_server->last_seen.tv_nsec);
1351
/* total time is 10s/10,000ms.
1352
Converting to s from ms by dividing by 1,000,
1353
and ns to ms by dividing by 1,000,000. */
1354
block_time = ((retry_interval
1355
- ((intmax_t)waited_time.tv_sec * 1000))
1356
- ((intmax_t)waited_time.tv_nsec / 1000000));
1359
fprintf_plus(stderr, "Blocking for %" PRIdMAX " ms\n",
1363
if(block_time <= 0){
1364
ret = start_mandos_communication(mc.current_server->ip,
1365
mc.current_server->port,
1366
mc.current_server->if_index,
1367
mc.current_server->af);
1369
avahi_simple_poll_quit(mc.simple_poll);
1372
ret = clock_gettime(CLOCK_MONOTONIC,
1373
&mc.current_server->last_seen);
1375
perror_plus("clock_gettime");
1378
mc.current_server = mc.current_server->next;
1379
block_time = 0; /* Call avahi to find new Mandos
1380
servers, but don't block */
1383
ret = avahi_simple_poll_iterate(s, (int)block_time);
1386
if (ret > 0 or errno != EINTR){
1387
return (ret != 1) ? ret : 0;
1393
/* Set effective uid to 0, return errno */
1394
error_t raise_privileges(void){
1395
error_t old_errno = errno;
1396
error_t ret_errno = 0;
1397
if(seteuid(0) == -1){
1399
perror_plus("seteuid");
1405
/* Set effective and real user ID to 0. Return errno. */
1406
error_t raise_privileges_permanently(void){
1407
error_t old_errno = errno;
1408
error_t ret_errno = raise_privileges();
1413
if(setuid(0) == -1){
1415
perror_plus("seteuid");
1421
/* Set effective user ID to unprivileged saved user ID */
1422
error_t lower_privileges(void){
1423
error_t old_errno = errno;
1424
error_t ret_errno = 0;
1425
if(seteuid(uid) == -1){
1427
perror_plus("seteuid");
1433
/* Lower privileges permanently */
1434
error_t lower_privileges_permanently(void){
1435
error_t old_errno = errno;
1436
error_t ret_errno = 0;
1437
if(setuid(uid) == -1){
1439
perror_plus("setuid");
1445
bool run_network_hooks(const char *mode, const char *interface,
1447
struct dirent **direntries;
1448
struct dirent *direntry;
1450
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1453
if(errno == ENOENT){
1455
fprintf_plus(stderr, "Network hook directory \"%s\" not"
1456
" found\n", hookdir);
1459
perror_plus("scandir");
1462
int devnull = open("/dev/null", O_RDONLY);
1463
for(int i = 0; i < numhooks; i++){
1464
direntry = direntries[i];
1465
char *fullname = NULL;
1466
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1468
perror_plus("asprintf");
1472
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1475
pid_t hook_pid = fork();
1478
/* Raise privileges */
1479
raise_privileges_permanently();
1484
perror_plus("setgid");
1486
/* Reset supplementary groups */
1488
ret = setgroups(0, NULL);
1490
perror_plus("setgroups");
1492
dup2(devnull, STDIN_FILENO);
1494
dup2(STDERR_FILENO, STDOUT_FILENO);
1495
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1497
perror_plus("setenv");
1500
ret = setenv("DEVICE", interface, 1);
1502
perror_plus("setenv");
1505
ret = setenv("VERBOSITY", debug ? "1" : "0", 1);
1507
perror_plus("setenv");
1510
ret = setenv("MODE", mode, 1);
1512
perror_plus("setenv");
1516
ret = asprintf(&delaystring, "%f", delay);
1518
perror_plus("asprintf");
1521
ret = setenv("DELAY", delaystring, 1);
1524
perror_plus("setenv");
1528
if(connect_to != NULL){
1529
ret = setenv("CONNECT", connect_to, 1);
1531
perror_plus("setenv");
1535
if(execl(fullname, direntry->d_name, mode, NULL) == -1){
1536
perror_plus("execl");
1537
_exit(EXIT_FAILURE);
1541
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1542
perror_plus("waitpid");
1546
if(WIFEXITED(status)){
1547
if(WEXITSTATUS(status) != 0){
1548
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1549
" with status %d\n", direntry->d_name,
1550
WEXITSTATUS(status));
1554
} else if(WIFSIGNALED(status)){
1555
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1556
" signal %d\n", direntry->d_name,
1561
fprintf_plus(stderr, "Warning: network hook \"%s\""
1562
" crashed\n", direntry->d_name);
1569
fprintf_plus(stderr, "Network hook \"%s\" ran successfully\n",
1578
error_t bring_up_interface(const char *const interface,
1581
error_t old_errno = errno;
1582
error_t ret_errno = 0;
1583
int ret, ret_setflags;
1584
struct ifreq network;
1585
unsigned int if_index = if_nametoindex(interface);
1587
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1597
if(not interface_is_up(interface)){
1598
if(not get_flags(interface, &network) and debug){
1600
fprintf_plus(stderr, "Failed to get flags for interface "
1601
"\"%s\"\n", interface);
1604
network.ifr_flags |= IFF_UP;
1606
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1609
perror_plus("socket");
1621
fprintf_plus(stderr, "Bringing up interface \"%s\"\n",
1625
/* Raise priviliges */
1629
/* Lower kernel loglevel to KERN_NOTICE to avoid KERN_INFO
1630
messages about the network interface to mess up the prompt */
1631
int ret_linux = klogctl(8, NULL, 5);
1632
bool restore_loglevel = true;
1633
if(ret_linux == -1){
1634
restore_loglevel = false;
1635
perror_plus("klogctl");
1637
#endif /* __linux__ */
1638
ret_setflags = ioctl(sd, SIOCSIFFLAGS, &network);
1641
if(restore_loglevel){
1642
ret_linux = klogctl(7, NULL, 0);
1643
if(ret_linux == -1){
1644
perror_plus("klogctl");
1647
#endif /* __linux__ */
1649
/* Lower privileges */
1652
/* Close the socket */
1653
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1655
perror_plus("close");
1658
if(ret_setflags == -1){
1660
perror_plus("ioctl SIOCSIFFLAGS +IFF_UP");
1665
fprintf_plus(stderr, "Interface \"%s\" is already up; good\n",
1669
/* Sleep checking until interface is running.
1670
Check every 0.25s, up to total time of delay */
1671
for(int i=0; i < delay * 4; i++){
1672
if(interface_is_running(interface)){
1675
struct timespec sleeptime = { .tv_nsec = 250000000 };
1676
ret = nanosleep(&sleeptime, NULL);
1677
if(ret == -1 and errno != EINTR){
1678
perror_plus("nanosleep");
1686
error_t take_down_interface(const char *const interface){
1688
error_t old_errno = errno;
1689
error_t ret_errno = 0;
1690
int ret, ret_setflags;
1691
struct ifreq network;
1692
unsigned int if_index = if_nametoindex(interface);
1694
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1698
if(interface_is_up(interface)){
1699
if(not get_flags(interface, &network) and debug){
1701
fprintf_plus(stderr, "Failed to get flags for interface "
1702
"\"%s\"\n", interface);
1705
network.ifr_flags &= ~(short)IFF_UP; /* clear flag */
1707
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1710
perror_plus("socket");
1716
fprintf_plus(stderr, "Taking down interface \"%s\"\n",
1720
/* Raise priviliges */
1723
ret_setflags = ioctl(sd, SIOCSIFFLAGS, &network);
1726
/* Lower privileges */
1729
/* Close the socket */
1730
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1732
perror_plus("close");
1735
if(ret_setflags == -1){
1737
perror_plus("ioctl SIOCSIFFLAGS -IFF_UP");
1742
fprintf_plus(stderr, "Interface \"%s\" is already down; odd\n",
1750
int main(int argc, char *argv[]){
1751
AvahiSServiceBrowser *sb = NULL;
1756
int exitcode = EXIT_SUCCESS;
1757
char *interfaces = NULL;
1758
size_t interfaces_size = 0;
1759
char *interfaces_to_take_down = NULL;
1760
size_t interfaces_to_take_down_size = 0;
1761
char tempdir[] = "/tmp/mandosXXXXXX";
1762
bool tempdir_created = false;
1763
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
1764
const char *seckey = PATHDIR "/" SECKEY;
1765
const char *pubkey = PATHDIR "/" PUBKEY;
1766
char *interfaces_hooks = NULL;
1767
size_t interfaces_hooks_size = 0;
1769
bool gnutls_initialized = false;
1770
bool gpgme_initialized = false;
1772
double retry_interval = 10; /* 10s between trying a server and
1773
retrying the same server again */
1775
struct sigaction old_sigterm_action = { .sa_handler = SIG_DFL };
1776
struct sigaction sigterm_action = { .sa_handler = handle_sigterm };
1781
/* Lower any group privileges we might have, just to be safe */
1785
perror_plus("setgid");
1788
/* Lower user privileges (temporarily) */
1792
perror_plus("seteuid");
1800
struct argp_option options[] = {
1801
{ .name = "debug", .key = 128,
1802
.doc = "Debug mode", .group = 3 },
1803
{ .name = "connect", .key = 'c',
1804
.arg = "ADDRESS:PORT",
1805
.doc = "Connect directly to a specific Mandos server",
1807
{ .name = "interface", .key = 'i',
1809
.doc = "Network interface that will be used to search for"
1812
{ .name = "seckey", .key = 's',
1814
.doc = "OpenPGP secret key file base name",
1816
{ .name = "pubkey", .key = 'p',
1818
.doc = "OpenPGP public key file base name",
1820
{ .name = "dh-bits", .key = 129,
1822
.doc = "Bit length of the prime number used in the"
1823
" Diffie-Hellman key exchange",
1825
{ .name = "priority", .key = 130,
1827
.doc = "GnuTLS priority string for the TLS handshake",
1829
{ .name = "delay", .key = 131,
1831
.doc = "Maximum delay to wait for interface startup",
1833
{ .name = "retry", .key = 132,
1835
.doc = "Retry interval used when denied by the Mandos server",
1837
{ .name = "network-hook-dir", .key = 133,
1839
.doc = "Directory where network hooks are located",
1842
* These reproduce what we would get without ARGP_NO_HELP
1844
{ .name = "help", .key = '?',
1845
.doc = "Give this help list", .group = -1 },
1846
{ .name = "usage", .key = -3,
1847
.doc = "Give a short usage message", .group = -1 },
1848
{ .name = "version", .key = 'V',
1849
.doc = "Print program version", .group = -1 },
1853
error_t parse_opt(int key, char *arg,
1854
struct argp_state *state){
1857
case 128: /* --debug */
1860
case 'c': /* --connect */
1863
case 'i': /* --interface */
1864
ret_errno = argz_add_sep(&interfaces, &interfaces_size, arg,
1867
argp_error(state, "%s", strerror(ret_errno));
1870
case 's': /* --seckey */
1873
case 'p': /* --pubkey */
1876
case 129: /* --dh-bits */
1878
tmpmax = strtoimax(arg, &tmp, 10);
1879
if(errno != 0 or tmp == arg or *tmp != '\0'
1880
or tmpmax != (typeof(mc.dh_bits))tmpmax){
1881
argp_error(state, "Bad number of DH bits");
1883
mc.dh_bits = (typeof(mc.dh_bits))tmpmax;
1885
case 130: /* --priority */
1888
case 131: /* --delay */
1890
delay = strtof(arg, &tmp);
1891
if(errno != 0 or tmp == arg or *tmp != '\0'){
1892
argp_error(state, "Bad delay");
1894
case 132: /* --retry */
1896
retry_interval = strtod(arg, &tmp);
1897
if(errno != 0 or tmp == arg or *tmp != '\0'
1898
or (retry_interval * 1000) > INT_MAX
1899
or retry_interval < 0){
1900
argp_error(state, "Bad retry interval");
1903
case 133: /* --network-hook-dir */
1907
* These reproduce what we would get without ARGP_NO_HELP
1909
case '?': /* --help */
1910
argp_state_help(state, state->out_stream,
1911
(ARGP_HELP_STD_HELP | ARGP_HELP_EXIT_ERR)
1912
& ~(unsigned int)ARGP_HELP_EXIT_OK);
1913
case -3: /* --usage */
1914
argp_state_help(state, state->out_stream,
1915
ARGP_HELP_USAGE | ARGP_HELP_EXIT_ERR);
1916
case 'V': /* --version */
1917
fprintf_plus(state->out_stream, "%s\n", argp_program_version);
1918
exit(argp_err_exit_status);
1921
return ARGP_ERR_UNKNOWN;
1926
struct argp argp = { .options = options, .parser = parse_opt,
1928
.doc = "Mandos client -- Get and decrypt"
1929
" passwords from a Mandos server" };
1930
ret = argp_parse(&argp, argc, argv,
1931
ARGP_IN_ORDER | ARGP_NO_HELP, 0, NULL);
603
case AVAHI_BROWSER_FAILURE:
605
fprintf(stderr, "(Browser) %s\n",
606
avahi_strerror(avahi_server_errno(server)));
607
avahi_simple_poll_quit(simple_poll);
610
case AVAHI_BROWSER_NEW:
611
/* We ignore the returned resolver object. In the callback
612
function we free it. If the server is terminated before
613
the callback function is called the server will free
614
the resolver for us. */
616
if (!(avahi_s_service_resolver_new(s, interface, protocol, name,
618
AVAHI_PROTO_INET6, 0,
619
resolve_callback, s)))
620
fprintf(stderr, "Failed to resolve service '%s': %s\n", name,
621
avahi_strerror(avahi_server_errno(s)));
624
case AVAHI_BROWSER_REMOVE:
627
case AVAHI_BROWSER_ALL_FOR_NOW:
628
case AVAHI_BROWSER_CACHE_EXHAUSTED:
633
/* Combines file name and path and returns the malloced new
634
string. some sane checks could/should be added */
635
static const char *combinepath(const char *first, const char *second){
636
size_t f_len = strlen(first);
637
size_t s_len = strlen(second);
638
char *tmp = malloc(f_len + s_len + 2);
643
memcpy(tmp, first, f_len);
647
memcpy(tmp + f_len + 1, second, s_len);
649
tmp[f_len + 1 + s_len] = '\0';
654
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
1938
perror_plus("argp_parse");
1939
exitcode = EX_OSERR;
1942
exitcode = EX_USAGE;
1948
/* Work around Debian bug #633582:
1949
<http://bugs.debian.org/633582> */
1951
/* Re-raise priviliges */
1952
if(raise_privileges() == 0){
1955
if(strcmp(seckey, PATHDIR "/" SECKEY) == 0){
1956
int seckey_fd = open(seckey, O_RDONLY);
1957
if(seckey_fd == -1){
1958
perror_plus("open");
1960
ret = (int)TEMP_FAILURE_RETRY(fstat(seckey_fd, &st));
1962
perror_plus("fstat");
1964
if(S_ISREG(st.st_mode)
1965
and st.st_uid == 0 and st.st_gid == 0){
1966
ret = fchown(seckey_fd, uid, gid);
1968
perror_plus("fchown");
1972
TEMP_FAILURE_RETRY(close(seckey_fd));
1976
if(strcmp(pubkey, PATHDIR "/" PUBKEY) == 0){
1977
int pubkey_fd = open(pubkey, O_RDONLY);
1978
if(pubkey_fd == -1){
1979
perror_plus("open");
1981
ret = (int)TEMP_FAILURE_RETRY(fstat(pubkey_fd, &st));
1983
perror_plus("fstat");
1985
if(S_ISREG(st.st_mode)
1986
and st.st_uid == 0 and st.st_gid == 0){
1987
ret = fchown(pubkey_fd, uid, gid);
1989
perror_plus("fchown");
1993
TEMP_FAILURE_RETRY(close(pubkey_fd));
1997
/* Lower privileges */
2001
perror_plus("seteuid");
2006
/* Remove empty interface names */
2008
char *interface = NULL;
2009
while((interface = argz_next(interfaces, interfaces_size,
2011
if(if_nametoindex(interface) == 0){
2012
if(interface[0] != '\0' and strcmp(interface, "none") != 0){
2013
fprintf_plus(stderr, "Not using nonexisting interface"
2014
" \"%s\"\n", interface);
2016
argz_delete(&interfaces, &interfaces_size, interface);
2022
/* Run network hooks */
2025
if(interfaces != NULL){
2026
interfaces_hooks = malloc(interfaces_size);
2027
if(interfaces_hooks == NULL){
2028
perror_plus("malloc");
2031
memcpy(interfaces_hooks, interfaces, interfaces_size);
2032
interfaces_hooks_size = interfaces_size;
2033
argz_stringify(interfaces_hooks, interfaces_hooks_size,
2036
if(not run_network_hooks("start", interfaces_hooks != NULL ?
2037
interfaces_hooks : "", delay)){
2043
avahi_set_log_function(empty_log);
2046
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
2047
from the signal handler */
2048
/* Initialize the pseudo-RNG for Avahi */
2049
srand((unsigned int) time(NULL));
2050
mc.simple_poll = avahi_simple_poll_new();
2051
if(mc.simple_poll == NULL){
2052
fprintf_plus(stderr,
2053
"Avahi: Failed to create simple poll object.\n");
2054
exitcode = EX_UNAVAILABLE;
2058
sigemptyset(&sigterm_action.sa_mask);
2059
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
2061
perror_plus("sigaddset");
2062
exitcode = EX_OSERR;
2065
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
2067
perror_plus("sigaddset");
2068
exitcode = EX_OSERR;
2071
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
2073
perror_plus("sigaddset");
2074
exitcode = EX_OSERR;
2077
/* Need to check if the handler is SIG_IGN before handling:
2078
| [[info:libc:Initial Signal Actions]] |
2079
| [[info:libc:Basic Signal Handling]] |
2081
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
2083
perror_plus("sigaction");
2086
if(old_sigterm_action.sa_handler != SIG_IGN){
2087
ret = sigaction(SIGINT, &sigterm_action, NULL);
2089
perror_plus("sigaction");
2090
exitcode = EX_OSERR;
2094
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
2096
perror_plus("sigaction");
2099
if(old_sigterm_action.sa_handler != SIG_IGN){
2100
ret = sigaction(SIGHUP, &sigterm_action, NULL);
2102
perror_plus("sigaction");
2103
exitcode = EX_OSERR;
2107
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
2109
perror_plus("sigaction");
2112
if(old_sigterm_action.sa_handler != SIG_IGN){
2113
ret = sigaction(SIGTERM, &sigterm_action, NULL);
2115
perror_plus("sigaction");
2116
exitcode = EX_OSERR;
2121
/* If no interfaces were specified, make a list */
2122
if(interfaces == NULL){
2123
struct dirent **direntries;
2124
/* Look for any good interfaces */
2125
ret = scandir(sys_class_net, &direntries, good_interface,
2128
/* Add all found interfaces to interfaces list */
2129
for(int i = 0; i < ret; ++i){
2130
ret_errno = argz_add(&interfaces, &interfaces_size,
2131
direntries[i]->d_name);
2133
perror_plus("argz_add");
2137
fprintf_plus(stderr, "Will use interface \"%s\"\n",
2138
direntries[i]->d_name);
2144
fprintf_plus(stderr, "Could not find a network interface\n");
2145
exitcode = EXIT_FAILURE;
2150
/* If we only got one interface, explicitly use only that one */
2151
if(argz_count(interfaces, interfaces_size) == 1){
2153
fprintf_plus(stderr, "Using only interface \"%s\"\n",
2156
if_index = (AvahiIfIndex)if_nametoindex(interfaces);
2159
/* Bring up interfaces which are down */
2160
if(not (argz_count(interfaces, interfaces_size) == 1
2161
and strcmp(interfaces, "none") == 0)){
2162
char *interface = NULL;
2163
while((interface = argz_next(interfaces, interfaces_size,
2165
bool interface_was_up = interface_is_up(interface);
2166
ret = bring_up_interface(interface, delay);
2167
if(not interface_was_up){
2170
perror_plus("Failed to bring up interface");
2172
ret_errno = argz_add(&interfaces_to_take_down,
2173
&interfaces_to_take_down_size,
2180
interfaces_size = 0;
2181
if(debug and (interfaces_to_take_down == NULL)){
2182
fprintf_plus(stderr, "No interfaces were brought up\n");
2190
ret = init_gnutls_global(pubkey, seckey);
2192
fprintf_plus(stderr, "init_gnutls_global failed\n");
2193
exitcode = EX_UNAVAILABLE;
2196
gnutls_initialized = true;
2203
if(mkdtemp(tempdir) == NULL){
2204
perror_plus("mkdtemp");
2207
tempdir_created = true;
2213
if(not init_gpgme(pubkey, seckey, tempdir)){
2214
fprintf_plus(stderr, "init_gpgme failed\n");
2215
exitcode = EX_UNAVAILABLE;
2218
gpgme_initialized = true;
2225
if(connect_to != NULL){
2226
/* Connect directly, do not use Zeroconf */
2227
/* (Mainly meant for debugging) */
2228
char *address = strrchr(connect_to, ':');
2230
if(address == NULL){
2231
fprintf_plus(stderr, "No colon in address\n");
2232
exitcode = EX_USAGE;
2242
tmpmax = strtoimax(address+1, &tmp, 10);
2243
if(errno != 0 or tmp == address+1 or *tmp != '\0'
2244
or tmpmax != (in_port_t)tmpmax){
2245
fprintf_plus(stderr, "Bad port number\n");
2246
exitcode = EX_USAGE;
2254
port = (in_port_t)tmpmax;
2256
/* Colon in address indicates IPv6 */
2258
if(strchr(connect_to, ':') != NULL){
2260
/* Accept [] around IPv6 address - see RFC 5952 */
2261
if(connect_to[0] == '[' and address[-1] == ']')
2269
address = connect_to;
2275
while(not quit_now){
2276
ret = start_mandos_communication(address, port, if_index, af);
2277
if(quit_now or ret == 0){
2281
fprintf_plus(stderr, "Retrying in %d seconds\n",
2282
(int)retry_interval);
2284
sleep((int)retry_interval);
2288
exitcode = EXIT_SUCCESS;
655
2299
AvahiServerConfig config;
656
AvahiSServiceBrowser *sb = NULL;
659
int returncode = EXIT_SUCCESS;
660
const char *interface = "eth0";
661
struct ifreq network;
663
char *connect_to = NULL;
664
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
667
static struct option long_options[] = {
668
{"debug", no_argument, (int *)&debug, 1},
669
{"connect", required_argument, 0, 'C'},
670
{"interface", required_argument, 0, 'i'},
671
{"certdir", required_argument, 0, 'd'},
672
{"certkey", required_argument, 0, 'c'},
673
{"certfile", required_argument, 0, 'k'},
676
int option_index = 0;
677
ret = getopt_long (argc, argv, "i:", long_options,
707
certfile = combinepath(certdir, certfile);
708
if (certfile == NULL){
709
perror("combinepath");
710
returncode = EXIT_FAILURE;
714
certkey = combinepath(certdir, certkey);
715
if (certkey == NULL){
716
perror("combinepath");
717
returncode = EXIT_FAILURE;
721
if_index = (AvahiIfIndex) if_nametoindex(interface);
723
fprintf(stderr, "No such interface: \"%s\"\n", interface);
727
if(connect_to != NULL){
728
/* Connect directly, do not use Zeroconf */
729
/* (Mainly meant for debugging) */
730
char *address = strrchr(connect_to, ':');
732
fprintf(stderr, "No colon in address\n");
736
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
738
perror("Bad port number");
742
address = connect_to;
743
ret = start_mandos_communication(address, port, if_index);
751
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
754
returncode = EXIT_FAILURE;
757
strcpy(network.ifr_name, interface);
758
ret = ioctl(sd, SIOCGIFFLAGS, &network);
761
perror("ioctl SIOCGIFFLAGS");
762
returncode = EXIT_FAILURE;
765
if((network.ifr_flags & IFF_UP) == 0){
766
network.ifr_flags |= IFF_UP;
767
ret = ioctl(sd, SIOCSIFFLAGS, &network);
769
perror("ioctl SIOCSIFFLAGS");
770
returncode = EXIT_FAILURE;
777
avahi_set_log_function(empty_log);
780
/* Initialize the psuedo-RNG */
781
srand((unsigned int) time(NULL));
783
/* Allocate main loop object */
784
if (!(simple_poll = avahi_simple_poll_new())) {
785
fprintf(stderr, "Failed to create simple poll object.\n");
786
returncode = EXIT_FAILURE;
790
/* Do not publish any local records */
2300
/* Do not publish any local Zeroconf records */
791
2301
avahi_server_config_init(&config);
792
2302
config.publish_hinfo = 0;
793
2303
config.publish_addresses = 0;
794
2304
config.publish_workstation = 0;
795
2305
config.publish_domain = 0;
797
2307
/* Allocate a new server */
798
server = avahi_server_new(avahi_simple_poll_get(simple_poll),
799
&config, NULL, NULL, &error);
801
/* Free the configuration data */
2308
mc.server = avahi_server_new(avahi_simple_poll_get
2309
(mc.simple_poll), &config, NULL,
2312
/* Free the Avahi configuration data */
802
2313
avahi_server_config_free(&config);
804
/* Check if creating the server object succeeded */
806
fprintf(stderr, "Failed to create server: %s\n",
807
avahi_strerror(error));
808
returncode = EXIT_FAILURE;
812
/* Create the service browser */
813
sb = avahi_s_service_browser_new(server, if_index,
815
"_mandos._tcp", NULL, 0,
816
browse_callback, server);
818
fprintf(stderr, "Failed to create service browser: %s\n",
819
avahi_strerror(avahi_server_errno(server)));
820
returncode = EXIT_FAILURE;
824
/* Run the main loop */
827
fprintf(stderr, "Starting avahi loop search\n");
830
avahi_simple_poll_loop(simple_poll);
835
fprintf(stderr, "%s exiting\n", argv[0]);
840
avahi_s_service_browser_free(sb);
843
avahi_server_free(server);
846
avahi_simple_poll_free(simple_poll);
2316
/* Check if creating the Avahi server object succeeded */
2317
if(mc.server == NULL){
2318
fprintf_plus(stderr, "Failed to create Avahi server: %s\n",
2319
avahi_strerror(ret_errno));
2320
exitcode = EX_UNAVAILABLE;
2328
/* Create the Avahi service browser */
2329
sb = avahi_s_service_browser_new(mc.server, if_index,
2330
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
2331
NULL, 0, browse_callback, NULL);
2333
fprintf_plus(stderr, "Failed to create service browser: %s\n",
2334
avahi_strerror(avahi_server_errno(mc.server)));
2335
exitcode = EX_UNAVAILABLE;
2343
/* Run the main loop */
2346
fprintf_plus(stderr, "Starting Avahi loop search\n");
2349
ret = avahi_loop_with_timeout(mc.simple_poll,
2350
(int)(retry_interval * 1000));
2352
fprintf_plus(stderr, "avahi_loop_with_timeout exited %s\n",
2353
(ret == 0) ? "successfully" : "with error");
2359
fprintf_plus(stderr, "%s exiting\n", argv[0]);
2362
/* Cleanup things */
2364
avahi_s_service_browser_free(sb);
2366
if(mc.server != NULL)
2367
avahi_server_free(mc.server);
2369
if(mc.simple_poll != NULL)
2370
avahi_simple_poll_free(mc.simple_poll);
2372
if(gnutls_initialized){
2373
gnutls_certificate_free_credentials(mc.cred);
2374
gnutls_global_deinit();
2375
gnutls_dh_params_deinit(mc.dh_params);
2378
if(gpgme_initialized){
2379
gpgme_release(mc.ctx);
2382
/* Cleans up the circular linked list of Mandos servers the client
2384
if(mc.current_server != NULL){
2385
mc.current_server->prev->next = NULL;
2386
while(mc.current_server != NULL){
2387
server *next = mc.current_server->next;
2388
free(mc.current_server);
2389
mc.current_server = next;
2393
/* Re-raise priviliges */
2397
/* Run network hooks */
2398
run_network_hooks("stop", interfaces_hooks != NULL ?
2399
interfaces_hooks : "", delay);
2401
/* Take down the network interfaces which were brought up */
2403
char *interface = NULL;
2404
while((interface=argz_next(interfaces_to_take_down,
2405
interfaces_to_take_down_size,
2407
ret_errno = take_down_interface(interface);
2410
perror_plus("Failed to take down interface");
2413
if(debug and (interfaces_to_take_down == NULL)){
2414
fprintf_plus(stderr, "No interfaces needed to be taken"
2419
lower_privileges_permanently();
2422
free(interfaces_to_take_down);
2423
free(interfaces_hooks);
2425
/* Removes the GPGME temp directory and all files inside */
2426
if(tempdir_created){
2427
struct dirent **direntries = NULL;
2428
struct dirent *direntry = NULL;
2429
int numentries = scandir(tempdir, &direntries, notdotentries,
2431
if (numentries > 0){
2432
for(int i = 0; i < numentries; i++){
2433
direntry = direntries[i];
2434
char *fullname = NULL;
2435
ret = asprintf(&fullname, "%s/%s", tempdir,
2438
perror_plus("asprintf");
2441
ret = remove(fullname);
2443
fprintf_plus(stderr, "remove(\"%s\"): %s\n", fullname,
2450
/* need to clean even if 0 because man page doesn't specify */
2452
if (numentries == -1){
2453
perror_plus("scandir");
2455
ret = rmdir(tempdir);
2456
if(ret == -1 and errno != ENOENT){
2457
perror_plus("rmdir");
2462
sigemptyset(&old_sigterm_action.sa_mask);
2463
old_sigterm_action.sa_handler = SIG_DFL;
2464
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
2465
&old_sigterm_action,
2468
perror_plus("sigaction");
2471
ret = raise(signal_received);
2472
} while(ret != 0 and errno == EINTR);
2474
perror_plus("raise");
2477
TEMP_FAILURE_RETRY(pause());