103
46
#include <avahi-common/malloc.h>
104
47
#include <avahi-common/error.h>
107
#include <gnutls/gnutls.h> /* All GnuTLS types, constants and
110
init_gnutls_session(),
112
#include <gnutls/openpgp.h>
113
/* gnutls_certificate_set_openpgp_key_file(),
114
GNUTLS_OPENPGP_FMT_BASE64 */
117
#include <gpgme.h> /* All GPGME types, constants and
120
GPGME_PROTOCOL_OpenPGP,
50
#include <sys/types.h> /* socket(), setsockopt(), inet_pton() */
51
#include <sys/socket.h> /* socket(), setsockopt(), struct sockaddr_in6, struct in6_addr, inet_pton() */
52
#include <gnutls/gnutls.h> /* ALL GNUTLS STUFF */
53
#include <gnutls/openpgp.h> /* gnutls with openpgp stuff */
55
#include <unistd.h> /* close() */
56
#include <netinet/in.h>
57
#include <stdbool.h> /* true */
58
#include <string.h> /* memset */
59
#include <arpa/inet.h> /* inet_pton() */
60
#include <iso646.h> /* not */
63
#include <errno.h> /* perror() */
68
#define CERT_ROOT "/conf/conf.d/cryptkeyreq/"
70
#define CERTFILE CERT_ROOT "openpgp-client.txt"
71
#define KEYFILE CERT_ROOT "openpgp-client-key.txt"
123
72
#define BUFFER_SIZE 256
125
#define PATHDIR "/conf/conf.d/mandos"
126
#define SECKEY "seckey.txt"
127
#define PUBKEY "pubkey.txt"
128
#define HOOKDIR "/lib/mandos/network-hooks.d"
131
static const char mandos_protocol_version[] = "1";
132
const char *argp_program_version = "mandos-client " VERSION;
133
const char *argp_program_bug_address = "<mandos@recompile.se>";
134
static const char sys_class_net[] = "/sys/class/net";
135
char *connect_to = NULL;
136
const char *hookdir = HOOKDIR;
138
/* Doubly linked list that need to be circularly linked when used */
139
typedef struct server{
142
AvahiIfIndex if_index;
144
struct timespec last_seen;
149
/* Used for passing in values through the Avahi callback functions */
151
AvahiSimplePoll *simple_poll;
76
gnutls_session_t session;
153
77
gnutls_certificate_credentials_t cred;
154
unsigned int dh_bits;
155
78
gnutls_dh_params_t dh_params;
156
const char *priority;
82
ssize_t gpg_packet_decrypt (char *packet, size_t packet_size, char **new_packet, char *homedir){
83
gpgme_data_t dh_crypto, dh_plain;
158
server *current_server;
161
/* global context so signal handler can reach it*/
162
mandos_context mc = { .simple_poll = NULL, .server = NULL,
163
.dh_bits = 1024, .priority = "SECURE256"
164
":!CTYPE-X.509:+CTYPE-OPENPGP",
165
.current_server = NULL };
167
sig_atomic_t quit_now = 0;
168
int signal_received = 0;
170
/* Function to use when printing errors */
171
void perror_plus(const char *print_text){
172
fprintf(stderr, "Mandos plugin %s: ",
173
program_invocation_short_name);
177
int fprintf_plus(FILE *stream, const char *format, ...){
179
va_start (ap, format);
181
TEMP_FAILURE_RETRY(fprintf(stream, "Mandos plugin %s: ",
182
program_invocation_short_name));
183
return TEMP_FAILURE_RETRY(vfprintf(stream, format, ap));
187
* Make additional room in "buffer" for at least BUFFER_SIZE more
188
* bytes. "buffer_capacity" is how much is currently allocated,
189
* "buffer_length" is how much is already used.
191
size_t incbuffer(char **buffer, size_t buffer_length,
192
size_t buffer_capacity){
193
if(buffer_length + BUFFER_SIZE > buffer_capacity){
194
*buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
198
buffer_capacity += BUFFER_SIZE;
200
return buffer_capacity;
203
/* Add server to set of servers to retry periodically */
204
int add_server(const char *ip, uint16_t port, AvahiIfIndex if_index,
207
server *new_server = malloc(sizeof(server));
208
if(new_server == NULL){
209
perror_plus("malloc");
212
*new_server = (server){ .ip = strdup(ip),
214
.if_index = if_index,
216
if(new_server->ip == NULL){
217
perror_plus("strdup");
220
/* Special case of first server */
221
if (mc.current_server == NULL){
222
new_server->next = new_server;
223
new_server->prev = new_server;
224
mc.current_server = new_server;
225
/* Place the new server last in the list */
227
new_server->next = mc.current_server;
228
new_server->prev = mc.current_server->prev;
229
new_server->prev->next = new_server;
230
mc.current_server->prev = new_server;
232
ret = clock_gettime(CLOCK_MONOTONIC, &mc.current_server->last_seen);
234
perror_plus("clock_gettime");
243
static bool init_gpgme(const char *seckey, const char *pubkey,
244
const char *tempdir){
87
size_t new_packet_capacity = 0;
88
size_t new_packet_length = 0;
246
89
gpgme_engine_info_t engine_info;
250
* Helper function to insert pub and seckey to the engine keyring.
252
bool import_key(const char *filename){
255
gpgme_data_t pgp_data;
257
fd = (int)TEMP_FAILURE_RETRY(open(filename, O_RDONLY));
263
rc = gpgme_data_new_from_fd(&pgp_data, fd);
264
if(rc != GPG_ERR_NO_ERROR){
265
fprintf_plus(stderr, "bad gpgme_data_new_from_fd: %s: %s\n",
266
gpgme_strsource(rc), gpgme_strerror(rc));
270
rc = gpgme_op_import(mc.ctx, pgp_data);
271
if(rc != GPG_ERR_NO_ERROR){
272
fprintf_plus(stderr, "bad gpgme_op_import: %s: %s\n",
273
gpgme_strsource(rc), gpgme_strerror(rc));
277
ret = (int)TEMP_FAILURE_RETRY(close(fd));
279
perror_plus("close");
281
gpgme_data_release(pgp_data);
286
fprintf_plus(stderr, "Initializing GPGME\n");
290
92
gpgme_check_version(NULL);
291
rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
292
if(rc != GPG_ERR_NO_ERROR){
293
fprintf_plus(stderr, "bad gpgme_engine_check_version: %s: %s\n",
294
gpgme_strsource(rc), gpgme_strerror(rc));
93
gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
298
/* Set GPGME home directory for the OpenPGP engine only */
299
rc = gpgme_get_engine_info(&engine_info);
300
if(rc != GPG_ERR_NO_ERROR){
301
fprintf_plus(stderr, "bad gpgme_get_engine_info: %s: %s\n",
302
gpgme_strsource(rc), gpgme_strerror(rc));
95
/* Set GPGME home directory */
96
rc = gpgme_get_engine_info (&engine_info);
97
if (rc != GPG_ERR_NO_ERROR){
98
fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
99
gpgme_strsource(rc), gpgme_strerror(rc));
305
102
while(engine_info != NULL){
306
103
if(engine_info->protocol == GPGME_PROTOCOL_OpenPGP){
307
104
gpgme_set_engine_info(GPGME_PROTOCOL_OpenPGP,
308
engine_info->file_name, tempdir);
105
engine_info->file_name, homedir);
311
108
engine_info = engine_info->next;
313
110
if(engine_info == NULL){
314
fprintf_plus(stderr, "Could not set GPGME home dir to %s\n",
319
/* Create new GPGME "context" */
320
rc = gpgme_new(&(mc.ctx));
321
if(rc != GPG_ERR_NO_ERROR){
322
fprintf_plus(stderr, "Mandos plugin mandos-client: "
323
"bad gpgme_new: %s: %s\n", gpgme_strsource(rc),
328
if(not import_key(pubkey) or not import_key(seckey)){
336
* Decrypt OpenPGP data.
337
* Returns -1 on error
339
static ssize_t pgp_packet_decrypt(const char *cryptotext,
342
gpgme_data_t dh_crypto, dh_plain;
345
size_t plaintext_capacity = 0;
346
ssize_t plaintext_length = 0;
349
fprintf_plus(stderr, "Trying to decrypt OpenPGP data\n");
352
/* Create new GPGME data buffer from memory cryptotext */
353
rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
355
if(rc != GPG_ERR_NO_ERROR){
356
fprintf_plus(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
357
gpgme_strsource(rc), gpgme_strerror(rc));
111
fprintf(stderr, "Could not set home dir to %s\n", homedir);
115
/* Create new GPGME data buffer from packet buffer */
116
rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
117
if (rc != GPG_ERR_NO_ERROR){
118
fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
119
gpgme_strsource(rc), gpgme_strerror(rc));
361
123
/* Create new empty GPGME data buffer for the plaintext */
362
124
rc = gpgme_data_new(&dh_plain);
363
if(rc != GPG_ERR_NO_ERROR){
364
fprintf_plus(stderr, "Mandos plugin mandos-client: "
365
"bad gpgme_data_new: %s: %s\n",
366
gpgme_strsource(rc), gpgme_strerror(rc));
367
gpgme_data_release(dh_crypto);
371
/* Decrypt data from the cryptotext data buffer to the plaintext
373
rc = gpgme_op_decrypt(mc.ctx, dh_crypto, dh_plain);
374
if(rc != GPG_ERR_NO_ERROR){
375
fprintf_plus(stderr, "bad gpgme_op_decrypt: %s: %s\n",
376
gpgme_strsource(rc), gpgme_strerror(rc));
377
plaintext_length = -1;
379
gpgme_decrypt_result_t result;
380
result = gpgme_op_decrypt_result(mc.ctx);
382
fprintf_plus(stderr, "gpgme_op_decrypt_result failed\n");
384
fprintf_plus(stderr, "Unsupported algorithm: %s\n",
385
result->unsupported_algorithm);
386
fprintf_plus(stderr, "Wrong key usage: %u\n",
387
result->wrong_key_usage);
388
if(result->file_name != NULL){
389
fprintf_plus(stderr, "File name: %s\n", result->file_name);
391
gpgme_recipient_t recipient;
392
recipient = result->recipients;
393
while(recipient != NULL){
394
fprintf_plus(stderr, "Public key algorithm: %s\n",
395
gpgme_pubkey_algo_name
396
(recipient->pubkey_algo));
397
fprintf_plus(stderr, "Key ID: %s\n", recipient->keyid);
398
fprintf_plus(stderr, "Secret key available: %s\n",
399
recipient->status == GPG_ERR_NO_SECKEY
401
recipient = recipient->next;
409
fprintf_plus(stderr, "Decryption of OpenPGP data succeeded\n");
125
if (rc != GPG_ERR_NO_ERROR){
126
fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
127
gpgme_strsource(rc), gpgme_strerror(rc));
131
/* Create new GPGME "context" */
132
rc = gpgme_new(&ctx);
133
if (rc != GPG_ERR_NO_ERROR){
134
fprintf(stderr, "bad gpgme_new: %s: %s\n",
135
gpgme_strsource(rc), gpgme_strerror(rc));
139
/* Decrypt data from the FILE pointer to the plaintext data buffer */
140
rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
141
if (rc != GPG_ERR_NO_ERROR){
142
fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
143
gpgme_strsource(rc), gpgme_strerror(rc));
147
/* gpgme_decrypt_result_t result; */
148
/* result = gpgme_op_decrypt_result(ctx); */
149
/* fprintf(stderr, "Unsupported algorithm: %s\n", result->unsupported_algorithm); */
150
/* fprintf(stderr, "Wrong key usage: %d\n", result->wrong_key_usage); */
151
/* if(result->file_name != NULL){ */
152
/* fprintf(stderr, "File name: %s\n", result->file_name); */
154
/* gpgme_recipient_t recipient; */
155
/* recipient = result->recipients; */
157
/* while(recipient != NULL){ */
158
/* fprintf(stderr, "Public key algorithm: %s\n", */
159
/* gpgme_pubkey_algo_name(recipient->pubkey_algo)); */
160
/* fprintf(stderr, "Key ID: %s\n", recipient->keyid); */
161
/* fprintf(stderr, "Secret key available: %s\n", */
162
/* recipient->status == GPG_ERR_NO_SECKEY ? "No" : "Yes"); */
163
/* recipient = recipient->next; */
167
/* Delete the GPGME FILE pointer cryptotext data buffer */
168
gpgme_data_release(dh_crypto);
412
170
/* Seek back to the beginning of the GPGME plaintext data buffer */
413
if(gpgme_data_seek(dh_plain, (off_t)0, SEEK_SET) == -1){
414
perror_plus("gpgme_data_seek");
415
plaintext_length = -1;
171
gpgme_data_seek(dh_plain, 0, SEEK_SET);
421
plaintext_capacity = incbuffer(plaintext,
422
(size_t)plaintext_length,
424
if(plaintext_capacity == 0){
425
perror_plus("incbuffer");
426
plaintext_length = -1;
175
if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
176
*new_packet = realloc(*new_packet, new_packet_capacity + BUFFER_SIZE);
177
if (*new_packet == NULL){
181
new_packet_capacity += BUFFER_SIZE;
430
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
184
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length, BUFFER_SIZE);
432
185
/* Print the data, if any */
187
/* If password is empty, then a incorrect error will be printed */
438
perror_plus("gpgme_data_read");
439
plaintext_length = -1;
442
plaintext_length += ret;
446
fprintf_plus(stderr, "Decrypted password is: ");
447
for(ssize_t i = 0; i < plaintext_length; i++){
448
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
450
fprintf(stderr, "\n");
455
/* Delete the GPGME cryptotext data buffer */
456
gpgme_data_release(dh_crypto);
458
/* Delete the GPGME plaintext data buffer */
191
perror("gpgme_data_read");
194
new_packet_length += ret;
197
/* Delete the GPGME plaintext data buffer */
459
198
gpgme_data_release(dh_plain);
460
return plaintext_length;
199
return new_packet_length;
463
static const char * safer_gnutls_strerror(int value){
464
const char *ret = gnutls_strerror(value); /* Spurious warning from
465
-Wunreachable-code */
202
static const char * safer_gnutls_strerror (int value) {
203
const char *ret = gnutls_strerror (value);
467
205
ret = "(unknown)";
471
/* GnuTLS log function callback */
472
static void debuggnutls(__attribute__((unused)) int level,
474
fprintf_plus(stderr, "GnuTLS: %s", string);
209
void debuggnutls(int level, const char* string){
210
fprintf(stderr, "%s", string);
477
static int init_gnutls_global(const char *pubkeyfilename,
478
const char *seckeyfilename){
213
int initgnutls(encrypted_session *es){
482
fprintf_plus(stderr, "Initializing GnuTLS\n");
485
ret = gnutls_global_init();
486
if(ret != GNUTLS_E_SUCCESS){
487
fprintf_plus(stderr, "GnuTLS global_init: %s\n",
488
safer_gnutls_strerror(ret));
493
/* "Use a log level over 10 to enable all debugging options."
496
gnutls_global_set_log_level(11);
497
gnutls_global_set_log_function(debuggnutls);
500
/* OpenPGP credentials */
501
ret = gnutls_certificate_allocate_credentials(&mc.cred);
502
if(ret != GNUTLS_E_SUCCESS){
503
fprintf_plus(stderr, "GnuTLS memory error: %s\n",
504
safer_gnutls_strerror(ret));
505
gnutls_global_deinit();
510
fprintf_plus(stderr, "Attempting to use OpenPGP public key %s and"
511
" secret key %s as GnuTLS credentials\n",
217
if ((ret = gnutls_global_init ())
218
!= GNUTLS_E_SUCCESS) {
219
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
223
/* Uncomment to enable full debuggin on the gnutls library */
224
/* gnutls_global_set_log_level(11); */
225
/* gnutls_global_set_log_function(debuggnutls); */
228
/* openpgp credentials */
229
if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
230
!= GNUTLS_E_SUCCESS) {
231
fprintf (stderr, "memory error: %s\n", safer_gnutls_strerror(ret));
516
235
ret = gnutls_certificate_set_openpgp_key_file
517
(mc.cred, pubkeyfilename, seckeyfilename,
518
GNUTLS_OPENPGP_FMT_BASE64);
519
if(ret != GNUTLS_E_SUCCESS){
521
"Error[%d] while reading the OpenPGP key pair ('%s',"
522
" '%s')\n", ret, pubkeyfilename, seckeyfilename);
523
fprintf_plus(stderr, "The GnuTLS error is: %s\n",
524
safer_gnutls_strerror(ret));
528
/* GnuTLS server initialization */
529
ret = gnutls_dh_params_init(&mc.dh_params);
530
if(ret != GNUTLS_E_SUCCESS){
531
fprintf_plus(stderr, "Error in GnuTLS DH parameter"
532
" initialization: %s\n",
533
safer_gnutls_strerror(ret));
536
ret = gnutls_dh_params_generate2(mc.dh_params, mc.dh_bits);
537
if(ret != GNUTLS_E_SUCCESS){
538
fprintf_plus(stderr, "Error in GnuTLS prime generation: %s\n",
539
safer_gnutls_strerror(ret));
543
gnutls_certificate_set_dh_params(mc.cred, mc.dh_params);
549
gnutls_certificate_free_credentials(mc.cred);
550
gnutls_global_deinit();
551
gnutls_dh_params_deinit(mc.dh_params);
555
static int init_gnutls_session(gnutls_session_t *session){
557
/* GnuTLS session creation */
559
ret = gnutls_init(session, GNUTLS_SERVER);
563
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
564
if(ret != GNUTLS_E_SUCCESS){
566
"Error in GnuTLS session initialization: %s\n",
567
safer_gnutls_strerror(ret));
573
ret = gnutls_priority_set_direct(*session, mc.priority, &err);
575
gnutls_deinit(*session);
578
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
579
if(ret != GNUTLS_E_SUCCESS){
580
fprintf_plus(stderr, "Syntax error at: %s\n", err);
581
fprintf_plus(stderr, "GnuTLS error: %s\n",
582
safer_gnutls_strerror(ret));
583
gnutls_deinit(*session);
589
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
592
gnutls_deinit(*session);
595
} while(ret == GNUTLS_E_INTERRUPTED or ret == GNUTLS_E_AGAIN);
596
if(ret != GNUTLS_E_SUCCESS){
597
fprintf_plus(stderr, "Error setting GnuTLS credentials: %s\n",
598
safer_gnutls_strerror(ret));
599
gnutls_deinit(*session);
236
(es->cred, CERTFILE, KEYFILE, GNUTLS_OPENPGP_FMT_BASE64);
237
if (ret != GNUTLS_E_SUCCESS) {
239
(stderr, "Error[%d] while reading the OpenPGP key pair ('%s', '%s')\n",
240
ret, CERTFILE, KEYFILE);
241
fprintf(stdout, "The Error is: %s\n",
242
safer_gnutls_strerror(ret));
246
//Gnutls server initialization
247
if ((ret = gnutls_dh_params_init (&es->dh_params))
248
!= GNUTLS_E_SUCCESS) {
249
fprintf (stderr, "Error in dh parameter initialization: %s\n",
250
safer_gnutls_strerror(ret));
254
if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
255
!= GNUTLS_E_SUCCESS) {
256
fprintf (stderr, "Error in prime generation: %s\n",
257
safer_gnutls_strerror(ret));
261
gnutls_certificate_set_dh_params (es->cred, es->dh_params);
263
// Gnutls session creation
264
if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
265
!= GNUTLS_E_SUCCESS){
266
fprintf(stderr, "Error in gnutls session initialization: %s\n",
267
safer_gnutls_strerror(ret));
270
if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
271
!= GNUTLS_E_SUCCESS) {
272
fprintf(stderr, "Syntax error at: %s\n", err);
273
fprintf(stderr, "Gnutls error: %s\n",
274
safer_gnutls_strerror(ret));
278
if ((ret = gnutls_credentials_set
279
(es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
280
!= GNUTLS_E_SUCCESS) {
281
fprintf(stderr, "Error setting a credentials set: %s\n",
282
safer_gnutls_strerror(ret));
603
286
/* ignore client certificate if any. */
604
gnutls_certificate_server_set_request(*session, GNUTLS_CERT_IGNORE);
287
gnutls_certificate_server_set_request (es->session, GNUTLS_CERT_IGNORE);
606
gnutls_dh_set_prime_bits(*session, mc.dh_bits);
289
gnutls_dh_set_prime_bits (es->session, DH_BITS);
611
/* Avahi log function callback */
612
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
613
__attribute__((unused)) const char *txt){}
294
void empty_log(AvahiLogLevel level, const char *txt){}
615
/* Called when a Mandos server is found */
616
static int start_mandos_communication(const char *ip, uint16_t port,
617
AvahiIfIndex if_index,
619
int ret, tcp_sd = -1;
622
struct sockaddr_in in;
623
struct sockaddr_in6 in6;
296
int start_mandos_communcation(char *ip, uint16_t port){
298
struct sockaddr_in6 to;
299
struct in6_addr ip_addr;
300
encrypted_session es;
625
301
char *buffer = NULL;
626
char *decrypted_buffer = NULL;
302
char *decrypted_buffer;
627
303
size_t buffer_length = 0;
628
304
size_t buffer_capacity = 0;
631
gnutls_session_t session;
632
int pf; /* Protocol family */
649
fprintf_plus(stderr, "Bad address family: %d\n", af);
654
ret = init_gnutls_session(&session);
660
fprintf_plus(stderr, "Setting up a TCP connection to %s, port %"
661
PRIu16 "\n", ip, port);
664
tcp_sd = socket(pf, SOCK_STREAM, 0);
667
perror_plus("socket");
677
memset(&to, 0, sizeof(to));
679
to.in6.sin6_family = (sa_family_t)af;
680
ret = inet_pton(af, ip, &to.in6.sin6_addr);
682
to.in.sin_family = (sa_family_t)af;
683
ret = inet_pton(af, ip, &to.in.sin_addr);
687
perror_plus("inet_pton");
305
ssize_t decrypted_buffer_size;
309
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
315
ret = setsockopt(tcp_sd, SOL_SOCKET, SO_BINDTODEVICE, "eth0", 5);
317
perror("setsockopt bindtodevice");
321
memset(&to,0,sizeof(to));
322
to.sin6_family = AF_INET6;
323
ret = inet_pton(AF_INET6, ip, &ip_addr);
693
fprintf_plus(stderr, "Bad address: %s\n", ip);
698
to.in6.sin6_port = htons(port); /* Spurious warnings from
700
-Wunreachable-code */
702
if(IN6_IS_ADDR_LINKLOCAL /* Spurious warnings from */
703
(&to.in6.sin6_addr)){ /* -Wstrict-aliasing=2 or lower and
705
if(if_index == AVAHI_IF_UNSPEC){
706
fprintf_plus(stderr, "An IPv6 link-local address is"
707
" incomplete without a network interface\n");
711
/* Set the network interface number as scope */
712
to.in6.sin6_scope_id = (uint32_t)if_index;
715
to.in.sin_port = htons(port); /* Spurious warnings from
717
-Wunreachable-code */
726
if(af == AF_INET6 and if_index != AVAHI_IF_UNSPEC){
727
char interface[IF_NAMESIZE];
728
if(if_indextoname((unsigned int)if_index, interface) == NULL){
729
perror_plus("if_indextoname");
731
fprintf_plus(stderr, "Connection to: %s%%%s, port %" PRIu16
732
"\n", ip, interface, port);
735
fprintf_plus(stderr, "Connection to: %s, port %" PRIu16 "\n",
738
char addrstr[(INET_ADDRSTRLEN > INET6_ADDRSTRLEN) ?
739
INET_ADDRSTRLEN : INET6_ADDRSTRLEN] = "";
742
pcret = inet_ntop(af, &(to.in6.sin6_addr), addrstr,
745
pcret = inet_ntop(af, &(to.in.sin_addr), addrstr,
749
perror_plus("inet_ntop");
751
if(strcmp(addrstr, ip) != 0){
752
fprintf_plus(stderr, "Canonical address form: %s\n", addrstr);
763
ret = connect(tcp_sd, &to.in6, sizeof(to));
765
ret = connect(tcp_sd, &to.in, sizeof(to)); /* IPv4 */
768
if ((errno != ECONNREFUSED and errno != ENETUNREACH) or debug){
770
perror_plus("connect");
781
const char *out = mandos_protocol_version;
784
size_t out_size = strlen(out);
785
ret = (int)TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
786
out_size - written));
789
perror_plus("write");
793
written += (size_t)ret;
794
if(written < out_size){
797
if(out == mandos_protocol_version){
812
fprintf_plus(stderr, "Establishing TLS session with %s\n", ip);
820
/* Spurious warning from -Wint-to-pointer-cast */
821
gnutls_transport_set_ptr(session, (gnutls_transport_ptr_t) tcp_sd);
829
ret = gnutls_handshake(session);
834
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
836
if(ret != GNUTLS_E_SUCCESS){
838
fprintf_plus(stderr, "*** GnuTLS Handshake failed ***\n");
845
/* Read OpenPGP packet that contains the wanted password */
848
fprintf_plus(stderr, "Retrieving OpenPGP encrypted password from"
859
buffer_capacity = incbuffer(&buffer, buffer_length,
861
if(buffer_capacity == 0){
863
perror_plus("incbuffer");
873
sret = gnutls_record_recv(session, buffer+buffer_length,
329
fprintf(stderr, "Bad address: %s\n", ip);
332
to.sin6_port = htons(port);
333
to.sin6_scope_id = if_nametoindex("eth0");
335
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
341
ret = initgnutls (&es);
348
gnutls_transport_set_ptr (es.session, (gnutls_transport_ptr_t) tcp_sd);
350
ret = gnutls_handshake (es.session);
352
if (ret != GNUTLS_E_SUCCESS){
353
fprintf(stderr, "\n*** Handshake failed ***\n");
361
if (buffer_length + BUFFER_SIZE > buffer_capacity){
362
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
367
buffer_capacity += BUFFER_SIZE;
370
ret = gnutls_record_recv
371
(es.session, buffer+buffer_length, BUFFER_SIZE);
880
377
case GNUTLS_E_INTERRUPTED:
881
378
case GNUTLS_E_AGAIN:
883
380
case GNUTLS_E_REHANDSHAKE:
885
ret = gnutls_handshake(session);
891
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
893
fprintf_plus(stderr, "*** GnuTLS Re-handshake failed "
381
ret = gnutls_handshake (es.session);
383
fprintf(stderr, "\n*** Handshake failed ***\n");
901
fprintf_plus(stderr, "Unknown error while reading data from"
902
" encrypted session with Mandos server\n");
903
gnutls_bye(session, GNUTLS_SHUT_RDWR);
390
fprintf(stderr, "Unknown error while reading data from encrypted session with mandos server\n");
392
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
908
buffer_length += (size_t) sret;
913
fprintf_plus(stderr, "Closing TLS session\n");
922
ret = gnutls_bye(session, GNUTLS_SHUT_RDWR);
927
} while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
929
if(buffer_length > 0){
930
ssize_t decrypted_buffer_size;
931
decrypted_buffer_size = pgp_packet_decrypt(buffer, buffer_length,
933
if(decrypted_buffer_size >= 0){
936
while(written < (size_t) decrypted_buffer_size){
942
ret = (int)fwrite(decrypted_buffer + written, 1,
943
(size_t)decrypted_buffer_size - written,
945
if(ret == 0 and ferror(stdout)){
948
fprintf_plus(stderr, "Error writing encrypted data: %s\n",
954
written += (size_t)ret;
960
/* Shutdown procedure */
965
free(decrypted_buffer);
968
ret = (int)TEMP_FAILURE_RETRY(close(tcp_sd));
974
perror_plus("close");
976
gnutls_deinit(session);
396
buffer_length += ret;
400
if (buffer_length > 0){
401
if ((decrypted_buffer_size = gpg_packet_decrypt(buffer, buffer_length, &decrypted_buffer, CERT_ROOT)) == 0){
404
fwrite (decrypted_buffer, 1, decrypted_buffer_size, stdout);
405
free(decrypted_buffer);
412
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
415
gnutls_deinit (es.session);
416
gnutls_certificate_free_credentials (es.cred);
417
gnutls_global_deinit ();
986
static void resolve_callback(AvahiSServiceResolver *r,
987
AvahiIfIndex interface,
989
AvahiResolverEvent event,
993
const char *host_name,
994
const AvahiAddress *address,
996
AVAHI_GCC_UNUSED AvahiStringList *txt,
997
AVAHI_GCC_UNUSED AvahiLookupResultFlags
999
AVAHI_GCC_UNUSED void* userdata){
1002
/* Called whenever a service has been resolved successfully or
1011
case AVAHI_RESOLVER_FAILURE:
1012
fprintf_plus(stderr, "(Avahi Resolver) Failed to resolve service "
1013
"'%s' of type '%s' in domain '%s': %s\n", name, type,
1015
avahi_strerror(avahi_server_errno(mc.server)));
1018
case AVAHI_RESOLVER_FOUND:
1020
char ip[AVAHI_ADDRESS_STR_MAX];
1021
avahi_address_snprint(ip, sizeof(ip), address);
1023
fprintf_plus(stderr, "Mandos server \"%s\" found on %s (%s, %"
1024
PRIdMAX ") on port %" PRIu16 "\n", name,
1025
host_name, ip, (intmax_t)interface, port);
1027
int ret = start_mandos_communication(ip, port, interface,
1028
avahi_proto_to_af(proto));
1030
avahi_simple_poll_quit(mc.simple_poll);
1032
ret = add_server(ip, port, interface,
1033
avahi_proto_to_af(proto));
1037
avahi_s_service_resolver_free(r);
1040
static void browse_callback(AvahiSServiceBrowser *b,
1041
AvahiIfIndex interface,
1042
AvahiProtocol protocol,
1043
AvahiBrowserEvent event,
1047
AVAHI_GCC_UNUSED AvahiLookupResultFlags
1049
AVAHI_GCC_UNUSED void* userdata){
1052
/* Called whenever a new services becomes available on the LAN or
1053
is removed from the LAN */
1061
case AVAHI_BROWSER_FAILURE:
1063
fprintf_plus(stderr, "(Avahi browser) %s\n",
1064
avahi_strerror(avahi_server_errno(mc.server)));
1065
avahi_simple_poll_quit(mc.simple_poll);
1068
case AVAHI_BROWSER_NEW:
1069
/* We ignore the returned Avahi resolver object. In the callback
1070
function we free it. If the Avahi server is terminated before
1071
the callback function is called the Avahi server will free the
1074
if(avahi_s_service_resolver_new(mc.server, interface, protocol,
1075
name, type, domain, protocol, 0,
1076
resolve_callback, NULL) == NULL)
1077
fprintf_plus(stderr, "Avahi: Failed to resolve service '%s':"
1079
avahi_strerror(avahi_server_errno(mc.server)));
1082
case AVAHI_BROWSER_REMOVE:
1085
case AVAHI_BROWSER_ALL_FOR_NOW:
1086
case AVAHI_BROWSER_CACHE_EXHAUSTED:
1088
fprintf_plus(stderr, "No Mandos server found, still"
1095
/* Signal handler that stops main loop after SIGTERM */
1096
static void handle_sigterm(int sig){
1101
signal_received = sig;
1102
int old_errno = errno;
1103
/* set main loop to exit */
1104
if(mc.simple_poll != NULL){
1105
avahi_simple_poll_quit(mc.simple_poll);
1110
bool get_flags(const char *ifname, struct ifreq *ifr){
1113
int s = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1115
perror_plus("socket");
1118
strcpy(ifr->ifr_name, ifname);
1119
ret = ioctl(s, SIOCGIFFLAGS, ifr);
1122
perror_plus("ioctl SIOCGIFFLAGS");
1129
bool good_flags(const char *ifname, const struct ifreq *ifr){
1131
/* Reject the loopback device */
1132
if(ifr->ifr_flags & IFF_LOOPBACK){
1134
fprintf_plus(stderr, "Rejecting loopback interface \"%s\"\n",
1139
/* Accept point-to-point devices only if connect_to is specified */
1140
if(connect_to != NULL and (ifr->ifr_flags & IFF_POINTOPOINT)){
1142
fprintf_plus(stderr, "Accepting point-to-point interface"
1143
" \"%s\"\n", ifname);
1147
/* Otherwise, reject non-broadcast-capable devices */
1148
if(not (ifr->ifr_flags & IFF_BROADCAST)){
1150
fprintf_plus(stderr, "Rejecting non-broadcast interface"
1151
" \"%s\"\n", ifname);
1155
/* Reject non-ARP interfaces (including dummy interfaces) */
1156
if(ifr->ifr_flags & IFF_NOARP){
1158
fprintf_plus(stderr, "Rejecting non-ARP interface \"%s\"\n",
1164
/* Accept this device */
1166
fprintf_plus(stderr, "Interface \"%s\" is good\n", ifname);
1172
* This function determines if a directory entry in /sys/class/net
1173
* corresponds to an acceptable network device.
1174
* (This function is passed to scandir(3) as a filter function.)
1176
int good_interface(const struct dirent *if_entry){
1177
if(if_entry->d_name[0] == '.'){
1182
if(not get_flags(if_entry->d_name, &ifr)){
1184
fprintf_plus(stderr, "Failed to get flags for interface "
1185
"\"%s\"\n", if_entry->d_name);
1190
if(not good_flags(if_entry->d_name, &ifr)){
1197
* This function determines if a directory entry in /sys/class/net
1198
* corresponds to an acceptable network device which is up.
1199
* (This function is passed to scandir(3) as a filter function.)
1201
int up_interface(const struct dirent *if_entry){
1202
if(if_entry->d_name[0] == '.'){
1207
if(not get_flags(if_entry->d_name, &ifr)){
1209
fprintf_plus(stderr, "Failed to get flags for interface "
1210
"\"%s\"\n", if_entry->d_name);
1215
/* Reject down interfaces */
1216
if(not (ifr.ifr_flags & IFF_UP)){
1218
fprintf_plus(stderr, "Rejecting down interface \"%s\"\n",
1224
/* Reject non-running interfaces */
1225
if(not (ifr.ifr_flags & IFF_RUNNING)){
1227
fprintf_plus(stderr, "Rejecting non-running interface \"%s\"\n",
1233
if(not good_flags(if_entry->d_name, &ifr)){
1239
int notdotentries(const struct dirent *direntry){
1240
/* Skip "." and ".." */
1241
if(direntry->d_name[0] == '.'
1242
and (direntry->d_name[1] == '\0'
1243
or (direntry->d_name[1] == '.'
1244
and direntry->d_name[2] == '\0'))){
1250
/* Is this directory entry a runnable program? */
1251
int runnable_hook(const struct dirent *direntry){
1256
if((direntry->d_name)[0] == '\0'){
1261
sret = strspn(direntry->d_name, "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1262
"abcdefghijklmnopqrstuvwxyz"
1265
if((direntry->d_name)[sret] != '\0'){
1266
/* Contains non-allowed characters */
1268
fprintf_plus(stderr, "Ignoring hook \"%s\" with bad name\n",
1274
char *fullname = NULL;
1275
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1277
perror_plus("asprintf");
1281
ret = stat(fullname, &st);
1284
perror_plus("Could not stat hook");
1288
if(not (S_ISREG(st.st_mode))){
1289
/* Not a regular file */
1291
fprintf_plus(stderr, "Ignoring hook \"%s\" - not a file\n",
1296
if(not (st.st_mode & (S_IXUSR | S_IXGRP | S_IXOTH))){
1297
/* Not executable */
1299
fprintf_plus(stderr, "Ignoring hook \"%s\" - not executable\n",
1305
fprintf_plus(stderr, "Hook \"%s\" is acceptable\n",
1311
int avahi_loop_with_timeout(AvahiSimplePoll *s, int retry_interval){
1313
struct timespec now;
1314
struct timespec waited_time;
1315
intmax_t block_time;
1318
if(mc.current_server == NULL){
1320
fprintf_plus(stderr, "Wait until first server is found."
1323
ret = avahi_simple_poll_iterate(s, -1);
1326
fprintf_plus(stderr, "Check current_server if we should run"
1329
/* the current time */
1330
ret = clock_gettime(CLOCK_MONOTONIC, &now);
1332
perror_plus("clock_gettime");
1335
/* Calculating in ms how long time between now and server
1336
who we visted longest time ago. Now - last seen. */
1337
waited_time.tv_sec = (now.tv_sec
1338
- mc.current_server->last_seen.tv_sec);
1339
waited_time.tv_nsec = (now.tv_nsec
1340
- mc.current_server->last_seen.tv_nsec);
1341
/* total time is 10s/10,000ms.
1342
Converting to s from ms by dividing by 1,000,
1343
and ns to ms by dividing by 1,000,000. */
1344
block_time = ((retry_interval
1345
- ((intmax_t)waited_time.tv_sec * 1000))
1346
- ((intmax_t)waited_time.tv_nsec / 1000000));
1349
fprintf_plus(stderr, "Blocking for %" PRIdMAX " ms\n",
1353
if(block_time <= 0){
1354
ret = start_mandos_communication(mc.current_server->ip,
1355
mc.current_server->port,
1356
mc.current_server->if_index,
1357
mc.current_server->af);
1359
avahi_simple_poll_quit(mc.simple_poll);
1362
ret = clock_gettime(CLOCK_MONOTONIC,
1363
&mc.current_server->last_seen);
1365
perror_plus("clock_gettime");
1368
mc.current_server = mc.current_server->next;
1369
block_time = 0; /* Call avahi to find new Mandos
1370
servers, but don't block */
1373
ret = avahi_simple_poll_iterate(s, (int)block_time);
1376
if (ret > 0 or errno != EINTR){
1377
return (ret != 1) ? ret : 0;
1383
bool run_network_hooks(const char *mode, const char *interface,
1385
struct dirent **direntries;
1386
struct dirent *direntry;
1388
int numhooks = scandir(hookdir, &direntries, runnable_hook,
1391
perror_plus("scandir");
1393
int devnull = open("/dev/null", O_RDONLY);
1394
for(int i = 0; i < numhooks; i++){
1395
direntry = direntries[i];
1396
char *fullname = NULL;
1397
ret = asprintf(&fullname, "%s/%s", hookdir, direntry->d_name);
1399
perror_plus("asprintf");
1403
fprintf_plus(stderr, "Running network hook \"%s\"\n",
1406
pid_t hook_pid = fork();
1409
dup2(devnull, STDIN_FILENO);
1411
dup2(STDERR_FILENO, STDOUT_FILENO);
1412
ret = setenv("MANDOSNETHOOKDIR", hookdir, 1);
1414
perror_plus("setenv");
1417
ret = setenv("DEVICE", interface, 1);
1419
perror_plus("setenv");
1422
ret = setenv("VERBOSE", debug ? "1" : "0", 1);
1424
perror_plus("setenv");
1427
ret = setenv("MODE", mode, 1);
1429
perror_plus("setenv");
1433
ret = asprintf(&delaystring, "%f", delay);
1435
perror_plus("asprintf");
1438
ret = setenv("DELAY", delaystring, 1);
1441
perror_plus("setenv");
1445
ret = execl(fullname, direntry->d_name, mode, NULL);
1446
perror_plus("execl");
1449
if(TEMP_FAILURE_RETRY(waitpid(hook_pid, &status, 0)) == -1){
1450
perror_plus("waitpid");
1454
if(WIFEXITED(status)){
1455
if(WEXITSTATUS(status) != 0){
1456
fprintf_plus(stderr, "Warning: network hook \"%s\" exited"
1457
" with status %d\n", direntry->d_name,
1458
WEXITSTATUS(status));
1462
} else if(WIFSIGNALED(status)){
1463
fprintf_plus(stderr, "Warning: network hook \"%s\" died by"
1464
" signal %d\n", direntry->d_name,
1469
fprintf_plus(stderr, "Warning: network hook \"%s\""
1470
" crashed\n", direntry->d_name);
1477
fprintf_plus(stderr, "Network hook \"%s\" ran successfully\n",
1486
int main(int argc, char *argv[]){
1487
AvahiSServiceBrowser *sb = NULL;
1492
int exitcode = EXIT_SUCCESS;
1493
const char *interface = "";
1494
struct ifreq network;
1496
bool take_down_interface = false;
1499
char tempdir[] = "/tmp/mandosXXXXXX";
1500
bool tempdir_created = false;
1501
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
1502
const char *seckey = PATHDIR "/" SECKEY;
1503
const char *pubkey = PATHDIR "/" PUBKEY;
1505
bool gnutls_initialized = false;
1506
bool gpgme_initialized = false;
1508
double retry_interval = 10; /* 10s between trying a server and
1509
retrying the same server again */
1511
struct sigaction old_sigterm_action = { .sa_handler = SIG_DFL };
1512
struct sigaction sigterm_action = { .sa_handler = handle_sigterm };
1517
/* Lower any group privileges we might have, just to be safe */
1521
perror_plus("setgid");
1524
/* Lower user privileges (temporarily) */
1528
perror_plus("seteuid");
1536
struct argp_option options[] = {
1537
{ .name = "debug", .key = 128,
1538
.doc = "Debug mode", .group = 3 },
1539
{ .name = "connect", .key = 'c',
1540
.arg = "ADDRESS:PORT",
1541
.doc = "Connect directly to a specific Mandos server",
1543
{ .name = "interface", .key = 'i',
1545
.doc = "Network interface that will be used to search for"
1548
{ .name = "seckey", .key = 's',
1550
.doc = "OpenPGP secret key file base name",
1552
{ .name = "pubkey", .key = 'p',
1554
.doc = "OpenPGP public key file base name",
1556
{ .name = "dh-bits", .key = 129,
1558
.doc = "Bit length of the prime number used in the"
1559
" Diffie-Hellman key exchange",
1561
{ .name = "priority", .key = 130,
1563
.doc = "GnuTLS priority string for the TLS handshake",
1565
{ .name = "delay", .key = 131,
1567
.doc = "Maximum delay to wait for interface startup",
1569
{ .name = "retry", .key = 132,
1571
.doc = "Retry interval used when denied by the mandos server",
1573
{ .name = "network-hook-dir", .key = 133,
1575
.doc = "Directory where network hooks are located",
1578
* These reproduce what we would get without ARGP_NO_HELP
1580
{ .name = "help", .key = '?',
1581
.doc = "Give this help list", .group = -1 },
1582
{ .name = "usage", .key = -3,
1583
.doc = "Give a short usage message", .group = -1 },
1584
{ .name = "version", .key = 'V',
1585
.doc = "Print program version", .group = -1 },
1589
error_t parse_opt(int key, char *arg,
1590
struct argp_state *state){
1593
case 128: /* --debug */
1596
case 'c': /* --connect */
1599
case 'i': /* --interface */
1602
case 's': /* --seckey */
1605
case 'p': /* --pubkey */
1608
case 129: /* --dh-bits */
1610
tmpmax = strtoimax(arg, &tmp, 10);
1611
if(errno != 0 or tmp == arg or *tmp != '\0'
1612
or tmpmax != (typeof(mc.dh_bits))tmpmax){
1613
argp_error(state, "Bad number of DH bits");
1615
mc.dh_bits = (typeof(mc.dh_bits))tmpmax;
1617
case 130: /* --priority */
1620
case 131: /* --delay */
1622
delay = strtof(arg, &tmp);
1623
if(errno != 0 or tmp == arg or *tmp != '\0'){
1624
argp_error(state, "Bad delay");
1626
case 132: /* --retry */
1628
retry_interval = strtod(arg, &tmp);
1629
if(errno != 0 or tmp == arg or *tmp != '\0'
1630
or (retry_interval * 1000) > INT_MAX
1631
or retry_interval < 0){
1632
argp_error(state, "Bad retry interval");
1635
case 133: /* --network-hook-dir */
1639
* These reproduce what we would get without ARGP_NO_HELP
1641
case '?': /* --help */
1642
argp_state_help(state, state->out_stream,
1643
(ARGP_HELP_STD_HELP | ARGP_HELP_EXIT_ERR)
1644
& ~(unsigned int)ARGP_HELP_EXIT_OK);
1645
case -3: /* --usage */
1646
argp_state_help(state, state->out_stream,
1647
ARGP_HELP_USAGE | ARGP_HELP_EXIT_ERR);
1648
case 'V': /* --version */
1649
fprintf_plus(state->out_stream,
1650
"Mandos plugin mandos-client: ");
1651
fprintf_plus(state->out_stream, "%s\n", argp_program_version);
1652
exit(argp_err_exit_status);
1655
return ARGP_ERR_UNKNOWN;
1660
struct argp argp = { .options = options, .parser = parse_opt,
1662
.doc = "Mandos client -- Get and decrypt"
1663
" passwords from a Mandos server" };
1664
ret = argp_parse(&argp, argc, argv,
1665
ARGP_IN_ORDER | ARGP_NO_HELP, 0, NULL);
1672
perror_plus("argp_parse");
1673
exitcode = EX_OSERR;
1676
exitcode = EX_USAGE;
1682
/* Work around Debian bug #633582:
1683
<http://bugs.debian.org/633582> */
1686
/* Re-raise priviliges */
1690
perror_plus("seteuid");
1693
if(strcmp(seckey, PATHDIR "/" SECKEY) == 0){
1694
int seckey_fd = open(seckey, O_RDONLY);
1695
if(seckey_fd == -1){
1696
perror_plus("open");
1698
ret = (int)TEMP_FAILURE_RETRY(fstat(seckey_fd, &st));
1700
perror_plus("fstat");
1702
if(S_ISREG(st.st_mode)
1703
and st.st_uid == 0 and st.st_gid == 0){
1704
ret = fchown(seckey_fd, uid, gid);
1706
perror_plus("fchown");
1710
TEMP_FAILURE_RETRY(close(seckey_fd));
1714
if(strcmp(pubkey, PATHDIR "/" PUBKEY) == 0){
1715
int pubkey_fd = open(pubkey, O_RDONLY);
1716
if(pubkey_fd == -1){
1717
perror_plus("open");
1719
ret = (int)TEMP_FAILURE_RETRY(fstat(pubkey_fd, &st));
1721
perror_plus("fstat");
1723
if(S_ISREG(st.st_mode)
1724
and st.st_uid == 0 and st.st_gid == 0){
1725
ret = fchown(pubkey_fd, uid, gid);
1727
perror_plus("fchown");
1731
TEMP_FAILURE_RETRY(close(pubkey_fd));
1735
/* Lower privileges */
1739
perror_plus("seteuid");
1743
/* Run network hooks */
1746
/* Re-raise priviliges */
1750
perror_plus("seteuid");
1753
if(not run_network_hooks("start", interface, delay)){
1757
/* Lower privileges */
1761
perror_plus("seteuid");
421
static AvahiSimplePoll *simple_poll = NULL;
422
static AvahiServer *server = NULL;
424
static void resolve_callback(
425
AvahiSServiceResolver *r,
426
AVAHI_GCC_UNUSED AvahiIfIndex interface,
427
AVAHI_GCC_UNUSED AvahiProtocol protocol,
428
AvahiResolverEvent event,
432
const char *host_name,
433
const AvahiAddress *address,
435
AvahiStringList *txt,
436
AvahiLookupResultFlags flags,
437
AVAHI_GCC_UNUSED void* userdata) {
441
/* Called whenever a service has been resolved successfully or timed out */
444
case AVAHI_RESOLVER_FAILURE:
445
fprintf(stderr, "(Resolver) Failed to resolve service '%s' of type '%s' in domain '%s': %s\n", name, type, domain, avahi_strerror(avahi_server_errno(server)));
448
case AVAHI_RESOLVER_FOUND: {
449
char ip[AVAHI_ADDRESS_STR_MAX];
450
avahi_address_snprint(ip, sizeof(ip), address);
451
int ret = start_mandos_communcation(ip, port);
459
avahi_s_service_resolver_free(r);
462
static void browse_callback(
463
AvahiSServiceBrowser *b,
464
AvahiIfIndex interface,
465
AvahiProtocol protocol,
466
AvahiBrowserEvent event,
470
AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
473
AvahiServer *s = userdata;
476
/* Called whenever a new services becomes available on the LAN or is removed from the LAN */
480
case AVAHI_BROWSER_FAILURE:
482
fprintf(stderr, "(Browser) %s\n", avahi_strerror(avahi_server_errno(server)));
483
avahi_simple_poll_quit(simple_poll);
486
case AVAHI_BROWSER_NEW:
487
/* We ignore the returned resolver object. In the callback
488
function we free it. If the server is terminated before
489
the callback function is called the server will free
490
the resolver for us. */
492
if (!(avahi_s_service_resolver_new(s, interface, protocol, name, type, domain, AVAHI_PROTO_INET6, 0, resolve_callback, s)))
493
fprintf(stderr, "Failed to resolve service '%s': %s\n", name, avahi_strerror(avahi_server_errno(s)));
497
case AVAHI_BROWSER_REMOVE:
500
case AVAHI_BROWSER_ALL_FOR_NOW:
501
case AVAHI_BROWSER_CACHE_EXHAUSTED:
506
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
507
AvahiServerConfig config;
508
AvahiSServiceBrowser *sb = NULL;
1767
512
avahi_set_log_function(empty_log);
1770
if(interface[0] == '\0'){
1771
struct dirent **direntries;
1772
/* First look for interfaces that are up */
1773
ret = scandir(sys_class_net, &direntries, up_interface,
1776
/* No up interfaces, look for any good interfaces */
1778
ret = scandir(sys_class_net, &direntries, good_interface,
1782
/* Pick the first interface returned */
1783
interface = strdup(direntries[0]->d_name);
1785
fprintf_plus(stderr, "Using interface \"%s\"\n", interface);
1787
if(interface == NULL){
1788
perror_plus("malloc");
1790
exitcode = EXIT_FAILURE;
1796
fprintf_plus(stderr, "Could not find a network interface\n");
1797
exitcode = EXIT_FAILURE;
1802
/* Initialize Avahi early so avahi_simple_poll_quit() can be called
1803
from the signal handler */
1804
/* Initialize the pseudo-RNG for Avahi */
1805
srand((unsigned int) time(NULL));
1806
mc.simple_poll = avahi_simple_poll_new();
1807
if(mc.simple_poll == NULL){
1808
fprintf_plus(stderr,
1809
"Avahi: Failed to create simple poll object.\n");
1810
exitcode = EX_UNAVAILABLE;
1814
sigemptyset(&sigterm_action.sa_mask);
1815
ret = sigaddset(&sigterm_action.sa_mask, SIGINT);
1817
perror_plus("sigaddset");
1818
exitcode = EX_OSERR;
1821
ret = sigaddset(&sigterm_action.sa_mask, SIGHUP);
1823
perror_plus("sigaddset");
1824
exitcode = EX_OSERR;
1827
ret = sigaddset(&sigterm_action.sa_mask, SIGTERM);
1829
perror_plus("sigaddset");
1830
exitcode = EX_OSERR;
1833
/* Need to check if the handler is SIG_IGN before handling:
1834
| [[info:libc:Initial Signal Actions]] |
1835
| [[info:libc:Basic Signal Handling]] |
1837
ret = sigaction(SIGINT, NULL, &old_sigterm_action);
1839
perror_plus("sigaction");
1842
if(old_sigterm_action.sa_handler != SIG_IGN){
1843
ret = sigaction(SIGINT, &sigterm_action, NULL);
1845
perror_plus("sigaction");
1846
exitcode = EX_OSERR;
1850
ret = sigaction(SIGHUP, NULL, &old_sigterm_action);
1852
perror_plus("sigaction");
1855
if(old_sigterm_action.sa_handler != SIG_IGN){
1856
ret = sigaction(SIGHUP, &sigterm_action, NULL);
1858
perror_plus("sigaction");
1859
exitcode = EX_OSERR;
1863
ret = sigaction(SIGTERM, NULL, &old_sigterm_action);
1865
perror_plus("sigaction");
1868
if(old_sigterm_action.sa_handler != SIG_IGN){
1869
ret = sigaction(SIGTERM, &sigterm_action, NULL);
1871
perror_plus("sigaction");
1872
exitcode = EX_OSERR;
1877
/* If the interface is down, bring it up */
1878
if(strcmp(interface, "none") != 0){
1879
if_index = (AvahiIfIndex) if_nametoindex(interface);
1881
fprintf_plus(stderr, "No such interface: \"%s\"\n", interface);
1882
exitcode = EX_UNAVAILABLE;
1890
/* Re-raise priviliges */
1894
perror_plus("seteuid");
1898
/* Lower kernel loglevel to KERN_NOTICE to avoid KERN_INFO
1899
messages about the network interface to mess up the prompt */
1900
ret = klogctl(8, NULL, 5);
1901
bool restore_loglevel = true;
1903
restore_loglevel = false;
1904
perror_plus("klogctl");
1906
#endif /* __linux__ */
1908
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
1910
perror_plus("socket");
1911
exitcode = EX_OSERR;
1913
if(restore_loglevel){
1914
ret = klogctl(7, NULL, 0);
1916
perror_plus("klogctl");
1919
#endif /* __linux__ */
1920
/* Lower privileges */
1924
perror_plus("seteuid");
1928
strcpy(network.ifr_name, interface);
1929
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1931
perror_plus("ioctl SIOCGIFFLAGS");
1933
if(restore_loglevel){
1934
ret = klogctl(7, NULL, 0);
1936
perror_plus("klogctl");
1939
#endif /* __linux__ */
1940
exitcode = EX_OSERR;
1941
/* Lower privileges */
1945
perror_plus("seteuid");
1949
if((network.ifr_flags & IFF_UP) == 0){
1950
network.ifr_flags |= IFF_UP;
1951
take_down_interface = true;
1952
ret = ioctl(sd, SIOCSIFFLAGS, &network);
1954
take_down_interface = false;
1955
perror_plus("ioctl SIOCSIFFLAGS +IFF_UP");
1956
exitcode = EX_OSERR;
1958
if(restore_loglevel){
1959
ret = klogctl(7, NULL, 0);
1961
perror_plus("klogctl");
1964
#endif /* __linux__ */
1965
/* Lower privileges */
1969
perror_plus("seteuid");
1974
/* Sleep checking until interface is running.
1975
Check every 0.25s, up to total time of delay */
1976
for(int i=0; i < delay * 4; i++){
1977
ret = ioctl(sd, SIOCGIFFLAGS, &network);
1979
perror_plus("ioctl SIOCGIFFLAGS");
1980
} else if(network.ifr_flags & IFF_RUNNING){
1983
struct timespec sleeptime = { .tv_nsec = 250000000 };
1984
ret = nanosleep(&sleeptime, NULL);
1985
if(ret == -1 and errno != EINTR){
1986
perror_plus("nanosleep");
1989
if(not take_down_interface){
1990
/* We won't need the socket anymore */
1991
ret = (int)TEMP_FAILURE_RETRY(close(sd));
1993
perror_plus("close");
1997
if(restore_loglevel){
1998
/* Restores kernel loglevel to default */
1999
ret = klogctl(7, NULL, 0);
2001
perror_plus("klogctl");
2004
#endif /* __linux__ */
2005
/* Lower privileges */
2007
/* Lower privileges */
2010
perror_plus("seteuid");
2018
ret = init_gnutls_global(pubkey, seckey);
2020
fprintf_plus(stderr, "init_gnutls_global failed\n");
2021
exitcode = EX_UNAVAILABLE;
2024
gnutls_initialized = true;
2031
if(mkdtemp(tempdir) == NULL){
2032
perror_plus("mkdtemp");
2035
tempdir_created = true;
2041
if(not init_gpgme(pubkey, seckey, tempdir)){
2042
fprintf_plus(stderr, "init_gpgme failed\n");
2043
exitcode = EX_UNAVAILABLE;
2046
gpgme_initialized = true;
2053
if(connect_to != NULL){
2054
/* Connect directly, do not use Zeroconf */
2055
/* (Mainly meant for debugging) */
2056
char *address = strrchr(connect_to, ':');
2057
if(address == NULL){
2058
fprintf_plus(stderr, "No colon in address\n");
2059
exitcode = EX_USAGE;
2069
tmpmax = strtoimax(address+1, &tmp, 10);
2070
if(errno != 0 or tmp == address+1 or *tmp != '\0'
2071
or tmpmax != (uint16_t)tmpmax){
2072
fprintf_plus(stderr, "Bad port number\n");
2073
exitcode = EX_USAGE;
2081
port = (uint16_t)tmpmax;
2083
/* Colon in address indicates IPv6 */
2085
if(strchr(connect_to, ':') != NULL){
2087
/* Accept [] around IPv6 address - see RFC 5952 */
2088
if(connect_to[0] == '[' and address[-1] == ']')
2096
address = connect_to;
2102
while(not quit_now){
2103
ret = start_mandos_communication(address, port, if_index, af);
2104
if(quit_now or ret == 0){
2108
fprintf_plus(stderr, "Retrying in %d seconds\n",
2109
(int)retry_interval);
2111
sleep((int)retry_interval);
2115
exitcode = EXIT_SUCCESS;
2126
AvahiServerConfig config;
2127
/* Do not publish any local Zeroconf records */
514
/* Initialize the psuedo-RNG */
517
/* Allocate main loop object */
518
if (!(simple_poll = avahi_simple_poll_new())) {
519
fprintf(stderr, "Failed to create simple poll object.\n");
523
/* Do not publish any local records */
2128
524
avahi_server_config_init(&config);
2129
525
config.publish_hinfo = 0;
2130
526
config.publish_addresses = 0;
2131
527
config.publish_workstation = 0;
2132
528
config.publish_domain = 0;
530
/* /\* Set a unicast DNS server for wide area DNS-SD *\/ */
531
/* avahi_address_parse("193.11.177.11", AVAHI_PROTO_UNSPEC, &config.wide_area_servers[0]); */
532
/* config.n_wide_area_servers = 1; */
533
/* config.enable_wide_area = 1; */
2134
535
/* Allocate a new server */
2135
mc.server = avahi_server_new(avahi_simple_poll_get
2136
(mc.simple_poll), &config, NULL,
2139
/* Free the Avahi configuration data */
536
server = avahi_server_new(avahi_simple_poll_get(simple_poll), &config, NULL, NULL, &error);
538
/* Free the configuration data */
2140
539
avahi_server_config_free(&config);
2143
/* Check if creating the Avahi server object succeeded */
2144
if(mc.server == NULL){
2145
fprintf_plus(stderr, "Failed to create Avahi server: %s\n",
2146
avahi_strerror(error));
2147
exitcode = EX_UNAVAILABLE;
2155
/* Create the Avahi service browser */
2156
sb = avahi_s_service_browser_new(mc.server, if_index,
2157
AVAHI_PROTO_UNSPEC, "_mandos._tcp",
2158
NULL, 0, browse_callback, NULL);
2160
fprintf_plus(stderr, "Failed to create service browser: %s\n",
2161
avahi_strerror(avahi_server_errno(mc.server)));
2162
exitcode = EX_UNAVAILABLE;
2170
/* Run the main loop */
2173
fprintf_plus(stderr, "Starting Avahi loop search\n");
2176
ret = avahi_loop_with_timeout(mc.simple_poll,
2177
(int)(retry_interval * 1000));
2179
fprintf_plus(stderr, "avahi_loop_with_timeout exited %s\n",
2180
(ret == 0) ? "successfully" : "with error");
2186
fprintf_plus(stderr, "%s exiting\n", argv[0]);
2189
/* Cleanup things */
2191
avahi_s_service_browser_free(sb);
2193
if(mc.server != NULL)
2194
avahi_server_free(mc.server);
2196
if(mc.simple_poll != NULL)
2197
avahi_simple_poll_free(mc.simple_poll);
2199
if(gnutls_initialized){
2200
gnutls_certificate_free_credentials(mc.cred);
2201
gnutls_global_deinit();
2202
gnutls_dh_params_deinit(mc.dh_params);
2205
if(gpgme_initialized){
2206
gpgme_release(mc.ctx);
2209
/* Cleans up the circular linked list of Mandos servers the client
2211
if(mc.current_server != NULL){
2212
mc.current_server->prev->next = NULL;
2213
while(mc.current_server != NULL){
2214
server *next = mc.current_server->next;
2215
free(mc.current_server);
2216
mc.current_server = next;
2220
/* Re-raise priviliges */
2226
perror_plus("seteuid");
2230
/* Run network hooks */
2231
run_network_hooks("stop", interface, delay);
2233
/* Take down the network interface */
2234
if(take_down_interface and geteuid() == 0){
2235
ret = ioctl(sd, SIOCGIFFLAGS, &network);
2237
perror_plus("ioctl SIOCGIFFLAGS");
2238
} else if(network.ifr_flags & IFF_UP){
2239
network.ifr_flags &= ~(short)IFF_UP; /* clear flag */
2240
ret = ioctl(sd, SIOCSIFFLAGS, &network);
2242
perror_plus("ioctl SIOCSIFFLAGS -IFF_UP");
2245
ret = (int)TEMP_FAILURE_RETRY(close(sd));
2247
perror_plus("close");
2252
/* Lower privileges permanently */
2256
perror_plus("setuid");
2260
/* Removes the GPGME temp directory and all files inside */
2261
if(tempdir_created){
2262
struct dirent **direntries = NULL;
2263
struct dirent *direntry = NULL;
2264
int numentries = scandir(tempdir, &direntries, notdotentries,
2266
if (numentries > 0){
2267
for(int i = 0; i < numentries; i++){
2268
direntry = direntries[i];
2269
char *fullname = NULL;
2270
ret = asprintf(&fullname, "%s/%s", tempdir,
2273
perror_plus("asprintf");
2276
ret = remove(fullname);
2278
fprintf_plus(stderr, "remove(\"%s\"): %s\n", fullname,
2285
/* need to clean even if 0 because man page doesn't specify */
2287
if (numentries == -1){
2288
perror_plus("scandir");
2290
ret = rmdir(tempdir);
2291
if(ret == -1 and errno != ENOENT){
2292
perror_plus("rmdir");
2297
sigemptyset(&old_sigterm_action.sa_mask);
2298
old_sigterm_action.sa_handler = SIG_DFL;
2299
ret = (int)TEMP_FAILURE_RETRY(sigaction(signal_received,
2300
&old_sigterm_action,
2303
perror_plus("sigaction");
2306
ret = raise(signal_received);
2307
} while(ret != 0 and errno == EINTR);
2309
perror_plus("raise");
2312
TEMP_FAILURE_RETRY(pause());
541
/* Check wether creating the server object succeeded */
543
fprintf(stderr, "Failed to create server: %s\n", avahi_strerror(error));
547
/* Create the service browser */
548
if (!(sb = avahi_s_service_browser_new(server, if_nametoindex("eth0"), AVAHI_PROTO_INET6, "_mandos._tcp", NULL, 0, browse_callback, server))) {
549
fprintf(stderr, "Failed to create service browser: %s\n", avahi_strerror(avahi_server_errno(server)));
553
/* Run the main loop */
554
avahi_simple_poll_loop(simple_poll);
562
avahi_s_service_browser_free(sb);
565
avahi_server_free(server);
568
avahi_simple_poll_free(simple_poll);