1
/* -*- coding: utf-8 -*- */
3
* Mandos client - get and decrypt data from a Mandos server
4
* This file demonstrates how to use Avahi's core API, this is
5
* the embeddable mDNS stack for embedded applications.
5
* This program is partly derived from an example program for an Avahi
6
* service browser, downloaded from
7
* <http://avahi.org/browser/examples/core-browse-services.c>. This
8
* includes the following functions: "resolve_callback",
9
* "browse_callback", and parts of "main".
12
* Copyright © 2007-2008 Teddy Hogeborn & Björn Påhlsson
14
* This program is free software: you can redistribute it and/or
15
* modify it under the terms of the GNU General Public License as
16
* published by the Free Software Foundation, either version 3 of the
17
* License, or (at your option) any later version.
19
* This program is distributed in the hope that it will be useful, but
20
* WITHOUT ANY WARRANTY; without even the implied warranty of
21
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
22
* General Public License for more details.
24
* You should have received a copy of the GNU General Public License
25
* along with this program. If not, see
26
* <http://www.gnu.org/licenses/>.
28
* Contact the authors at <mandos@fukt.bsnet.se>.
7
* End user applications should *not* use this API and should use
8
* the D-Bus or C APIs, please see
9
* client-browse-services.c and glib-integration.c
11
* I repeat, you probably do *not* want to use this example.
31
/* Needed by GPGME, specifically gpgme_data_seek() */
15
This file is part of avahi.
17
avahi is free software; you can redistribute it and/or modify it
18
under the terms of the GNU Lesser General Public License as
19
published by the Free Software Foundation; either version 2.1 of the
20
License, or (at your option) any later version.
22
avahi is distributed in the hope that it will be useful, but WITHOUT
23
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
24
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
25
Public License for more details.
27
You should have received a copy of the GNU Lesser General Public
28
License along with avahi; if not, write to the Free Software
29
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
32
33
#define _LARGEFILE_SOURCE
33
34
#define _FILE_OFFSET_BITS 64
35
#define _GNU_SOURCE /* TEMP_FAILURE_RETRY() */
38
37
#include <assert.h>
39
38
#include <stdlib.h>
41
40
#include <net/if.h> /* if_nametoindex */
42
#include <sys/ioctl.h> /* ioctl, ifreq, SIOCGIFFLAGS, IFF_UP,
44
#include <net/if.h> /* ioctl, ifreq, SIOCGIFFLAGS, IFF_UP,
47
42
#include <avahi-core/core.h>
48
43
#include <avahi-core/lookup.h>
174
133
if (rc != GPG_ERR_NO_ERROR){
175
134
fprintf(stderr, "bad gpgme_new: %s: %s\n",
176
135
gpgme_strsource(rc), gpgme_strerror(rc));
177
plaintext_length = -1;
181
/* Decrypt data from the cryptotext data buffer to the plaintext
139
/* Decrypt data from the FILE pointer to the plaintext data buffer */
183
140
rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
184
141
if (rc != GPG_ERR_NO_ERROR){
185
142
fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
186
143
gpgme_strsource(rc), gpgme_strerror(rc));
187
plaintext_length = -1;
192
fprintf(stderr, "Decryption of OpenPGP data succeeded\n");
196
gpgme_decrypt_result_t result;
197
result = gpgme_op_decrypt_result(ctx);
199
fprintf(stderr, "gpgme_op_decrypt_result failed\n");
201
fprintf(stderr, "Unsupported algorithm: %s\n",
202
result->unsupported_algorithm);
203
fprintf(stderr, "Wrong key usage: %d\n",
204
result->wrong_key_usage);
205
if(result->file_name != NULL){
206
fprintf(stderr, "File name: %s\n", result->file_name);
208
gpgme_recipient_t recipient;
209
recipient = result->recipients;
211
while(recipient != NULL){
212
fprintf(stderr, "Public key algorithm: %s\n",
213
gpgme_pubkey_algo_name(recipient->pubkey_algo));
214
fprintf(stderr, "Key ID: %s\n", recipient->keyid);
215
fprintf(stderr, "Secret key available: %s\n",
216
recipient->status == GPG_ERR_NO_SECKEY
218
recipient = recipient->next;
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);
224
170
/* Seek back to the beginning of the GPGME plaintext data buffer */
225
if (gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET) == -1){
226
perror("pgpme_data_seek");
227
plaintext_length = -1;
171
gpgme_data_seek(dh_plain, 0, SEEK_SET);
233
plaintext_capacity = adjustbuffer(plaintext, (size_t)plaintext_length,
235
if (plaintext_capacity == 0){
236
perror("adjustbuffer");
237
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;
241
ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
184
ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length, BUFFER_SIZE);
243
185
/* Print the data, if any */
187
/* If password is empty, then a incorrect error will be printed */
249
191
perror("gpgme_data_read");
250
plaintext_length = -1;
253
plaintext_length += ret;
194
new_packet_length += ret;
257
fprintf(stderr, "Decrypted password is: ");
258
for(ssize_t i = 0; i < plaintext_length; i++){
259
fprintf(stderr, "%02hhX ", (*plaintext)[i]);
261
fprintf(stderr, "\n");
266
/* Delete the GPGME cryptotext data buffer */
267
gpgme_data_release(dh_crypto);
269
/* Delete the GPGME plaintext data buffer */
197
/* Delete the GPGME plaintext data buffer */
270
198
gpgme_data_release(dh_plain);
271
return plaintext_length;
199
return new_packet_length;
274
202
static const char * safer_gnutls_strerror (int value) {
281
/* GnuTLS log function callback */
282
static void debuggnutls(__attribute__((unused)) int level,
284
fprintf(stderr, "GnuTLS: %s", string);
209
void debuggnutls(int level, const char* string){
210
fprintf(stderr, "%s", string);
287
static int init_gnutls_global(mandos_context *mc,
288
const char *pubkeyfile,
289
const char *seckeyfile){
213
int initgnutls(encrypted_session *es){
293
fprintf(stderr, "Initializing GnuTLS\n");
296
217
if ((ret = gnutls_global_init ())
297
218
!= GNUTLS_E_SUCCESS) {
298
fprintf (stderr, "GnuTLS global_init: %s\n",
299
safer_gnutls_strerror(ret));
219
fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
304
/* "Use a log level over 10 to enable all debugging options."
307
gnutls_global_set_log_level(11);
308
gnutls_global_set_log_function(debuggnutls);
311
/* OpenPGP credentials */
312
if ((ret = gnutls_certificate_allocate_credentials (&mc->cred))
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))
313
230
!= GNUTLS_E_SUCCESS) {
314
fprintf (stderr, "GnuTLS memory error: %s\n",
315
safer_gnutls_strerror(ret));
231
fprintf (stderr, "memory error: %s\n", safer_gnutls_strerror(ret));
320
fprintf(stderr, "Attempting to use OpenPGP certificate %s"
321
" and keyfile %s as GnuTLS credentials\n", pubkeyfile,
325
235
ret = gnutls_certificate_set_openpgp_key_file
326
(mc->cred, pubkeyfile, seckeyfile, GNUTLS_OPENPGP_FMT_BASE64);
327
if (ret != GNUTLS_E_SUCCESS) {
329
"Error[%d] while reading the OpenPGP key pair ('%s',"
330
" '%s')\n", ret, pubkeyfile, seckeyfile);
331
fprintf(stdout, "The GnuTLS error is: %s\n",
332
safer_gnutls_strerror(ret));
336
/* GnuTLS server initialization */
337
ret = gnutls_dh_params_init(&mc->dh_params);
338
if (ret != GNUTLS_E_SUCCESS) {
339
fprintf (stderr, "Error in GnuTLS DH parameter initialization:"
340
" %s\n", safer_gnutls_strerror(ret));
343
ret = gnutls_dh_params_generate2(mc->dh_params, mc->dh_bits);
344
if (ret != GNUTLS_E_SUCCESS) {
345
fprintf (stderr, "Error in GnuTLS prime generation: %s\n",
346
safer_gnutls_strerror(ret));
350
gnutls_certificate_set_dh_params(mc->cred, mc->dh_params);
355
static int init_gnutls_session(mandos_context *mc, gnutls_session_t *session){
357
/* GnuTLS session creation */
358
ret = gnutls_init(session, GNUTLS_SERVER);
359
if (ret != GNUTLS_E_SUCCESS){
360
fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
361
safer_gnutls_strerror(ret));
366
ret = gnutls_priority_set_direct(*session, mc->priority, &err);
367
if (ret != GNUTLS_E_SUCCESS) {
368
fprintf(stderr, "Syntax error at: %s\n", err);
369
fprintf(stderr, "GnuTLS error: %s\n",
370
safer_gnutls_strerror(ret));
375
ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
377
if (ret != GNUTLS_E_SUCCESS) {
378
fprintf(stderr, "Error setting GnuTLS credentials: %s\n",
379
safer_gnutls_strerror(ret));
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));
383
286
/* ignore client certificate if any. */
384
gnutls_certificate_server_set_request (*session,
287
gnutls_certificate_server_set_request (es->session, GNUTLS_CERT_IGNORE);
387
gnutls_dh_set_prime_bits (*session, mc->dh_bits);
289
gnutls_dh_set_prime_bits (es->session, DH_BITS);
392
/* Avahi log function callback */
393
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
394
__attribute__((unused)) const char *txt){}
294
void empty_log(AvahiLogLevel level, const char *txt){}
396
/* Called when a Mandos server is found */
397
static int start_mandos_communication(const char *ip, uint16_t port,
398
AvahiIfIndex if_index,
296
int start_mandos_communcation(char *ip, uint16_t port){
401
298
struct sockaddr_in6 to;
299
struct in6_addr ip_addr;
300
encrypted_session es;
402
301
char *buffer = NULL;
403
302
char *decrypted_buffer;
404
303
size_t buffer_length = 0;
405
304
size_t buffer_capacity = 0;
406
305
ssize_t decrypted_buffer_size;
409
char interface[IF_NAMESIZE];
410
gnutls_session_t session;
411
gnutls_dh_params_t dh_params;
413
ret = init_gnutls_session (mc, &session);
419
fprintf(stderr, "Setting up a tcp connection to %s, port %d\n",
423
309
tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
425
311
perror("socket");
430
if(if_indextoname((unsigned int)if_index, interface) == NULL){
431
perror("if_indextoname");
434
fprintf(stderr, "Binding to interface %s\n", interface);
315
ret = setsockopt(tcp_sd, SOL_SOCKET, SO_BINDTODEVICE, "eth0", 5);
317
perror("setsockopt bindtodevice");
437
memset(&to,0,sizeof(to)); /* Spurious warning */
321
memset(&to,0,sizeof(to));
438
322
to.sin6_family = AF_INET6;
439
/* It would be nice to have a way to detect if we were passed an
440
IPv4 address here. Now we assume an IPv6 address. */
441
ret = inet_pton(AF_INET6, ip, &to.sin6_addr);
323
ret = inet_pton(AF_INET6, ip, &ip_addr);
443
325
perror("inet_pton");
447
329
fprintf(stderr, "Bad address: %s\n", ip);
450
to.sin6_port = htons(port); /* Spurious warning */
452
to.sin6_scope_id = (uint32_t)if_index;
455
fprintf(stderr, "Connection to: %s, port %d\n", ip, port);
456
char addrstr[INET6_ADDRSTRLEN] = "";
457
if(inet_ntop(to.sin6_family, &(to.sin6_addr), addrstr,
458
sizeof(addrstr)) == NULL){
461
if(strcmp(addrstr, ip) != 0){
462
fprintf(stderr, "Canonical address form: %s\n", addrstr);
332
to.sin6_port = htons(port);
333
to.sin6_scope_id = if_nametoindex("eth0");
467
335
ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
469
337
perror("connect");
341
ret = initgnutls (&es);
348
gnutls_transport_set_ptr (es.session, (gnutls_transport_ptr_t) tcp_sd);
473
const char *out = mandos_protocol_version;
476
size_t out_size = strlen(out);
477
ret = TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
478
out_size - written));
484
written += (size_t)ret;
485
if(written < out_size){
488
if (out == mandos_protocol_version){
498
fprintf(stderr, "Establishing TLS session with %s\n", ip);
501
gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) tcp_sd);
503
ret = gnutls_handshake (session);
350
ret = gnutls_handshake (es.session);
505
352
if (ret != GNUTLS_E_SUCCESS){
507
fprintf(stderr, "*** GnuTLS Handshake failed ***\n");
353
fprintf(stderr, "\n*** Handshake failed ***\n");
514
/* Read OpenPGP packet that contains the wanted password */
517
fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
522
buffer_capacity = adjustbuffer(&buffer, buffer_length, buffer_capacity);
523
if (buffer_capacity == 0){
524
perror("adjustbuffer");
361
if (buffer_length + BUFFER_SIZE > buffer_capacity){
362
buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
367
buffer_capacity += BUFFER_SIZE;
529
ret = gnutls_record_recv(session, buffer+buffer_length,
370
ret = gnutls_record_recv
371
(es.session, buffer+buffer_length, BUFFER_SIZE);
537
378
case GNUTLS_E_AGAIN:
539
380
case GNUTLS_E_REHANDSHAKE:
540
ret = gnutls_handshake (session);
381
ret = gnutls_handshake (es.session);
542
fprintf(stderr, "*** GnuTLS Re-handshake failed ***\n");
383
fprintf(stderr, "\n*** Handshake failed ***\n");
543
384
gnutls_perror (ret);
549
fprintf(stderr, "Unknown error while reading data from"
550
" encrypted session with Mandos server\n");
390
fprintf(stderr, "Unknown error while reading data from encrypted session with mandos server\n");
552
gnutls_bye (session, GNUTLS_SHUT_RDWR);
392
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
556
buffer_length += (size_t) ret;
396
buffer_length += ret;
561
fprintf(stderr, "Closing TLS session\n");
564
gnutls_bye (session, GNUTLS_SHUT_RDWR);
566
400
if (buffer_length > 0){
567
decrypted_buffer_size = pgp_packet_decrypt(buffer,
571
if (decrypted_buffer_size >= 0){
573
while(written < (size_t) decrypted_buffer_size){
574
ret = (int)fwrite (decrypted_buffer + written, 1,
575
(size_t)decrypted_buffer_size - written,
577
if(ret == 0 and ferror(stdout)){
579
fprintf(stderr, "Error writing encrypted data: %s\n",
585
written += (size_t)ret;
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);
587
405
free(decrypted_buffer);
593
/* Shutdown procedure */
412
gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
598
gnutls_deinit (session);
599
gnutls_certificate_free_credentials (mc->cred);
415
gnutls_deinit (es.session);
416
gnutls_certificate_free_credentials (es.cred);
600
417
gnutls_global_deinit ();
604
static void resolve_callback(AvahiSServiceResolver *r,
605
AvahiIfIndex interface,
606
AVAHI_GCC_UNUSED AvahiProtocol protocol,
607
AvahiResolverEvent event,
611
const char *host_name,
612
const AvahiAddress *address,
614
AVAHI_GCC_UNUSED AvahiStringList *txt,
615
AVAHI_GCC_UNUSED AvahiLookupResultFlags
618
mandos_context *mc = userdata;
619
assert(r); /* Spurious warning */
621
/* Called whenever a service has been resolved successfully or
626
case AVAHI_RESOLVER_FAILURE:
627
fprintf(stderr, "(Avahi Resolver) Failed to resolve service '%s'"
628
" of type '%s' in domain '%s': %s\n", name, type, domain,
629
avahi_strerror(avahi_server_errno(mc->server)));
632
case AVAHI_RESOLVER_FOUND:
634
char ip[AVAHI_ADDRESS_STR_MAX];
635
avahi_address_snprint(ip, sizeof(ip), address);
637
fprintf(stderr, "Mandos server \"%s\" found on %s (%s, %d) on"
638
" port %d\n", name, host_name, ip, interface, port);
640
int ret = start_mandos_communication(ip, port, interface, mc);
646
avahi_s_service_resolver_free(r);
649
static void browse_callback( AvahiSServiceBrowser *b,
650
AvahiIfIndex interface,
651
AvahiProtocol protocol,
652
AvahiBrowserEvent event,
656
AVAHI_GCC_UNUSED AvahiLookupResultFlags
659
mandos_context *mc = userdata;
660
assert(b); /* Spurious warning */
662
/* Called whenever a new services becomes available on the LAN or
663
is removed from the LAN */
667
case AVAHI_BROWSER_FAILURE:
669
fprintf(stderr, "(Avahi browser) %s\n",
670
avahi_strerror(avahi_server_errno(mc->server)));
671
avahi_simple_poll_quit(mc->simple_poll);
674
case AVAHI_BROWSER_NEW:
675
/* We ignore the returned Avahi resolver object. In the callback
676
function we free it. If the Avahi server is terminated before
677
the callback function is called the Avahi server will free the
680
if (!(avahi_s_service_resolver_new(mc->server, interface,
681
protocol, name, type, domain,
682
AVAHI_PROTO_INET6, 0,
683
resolve_callback, mc)))
684
fprintf(stderr, "Avahi: Failed to resolve service '%s': %s\n",
685
name, avahi_strerror(avahi_server_errno(mc->server)));
688
case AVAHI_BROWSER_REMOVE:
691
case AVAHI_BROWSER_ALL_FOR_NOW:
692
case AVAHI_BROWSER_CACHE_EXHAUSTED:
694
fprintf(stderr, "No Mandos server found, still searching...\n");
700
/* Combines file name and path and returns the malloced new
701
string. some sane checks could/should be added */
702
static const char *combinepath(const char *first, const char *second){
703
size_t f_len = strlen(first);
704
size_t s_len = strlen(second);
705
char *tmp = malloc(f_len + s_len + 2);
710
memcpy(tmp, first, f_len); /* Spurious warning */
714
memcpy(tmp + f_len + 1, second, s_len); /* Spurious warning */
716
tmp[f_len + 1 + s_len] = '\0';
721
int main(int argc, char *argv[]){
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;
722
508
AvahiSServiceBrowser *sb = NULL;
725
int exitcode = EXIT_SUCCESS;
726
const char *interface = "eth0";
727
struct ifreq network;
731
char *connect_to = NULL;
732
AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
733
const char *pubkeyfile = "pubkey.txt";
734
const char *seckeyfile = "seckey.txt";
735
mandos_context mc = { .simple_poll = NULL, .server = NULL,
736
.dh_bits = 1024, .priority = "SECURE256"};
739
struct argp_option options[] = {
740
{ .name = "debug", .key = 128,
741
.doc = "Debug mode", .group = 3 },
742
{ .name = "connect", .key = 'c',
744
.doc = "Connect directly to a sepcified mandos server", .group = 1 },
745
{ .name = "interface", .key = 'i',
747
.doc = "Interface that Avahi will conntect through", .group = 1 },
748
{ .name = "keydir", .key = 'd',
750
.doc = "Directory where the openpgp keyring is", .group = 1 },
751
{ .name = "seckey", .key = 's',
753
.doc = "Secret openpgp key for gnutls authentication", .group = 1 },
754
{ .name = "pubkey", .key = 'p',
756
.doc = "Public openpgp key for gnutls authentication", .group = 2 },
757
{ .name = "dh-bits", .key = 129,
759
.doc = "dh-bits to use in gnutls communication", .group = 2 },
760
{ .name = "priority", .key = 130,
762
.doc = "GNUTLS priority", .group = 1 },
767
error_t parse_opt (int key, char *arg, struct argp_state *state) {
768
/* Get the INPUT argument from `argp_parse', which we know is a
769
pointer to our plugin list pointer. */
791
mc.dh_bits = (unsigned int) strtol(arg, NULL, 10);
806
return ARGP_ERR_UNKNOWN;
811
struct argp argp = { .options = options, .parser = parse_opt,
813
.doc = "Mandos client -- Get and decrypt passwords from mandos server" };
814
argp_parse (&argp, argc, argv, 0, 0, NULL);
817
pubkeyfile = combinepath(keydir, pubkeyfile);
818
if (pubkeyfile == NULL){
819
perror("combinepath");
820
exitcode = EXIT_FAILURE;
824
seckeyfile = combinepath(keydir, seckeyfile);
825
if (seckeyfile == NULL){
826
perror("combinepath");
830
ret = init_gnutls_global(&mc, pubkeyfile, seckeyfile);
832
fprintf(stderr, "init_gnutls_global\n");
849
if_index = (AvahiIfIndex) if_nametoindex(interface);
851
fprintf(stderr, "No such interface: \"%s\"\n", interface);
855
if(connect_to != NULL){
856
/* Connect directly, do not use Zeroconf */
857
/* (Mainly meant for debugging) */
858
char *address = strrchr(connect_to, ':');
860
fprintf(stderr, "No colon in address\n");
861
exitcode = EXIT_FAILURE;
865
uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
867
perror("Bad port number");
868
exitcode = EXIT_FAILURE;
872
address = connect_to;
873
ret = start_mandos_communication(address, port, if_index, &mc);
875
exitcode = EXIT_FAILURE;
877
exitcode = EXIT_SUCCESS;
882
/* If the interface is down, bring it up */
884
sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
887
exitcode = EXIT_FAILURE;
890
strcpy(network.ifr_name, interface); /* Spurious warning */
891
ret = ioctl(sd, SIOCGIFFLAGS, &network);
893
perror("ioctl SIOCGIFFLAGS");
894
exitcode = EXIT_FAILURE;
897
if((network.ifr_flags & IFF_UP) == 0){
898
network.ifr_flags |= IFF_UP;
899
ret = ioctl(sd, SIOCSIFFLAGS, &network);
901
perror("ioctl SIOCSIFFLAGS");
902
exitcode = EXIT_FAILURE;
910
avahi_set_log_function(empty_log);
913
/* Initialize the pseudo-RNG for Avahi */
914
srand((unsigned int) time(NULL));
916
/* Allocate main Avahi loop object */
917
mc.simple_poll = avahi_simple_poll_new();
918
if (mc.simple_poll == NULL) {
919
fprintf(stderr, "Avahi: Failed to create simple poll"
921
exitcode = EXIT_FAILURE;
926
AvahiServerConfig config;
927
/* Do not publish any local Zeroconf records */
928
avahi_server_config_init(&config);
929
config.publish_hinfo = 0;
930
config.publish_addresses = 0;
931
config.publish_workstation = 0;
932
config.publish_domain = 0;
934
/* Allocate a new server */
935
mc.server = avahi_server_new(avahi_simple_poll_get
936
(mc.simple_poll), &config, NULL,
939
/* Free the Avahi configuration data */
940
avahi_server_config_free(&config);
943
/* Check if creating the Avahi server object succeeded */
944
if (mc.server == NULL) {
945
fprintf(stderr, "Failed to create Avahi server: %s\n",
946
avahi_strerror(error));
947
exitcode = EXIT_FAILURE;
951
/* Create the Avahi service browser */
952
sb = avahi_s_service_browser_new(mc.server, if_index,
954
"_mandos._tcp", NULL, 0,
955
browse_callback, &mc);
957
fprintf(stderr, "Failed to create service browser: %s\n",
958
avahi_strerror(avahi_server_errno(mc.server)));
959
exitcode = EXIT_FAILURE;
512
avahi_set_log_function(empty_log);
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 */
524
avahi_server_config_init(&config);
525
config.publish_hinfo = 0;
526
config.publish_addresses = 0;
527
config.publish_workstation = 0;
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; */
535
/* Allocate a new server */
536
server = avahi_server_new(avahi_simple_poll_get(simple_poll), &config, NULL, NULL, &error);
538
/* Free the configuration data */
539
avahi_server_config_free(&config);
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)));
963
553
/* Run the main loop */
966
fprintf(stderr, "Starting Avahi loop search\n");
969
avahi_simple_poll_loop(mc.simple_poll);
974
fprintf(stderr, "%s exiting\n", argv[0]);
554
avahi_simple_poll_loop(simple_poll);
977
560
/* Cleanup things */
979
562
avahi_s_service_browser_free(sb);
981
if (mc.server != NULL)
982
avahi_server_free(mc.server);
984
if (mc.simple_poll != NULL)
985
avahi_simple_poll_free(mc.simple_poll);
565
avahi_server_free(server);
568
avahi_simple_poll_free(simple_poll);