46
107
#include <avahi-common/malloc.h>
47
108
#include <avahi-common/error.h>
50
#include <sys/types.h> /* socket(), inet_pton() */
51
#include <sys/socket.h> /* socket(), struct sockaddr_in6,
52
struct in6_addr, inet_pton() */
53
#include <gnutls/gnutls.h> /* All GnuTLS stuff */
54
#include <gnutls/openpgp.h> /* GnuTLS with openpgp stuff */
56
#include <unistd.h> /* close() */
57
#include <netinet/in.h>
58
#include <stdbool.h> /* true */
59
#include <string.h> /* memset */
60
#include <arpa/inet.h> /* inet_pton() */
61
#include <iso646.h> /* not */
64
#include <errno.h> /* perror() */
71
#define CERT_ROOT "/conf/conf.d/cryptkeyreq/"
73
#define CERTFILE CERT_ROOT "openpgp-client.txt"
74
#define KEYFILE CERT_ROOT "openpgp-client-key.txt"
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,
75
127
#define BUFFER_SIZE 256
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
ssize_t pgp_packet_decrypt (char *packet, size_t packet_size,
88
char **new_packet, const char *homedir){
89
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){
93
ssize_t new_packet_capacity = 0;
94
ssize_t new_packet_length = 0;
95
255
gpgme_engine_info_t engine_info;
98
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");
102
299
gpgme_check_version(NULL);
103
gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
300
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
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));
105
/* Set GPGME home directory */
106
rc = gpgme_get_engine_info (&engine_info);
107
if (rc != GPG_ERR_NO_ERROR){
108
fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
109
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));
112
314
while(engine_info != NULL){
113
315
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
114
316
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
115
engine_info->file_name, homedir);
317
engine_info->file_name, tempdir);
118
320
engine_info = engine_info->next;
120
322
if(engine_info == NULL){
121
fprintf(stderr, "Could not set home dir to %s\n", homedir);
125
/* Create new GPGME data buffer from packet buffer */
126
rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
127
if (rc != GPG_ERR_NO_ERROR){
128
fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
129
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));
133
370
/* Create new empty GPGME data buffer for the plaintext */
134
371
rc = gpgme_data_new(&dh_plain);
135
if (rc != GPG_ERR_NO_ERROR){
136
fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
137
gpgme_strsource(rc), gpgme_strerror(rc));
141
/* Create new GPGME "context" */
142
rc = gpgme_new(&ctx);
143
if (rc != GPG_ERR_NO_ERROR){
144
fprintf(stderr, "bad gpgme_new: %s: %s\n",
145
gpgme_strsource(rc), gpgme_strerror(rc));
149
/* Decrypt data from the FILE pointer to the plaintext data
151
rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
152
if (rc != GPG_ERR_NO_ERROR){
153
fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
154
gpgme_strsource(rc), gpgme_strerror(rc));
159
fprintf(stderr, "Decryption of OpenPGP packet succeeded\n");
163
gpgme_decrypt_result_t result;
164
result = gpgme_op_decrypt_result(ctx);
166
fprintf(stderr, "gpgme_op_decrypt_result failed\n");
168
fprintf(stderr, "Unsupported algorithm: %s\n",
169
result->unsupported_algorithm);
170
fprintf(stderr, "Wrong key usage: %d\n",
171
result->wrong_key_usage);
172
if(result->file_name != NULL){
173
fprintf(stderr, "File name: %s\n", result->file_name);
175
gpgme_recipient_t recipient;
176
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;
178
402
while(recipient != NULL){
179
fprintf(stderr, "Public key algorithm: %s\n",
180
gpgme_pubkey_algo_name(recipient->pubkey_algo));
181
fprintf(stderr, "Key ID: %s\n", recipient->keyid);
182
fprintf(stderr, "Secret key available: %s\n",
183
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
185
410
recipient = recipient->next;
191
/* Delete the GPGME FILE pointer cryptotext data buffer */
192
gpgme_data_release(dh_crypto);
418
fprintf_plus(stderr, "Decryption of OpenPGP data succeeded\n");
194
421
/* Seek back to the beginning of the GPGME plaintext data buffer */
195
gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET);
422
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
423
perror_plus("gpgme_data_seek");
424
plaintext_length = -1;
199
if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
200
*new_packet = realloc(*new_packet,
201
(unsigned int)new_packet_capacity
203
if (*new_packet == NULL){
207
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;
210
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length,
439
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
212
441
/* Print the data, if any */
217
perror("gpgme_data_read");
220
new_packet_length += ret;
223
/* FIXME: check characters before printing to screen so to not print
224
terminal control characters */
226
/* fprintf(stderr, "decrypted password is: "); */
227
/* fwrite(*new_packet, 1, new_packet_length, stderr); */
228
/* 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);
231
467
/* Delete the GPGME plaintext data buffer */
232
468
gpgme_data_release(dh_plain);
233
return new_packet_length;
469
return plaintext_length;
236
static const char * safer_gnutls_strerror (int value) {
237
const char *ret = gnutls_strerror (value);
472
static const char * safer_gnutls_strerror(int value){
473
const char *ret = gnutls_strerror(value);
239
475
ret = "(unknown)";
243
void debuggnutls(__attribute__((unused)) int level,
245
fprintf(stderr, "%s", string);
479
/* GnuTLS log function callback */
480
static void debuggnutls(__attribute__((unused)) int level,
482
fprintf_plus(stderr, "GnuTLS: %s", string);
248
int initgnutls(encrypted_session *es){
485
static int init_gnutls_global(const char *pubkeyfilename,
486
const char *seckeyfilename){
253
fprintf(stderr, "Initializing GnuTLS\n");
490
fprintf_plus(stderr, "Initializing GnuTLS\n");
256
if ((ret = gnutls_global_init ())
257
!= GNUTLS_E_SUCCESS) {
258
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."
263
504
gnutls_global_set_log_level(11);
264
505
gnutls_global_set_log_function(debuggnutls);
267
/* openpgp credentials */
268
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
269
!= GNUTLS_E_SUCCESS) {
270
fprintf (stderr, "memory error: %s\n",
271
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();
276
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
277
" 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",
281
524
ret = gnutls_certificate_set_openpgp_key_file
282
(es->cred, CERTFILE, KEYFILE, GNUTLS_OPENPGP_FMT_BASE64);
283
if (ret != GNUTLS_E_SUCCESS) {
285
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s',"
287
ret, CERTFILE, KEYFILE);
288
fprintf(stdout, "The Error is: %s\n",
289
safer_gnutls_strerror(ret));
293
//GnuTLS server initialization
294
if ((ret = gnutls_dh_params_init (&es->dh_params))
295
!= GNUTLS_E_SUCCESS) {
296
fprintf (stderr, "Error in dh parameter initialization: %s\n",
297
safer_gnutls_strerror(ret));
301
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
302
!= GNUTLS_E_SUCCESS) {
303
fprintf (stderr, "Error in prime generation: %s\n",
304
safer_gnutls_strerror(ret));
308
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
310
// GnuTLS session creation
311
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
312
!= GNUTLS_E_SUCCESS){
313
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
314
safer_gnutls_strerror(ret));
317
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
318
!= GNUTLS_E_SUCCESS) {
319
fprintf(stderr, "Syntax error at: %s\n", err);
320
fprintf(stderr, "GnuTLS error: %s\n",
321
safer_gnutls_strerror(ret));
325
if ((ret = gnutls_credentials_set
326
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
327
!= GNUTLS_E_SUCCESS) {
328
fprintf(stderr, "Error setting a credentials set: %s\n",
329
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);
333
611
/* ignore client certificate if any. */
334
gnutls_certificate_server_set_request (es->session,
612
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
337
gnutls_dh_set_prime_bits (es->session, DH_BITS);
614
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
342
void empty_log(__attribute__((unused)) AvahiLogLevel level,
343
__attribute__((unused)) const char *txt){}
619
/* Avahi log function callback */
620
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
621
__attribute__((unused)) const char *txt){}
345
int start_mandos_communication(const char *ip, uint16_t port,
346
AvahiIfIndex if_index){
348
struct sockaddr_in6 to;
349
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;
350
633
char *buffer = NULL;
351
char *decrypted_buffer;
634
char *decrypted_buffer = NULL;
352
635
size_t buffer_length = 0;
353
636
size_t buffer_capacity = 0;
354
ssize_t decrypted_buffer_size;
357
char interface[IF_NAMESIZE];
360
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
364
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
370
if(if_indextoname((unsigned int)if_index, interface) == NULL){
372
perror("if_indextoname");
378
fprintf(stderr, "Binding to interface %s\n", interface);
381
memset(&to,0,sizeof(to)); /* Spurious warning */
382
to.sin6_family = AF_INET6;
383
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");
389
fprintf(stderr, "Bad address: %s\n", ip);
392
to.sin6_port = htons(port); /* Spurious warning */
394
to.sin6_scope_id = (uint32_t)if_index;
397
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
398
/* char addrstr[INET6_ADDRSTRLEN]; */
399
/* if(inet_ntop(to.sin6_family, &(to.sin6_addr), addrstr, */
400
/* sizeof(addrstr)) == NULL){ */
401
/* perror("inet_ntop"); */
403
/* fprintf(stderr, "Really connecting to: %s, port %d\n", */
404
/* addrstr, ntohs(to.sin6_port)); */
408
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
414
ret = initgnutls (&es);
420
gnutls_transport_set_ptr (es.session,
421
(gnutls_transport_ptr_t) tcp_sd);
424
fprintf(stderr, "Establishing TLS session with %s\n", ip);
427
ret = gnutls_handshake (es.session);
429
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){
431
fprintf(stderr, "\n*** Handshake failed ***\n");
846
fprintf_plus(stderr, "*** GnuTLS Handshake failed ***\n");
438
//Retrieve OpenPGP packet that contains the wanted password
853
/* Read OpenPGP packet that contains the wanted password */
441
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
856
fprintf_plus(stderr, "Retrieving OpenPGP encrypted password from"
446
if (buffer_length + BUFFER_SIZE > buffer_capacity){
447
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
452
buffer_capacity += BUFFER_SIZE;
455
ret = gnutls_record_recv
456
(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,
462
888
case GNUTLS_E_INTERRUPTED:
463
889
case GNUTLS_E_AGAIN:
465
891
case GNUTLS_E_REHANDSHAKE:
466
ret = gnutls_handshake (es.session);
468
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 "
475
fprintf(stderr, "Unknown error while reading data from"
476
" encrypted session with mandos server\n");
478
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);
482
buffer_length += (size_t) ret;
486
if (buffer_length > 0){
487
decrypted_buffer_size = pgp_packet_decrypt(buffer,
491
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){
492
944
while(written < (size_t) decrypted_buffer_size){
493
ret = (int)fwrite (decrypted_buffer + written, 1,
494
(size_t)decrypted_buffer_size - written,
950
ret = (int)fwrite(decrypted_buffer + written, 1,
951
(size_t)decrypted_buffer_size - written,
496
953
if(ret == 0 and ferror(stdout)){
498
fprintf(stderr, "Error writing encrypted data: %s\n",
956
fprintf_plus(stderr, "Error writing encrypted data: %s\n",
504
962
written += (size_t)ret;
506
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);
515
fprintf(stderr, "Closing TLS session\n");
519
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
522
gnutls_deinit (es.session);
523
gnutls_certificate_free_credentials (es.cred);
524
gnutls_global_deinit ();
528
static AvahiSimplePoll *simple_poll = NULL;
529
static AvahiServer *server = NULL;
531
static void resolve_callback(
532
AvahiSServiceResolver *r,
533
AvahiIfIndex interface,
534
AVAHI_GCC_UNUSED AvahiProtocol protocol,
535
AvahiResolverEvent event,
539
const char *host_name,
540
const AvahiAddress *address,
542
AVAHI_GCC_UNUSED AvahiStringList *txt,
543
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
544
AVAHI_GCC_UNUSED void* userdata) {
546
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){
548
1012
/* Called whenever a service has been resolved successfully or
553
1021
case AVAHI_RESOLVER_FAILURE:
554
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of"
555
" type '%s' in domain '%s': %s\n", name, type, domain,
556
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)));
559
1028
case AVAHI_RESOLVER_FOUND:
561
1030
char ip[AVAHI_ADDRESS_STR_MAX];
562
1031
avahi_address_snprint(ip, sizeof(ip), address);
564
fprintf(stderr, "Mandos server \"%s\" found on %s (%s) on"
565
" 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);
567
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"
573
1051
avahi_s_service_resolver_free(r);
576
static void browse_callback(
577
AvahiSServiceBrowser *b,
578
AvahiIfIndex interface,
579
AvahiProtocol protocol,
580
AvahiBrowserEvent event,
584
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
587
AvahiServer *s = userdata;
588
assert(b); /* Spurious warning */
590
/* Called whenever a new services becomes available on the LAN or
591
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);
595
case AVAHI_BROWSER_FAILURE:
597
fprintf(stderr, "(Browser) %s\n",
598
avahi_strerror(avahi_server_errno(server)));
599
avahi_simple_poll_quit(simple_poll);
602
case AVAHI_BROWSER_NEW:
603
/* We ignore the returned resolver object. In the callback
604
function we free it. If the server is terminated before
605
the callback function is called the server will free
606
the resolver for us. */
608
if (!(avahi_s_service_resolver_new(s, interface, protocol, name,
610
AVAHI_PROTO_INET6, 0,
611
resolve_callback, s)))
612
fprintf(stderr, "Failed to resolve service '%s': %s\n", name,
613
avahi_strerror(avahi_server_errno(s)));
616
case AVAHI_BROWSER_REMOVE:
619
case AVAHI_BROWSER_ALL_FOR_NOW:
620
case AVAHI_BROWSER_CACHE_EXHAUSTED:
625
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 */
2024
ret_errno = argz_append(&interfaces_hooks, &interfaces_hooks_size,
2025
interfaces, interfaces_size);
2028
perror_plus("argz_append");
2031
argz_stringify(interfaces_hooks, interfaces_hooks_size, (int)',');
2032
if(not run_network_hooks("start", interfaces_hooks, delay)){
2038
avahi_set_log_function(empty_log);
2041
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
2042
from the signal handler */
2043
/* Initialize the pseudo-RNG for Avahi */
2044
srand((unsigned int) time(NULL));
2045
mc.simple_poll = avahi_simple_poll_new();
2046
if(mc.simple_poll == NULL){
2047
fprintf_plus(stderr,
2048
"Avahi: Failed to create simple poll object.\n");
2049
exitcode = EX_UNAVAILABLE;
2053
sigemptyset(&sigterm_action.sa_mask);
2054
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
2056
perror_plus("sigaddset");
2057
exitcode = EX_OSERR;
2060
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
2062
perror_plus("sigaddset");
2063
exitcode = EX_OSERR;
2066
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
2068
perror_plus("sigaddset");
2069
exitcode = EX_OSERR;
2072
/* Need to check if the handler is SIG_IGN before handling:
2073
| [[info:libc:Initial Signal Actions]] |
2074
| [[info:libc:Basic Signal Handling]] |
2076
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
2078
perror_plus("sigaction");
2081
if(old_sigterm_action.sa_handler != SIG_IGN){
2082
ret = sigaction(SIGINT, &sigterm_action, NULL);
2084
perror_plus("sigaction");
2085
exitcode = EX_OSERR;
2089
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
2091
perror_plus("sigaction");
2094
if(old_sigterm_action.sa_handler != SIG_IGN){
2095
ret = sigaction(SIGHUP, &sigterm_action, NULL);
2097
perror_plus("sigaction");
2098
exitcode = EX_OSERR;
2102
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
2104
perror_plus("sigaction");
2107
if(old_sigterm_action.sa_handler != SIG_IGN){
2108
ret = sigaction(SIGTERM, &sigterm_action, NULL);
2110
perror_plus("sigaction");
2111
exitcode = EX_OSERR;
2116
/* If no interfaces were specified, make a list */
2117
if(interfaces == NULL){
2118
struct dirent **direntries;
2119
/* Look for any good interfaces */
2120
ret = scandir(sys_class_net, &direntries, good_interface,
2123
/* Add all found interfaces to interfaces list */
2124
for(int i = 0; i < ret; ++i){
2125
ret_errno = argz_add(&interfaces, &interfaces_size,
2126
direntries[i]->d_name);
2128
perror_plus("argz_add");
2132
fprintf_plus(stderr, "Will use interface \"%s\"\n",
2133
direntries[i]->d_name);
2139
fprintf_plus(stderr, "Could not find a network interface\n");
2140
exitcode = EXIT_FAILURE;
2145
/* If we only got one interface, explicitly use only that one */
2146
if(argz_count(interfaces, interfaces_size) == 1){
2148
fprintf_plus(stderr, "Using only interface \"%s\"\n",
2151
if_index = (AvahiIfIndex)if_nametoindex(interfaces);
2154
/* Bring up interfaces which are down */
2155
if(not (argz_count(interfaces, interfaces_size) == 1
2156
and strcmp(interfaces, "none") == 0)){
2157
char *interface = NULL;
2158
while((interface = argz_next(interfaces, interfaces_size,
2160
bool interface_was_up = interface_is_up(interface);
2161
ret = bring_up_interface(interface, delay);
2162
if(not interface_was_up){
2165
perror_plus("Failed to bring up interface");
2167
ret_errno = argz_add(&interfaces_to_take_down,
2168
&interfaces_to_take_down_size,
2175
interfaces_size = 0;
2176
if(debug and (interfaces_to_take_down == NULL)){
2177
fprintf_plus(stderr, "No interfaces were brought up\n");
2185
ret = init_gnutls_global(pubkey, seckey);
2187
fprintf_plus(stderr, "init_gnutls_global failed\n");
2188
exitcode = EX_UNAVAILABLE;
2191
gnutls_initialized = true;
2198
if(mkdtemp(tempdir) == NULL){
2199
perror_plus("mkdtemp");
2202
tempdir_created = true;
2208
if(not init_gpgme(pubkey, seckey, tempdir)){
2209
fprintf_plus(stderr, "init_gpgme failed\n");
2210
exitcode = EX_UNAVAILABLE;
2213
gpgme_initialized = true;
2220
if(connect_to != NULL){
2221
/* Connect directly, do not use Zeroconf */
2222
/* (Mainly meant for debugging) */
2223
char *address = strrchr(connect_to, ':');
2225
if(address == NULL){
2226
fprintf_plus(stderr, "No colon in address\n");
2227
exitcode = EX_USAGE;
2237
tmpmax = strtoimax(address+1, &tmp, 10);
2238
if(errno != 0 or tmp == address+1 or *tmp != '\0'
2239
or tmpmax != (in_port_t)tmpmax){
2240
fprintf_plus(stderr, "Bad port number\n");
2241
exitcode = EX_USAGE;
2249
port = (in_port_t)tmpmax;
2251
/* Colon in address indicates IPv6 */
2253
if(strchr(connect_to, ':') != NULL){
2255
/* Accept [] around IPv6 address - see RFC 5952 */
2256
if(connect_to[0] == '[' and address[-1] == ']')
2264
address = connect_to;
2270
while(not quit_now){
2271
ret = start_mandos_communication(address, port, if_index, af);
2272
if(quit_now or ret == 0){
2276
fprintf_plus(stderr, "Retrying in %d seconds\n",
2277
(int)retry_interval);
2279
sleep((int)retry_interval);
2283
exitcode = EXIT_SUCCESS;
626
2294
AvahiServerConfig config;
627
AvahiSServiceBrowser *sb = NULL;
630
int returncode = EXIT_SUCCESS;
631
const char *interface = NULL;
632
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
633
char *connect_to = NULL;
636
static struct option long_options[] = {
637
{"debug", no_argument, (int *)&debug, 1},
638
{"connect", required_argument, 0, 'c'},
639
{"interface", required_argument, 0, 'i'},
642
int option_index = 0;
643
ret = getopt_long (argc, argv, "i:", long_options,
664
if(interface != NULL){
665
if_index = (AvahiIfIndex) if_nametoindex(interface);
667
fprintf(stderr, "No such interface: \"%s\"\n", interface);
672
if(connect_to != NULL){
673
/* Connect directly, do not use Zeroconf */
674
/* (Mainly meant for debugging) */
675
char *address = strrchr(connect_to, ':');
677
fprintf(stderr, "No colon in address\n");
681
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
683
perror("Bad port number");
687
address = connect_to;
688
ret = start_mandos_communication(address, port, if_index);
697
avahi_set_log_function(empty_log);
700
/* Initialize the psuedo-RNG */
701
srand((unsigned int) time(NULL));
703
/* Allocate main loop object */
704
if (!(simple_poll = avahi_simple_poll_new())) {
705
fprintf(stderr, "Failed to create simple poll object.\n");
710
/* Do not publish any local records */
2295
/* Do not publish any local Zeroconf records */
711
2296
avahi_server_config_init(&config);
712
2297
config.publish_hinfo = 0;
713
2298
config.publish_addresses = 0;
714
2299
config.publish_workstation = 0;
715
2300
config.publish_domain = 0;
717
2302
/* Allocate a new server */
718
server = avahi_server_new(avahi_simple_poll_get(simple_poll),
719
&config, NULL, NULL, &error);
721
/* Free the configuration data */
2303
mc.server = avahi_server_new(avahi_simple_poll_get
2304
(mc.simple_poll), &config, NULL,
2307
/* Free the Avahi configuration data */
722
2308
avahi_server_config_free(&config);
724
/* Check if creating the server object succeeded */
726
fprintf(stderr, "Failed to create server: %s\n",
727
avahi_strerror(error));
728
returncode = EXIT_FAILURE;
732
/* Create the service browser */
733
sb = avahi_s_service_browser_new(server, if_index,
735
"_mandos._tcp", NULL, 0,
736
browse_callback, server);
738
fprintf(stderr, "Failed to create service browser: %s\n",
739
avahi_strerror(avahi_server_errno(server)));
740
returncode = EXIT_FAILURE;
744
/* Run the main loop */
747
fprintf(stderr, "Starting avahi loop search\n");
750
avahi_simple_poll_loop(simple_poll);
755
fprintf(stderr, "%s exiting\n", argv[0]);
760
avahi_s_service_browser_free(sb);
763
avahi_server_free(server);
766
avahi_simple_poll_free(simple_poll);
2311
/* Check if creating the Avahi server object succeeded */
2312
if(mc.server == NULL){
2313
fprintf_plus(stderr, "Failed to create Avahi server: %s\n",
2314
avahi_strerror(ret_errno));
2315
exitcode = EX_UNAVAILABLE;
2323
/* Create the Avahi service browser */
2324
sb = avahi_s_service_browser_new(mc.server, if_index,
2325
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
2326
NULL, 0, browse_callback, NULL);
2328
fprintf_plus(stderr, "Failed to create service browser: %s\n",
2329
avahi_strerror(avahi_server_errno(mc.server)));
2330
exitcode = EX_UNAVAILABLE;
2338
/* Run the main loop */
2341
fprintf_plus(stderr, "Starting Avahi loop search\n");
2344
ret = avahi_loop_with_timeout(mc.simple_poll,
2345
(int)(retry_interval * 1000));
2347
fprintf_plus(stderr, "avahi_loop_with_timeout exited %s\n",
2348
(ret == 0) ? "successfully" : "with error");
2354
fprintf_plus(stderr, "%s exiting\n", argv[0]);
2357
/* Cleanup things */
2359
avahi_s_service_browser_free(sb);
2361
if(mc.server != NULL)
2362
avahi_server_free(mc.server);
2364
if(mc.simple_poll != NULL)
2365
avahi_simple_poll_free(mc.simple_poll);
2367
if(gnutls_initialized){
2368
gnutls_certificate_free_credentials(mc.cred);
2369
gnutls_global_deinit();
2370
gnutls_dh_params_deinit(mc.dh_params);
2373
if(gpgme_initialized){
2374
gpgme_release(mc.ctx);
2377
/* Cleans up the circular linked list of Mandos servers the client
2379
if(mc.current_server != NULL){
2380
mc.current_server->prev->next = NULL;
2381
while(mc.current_server != NULL){
2382
server *next = mc.current_server->next;
2383
free(mc.current_server);
2384
mc.current_server = next;
2388
/* Re-raise priviliges */
2392
/* Run network hooks */
2393
run_network_hooks("stop", interfaces_hooks, delay);
2395
/* Take down the network interfaces which were brought up */
2397
char *interface = NULL;
2398
while((interface=argz_next(interfaces_to_take_down,
2399
interfaces_to_take_down_size,
2401
ret_errno = take_down_interface(interface);
2404
perror_plus("Failed to take down interface");
2407
if(debug and (interfaces_to_take_down == NULL)){
2408
fprintf_plus(stderr, "No interfaces needed to be taken"
2413
lower_privileges_permanently();
2416
free(interfaces_to_take_down);
2417
free(interfaces_hooks);
2419
/* Removes the GPGME temp directory and all files inside */
2420
if(tempdir_created){
2421
struct dirent **direntries = NULL;
2422
struct dirent *direntry = NULL;
2423
int numentries = scandir(tempdir, &direntries, notdotentries,
2425
if (numentries > 0){
2426
for(int i = 0; i < numentries; i++){
2427
direntry = direntries[i];
2428
char *fullname = NULL;
2429
ret = asprintf(&fullname, "%s/%s", tempdir,
2432
perror_plus("asprintf");
2435
ret = remove(fullname);
2437
fprintf_plus(stderr, "remove(\"%s\"): %s\n", fullname,
2444
/* need to clean even if 0 because man page doesn't specify */
2446
if (numentries == -1){
2447
perror_plus("scandir");
2449
ret = rmdir(tempdir);
2450
if(ret == -1 and errno != ENOENT){
2451
perror_plus("rmdir");
2456
sigemptyset(&old_sigterm_action.sa_mask);
2457
old_sigterm_action.sa_handler = SIG_DFL;
2458
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
2459
&old_sigterm_action,
2462
perror_plus("sigaction");
2465
ret = raise(signal_received);
2466
} while(ret != 0 and errno == EINTR);
2468
perror_plus("raise");
2471
TEMP_FAILURE_RETRY(pause());