/mandos/trunk

To get this branch, use:
bzr branch http://bzr.recompile.se/loggerhead/mandos/trunk

« back to all changes in this revision

Viewing changes to plugins.d/password-request.c

  • Committer: Teddy Hogeborn
  • Date: 2008-09-02 17:42:53 UTC
  • Revision ID: teddy@fukt.bsnet.se-20080902174253-p3wxrq7z6ccnv7fs
* plugins.d/password-request.c (main): Change default GnuTLS priority
                                       string to
                             "SECURE256":!CTYPE-X.509:+CTYPE-OPENPGP".

* plugins.d/password-request.xml (DESCRIPTION): Improve wording.
  (PURPOSE, OVERVIEW): New sections.
  (OPTIONS): Improved wording.
  (EXIT STATUS): Add text.
  (ENVIRONMENT): Commented out.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
/* $Id$ */
2
 
 
3
 
/* PLEASE NOTE *
4
 
 * This file demonstrates how to use Avahi's core API, this is
5
 
 * the embeddable mDNS stack for embedded applications.
 
1
/*  -*- coding: utf-8 -*- */
 
2
/*
 
3
 * Mandos client - get and decrypt data from a Mandos server
6
4
 *
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
10
 
 * 
11
 
 * I repeat, you probably do *not* want to use this example.
 
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".
 
10
 * 
 
11
 * Everything else is
 
12
 * Copyright © 2007-2008 Teddy Hogeborn & Björn Påhlsson
 
13
 * 
 
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.
 
18
 * 
 
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.
 
23
 * 
 
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/>.
 
27
 * 
 
28
 * Contact the authors at <mandos@fukt.bsnet.se>.
12
29
 */
13
30
 
14
 
/***
15
 
  This file is part of avahi.
16
 
 
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.
21
 
 
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.
26
 
 
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
30
 
  USA.
31
 
***/
32
 
 
 
31
/* Needed by GPGME, specifically gpgme_data_seek() */
33
32
#define _LARGEFILE_SOURCE
34
33
#define _FILE_OFFSET_BITS 64
35
34
 
36
 
#include <stdio.h>
37
 
#include <assert.h>
38
 
#include <stdlib.h>
39
 
#include <time.h>
40
 
#include <net/if.h>             /* if_nametoindex */
41
 
 
 
35
#define _GNU_SOURCE             /* TEMP_FAILURE_RETRY(), asprintf() */
 
36
 
 
37
#include <stdio.h>              /* fprintf(), stderr, fwrite(),
 
38
                                   stdout, ferror() */
 
39
#include <stdint.h>             /* uint16_t, uint32_t */
 
40
#include <stddef.h>             /* NULL, size_t, ssize_t */
 
41
#include <stdlib.h>             /* free(), EXIT_SUCCESS, EXIT_FAILURE,
 
42
                                   srand() */
 
43
#include <stdbool.h>            /* bool, true */
 
44
#include <string.h>             /* memset(), strcmp(), strlen(),
 
45
                                   strerror(), asprintf(), strcpy() */
 
46
#include <sys/ioctl.h>          /* ioctl */
 
47
#include <sys/types.h>          /* socket(), inet_pton(), sockaddr,
 
48
                                   sockaddr_in6, PF_INET6,
 
49
                                   SOCK_STREAM, INET6_ADDRSTRLEN,
 
50
                                   uid_t, gid_t */
 
51
#include <inttypes.h>           /* PRIu16 */
 
52
#include <sys/socket.h>         /* socket(), struct sockaddr_in6,
 
53
                                   struct in6_addr, inet_pton(),
 
54
                                   connect() */
 
55
#include <assert.h>             /* assert() */
 
56
#include <errno.h>              /* perror(), errno */
 
57
#include <time.h>               /* time() */
 
58
#include <net/if.h>             /* ioctl, ifreq, SIOCGIFFLAGS, IFF_UP,
 
59
                                   SIOCSIFFLAGS, if_indextoname(),
 
60
                                   if_nametoindex(), IF_NAMESIZE */
 
61
#include <unistd.h>             /* close(), SEEK_SET, off_t, write(),
 
62
                                   getuid(), getgid(), setuid(),
 
63
                                   setgid() */
 
64
#include <netinet/in.h>
 
65
#include <arpa/inet.h>          /* inet_pton(), htons */
 
66
#include <iso646.h>             /* not, and */
 
67
#include <argp.h>               /* struct argp_option, error_t, struct
 
68
                                   argp_state, struct argp,
 
69
                                   argp_parse(), ARGP_KEY_ARG,
 
70
                                   ARGP_KEY_END, ARGP_ERR_UNKNOWN */
 
71
 
 
72
/* Avahi */
 
73
/* All Avahi types, constants and functions
 
74
 Avahi*, avahi_*,
 
75
 AVAHI_* */
42
76
#include <avahi-core/core.h>
43
77
#include <avahi-core/lookup.h>
44
78
#include <avahi-core/log.h>
46
80
#include <avahi-common/malloc.h>
47
81
#include <avahi-common/error.h>
48
82
 
49
 
//mandos client part
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 */
54
 
 
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 */
61
 
 
62
 
// gpgme
63
 
#include <errno.h>              /* perror() */
64
 
#include <gpgme.h>
65
 
 
66
 
 
67
 
#ifndef CERT_ROOT
68
 
#define CERT_ROOT "/conf/conf.d/cryptkeyreq/"
69
 
#endif
70
 
#define CERTFILE CERT_ROOT "openpgp-client.txt"
71
 
#define KEYFILE CERT_ROOT "openpgp-client-key.txt"
 
83
/* GnuTLS */
 
84
#include <gnutls/gnutls.h>      /* All GnuTLS types, constants and
 
85
                                   functions:
 
86
                                   gnutls_*
 
87
                                   init_gnutls_session(),
 
88
                                   GNUTLS_* */
 
89
#include <gnutls/openpgp.h>     /* gnutls_certificate_set_openpgp_key_file(),
 
90
                                   GNUTLS_OPENPGP_FMT_BASE64 */
 
91
 
 
92
/* GPGME */
 
93
#include <gpgme.h>              /* All GPGME types, constants and
 
94
                                   functions:
 
95
                                   gpgme_*
 
96
                                   GPGME_PROTOCOL_OpenPGP,
 
97
                                   GPG_ERR_NO_* */
 
98
 
72
99
#define BUFFER_SIZE 256
73
 
#define DH_BITS 1024
74
 
 
 
100
 
 
101
bool debug = false;
 
102
static const char *keydir = "/conf/conf.d/mandos";
 
103
static const char mandos_protocol_version[] = "1";
 
104
const char *argp_program_version = "password-request 1.0";
 
105
const char *argp_program_bug_address = "<mandos@fukt.bsnet.se>";
 
106
 
 
107
/* Used for passing in values through the Avahi callback functions */
75
108
typedef struct {
76
 
  gnutls_session_t session;
 
109
  AvahiSimplePoll *simple_poll;
 
110
  AvahiServer *server;
77
111
  gnutls_certificate_credentials_t cred;
 
112
  unsigned int dh_bits;
78
113
  gnutls_dh_params_t dh_params;
79
 
} encrypted_session;
80
 
 
81
 
 
82
 
ssize_t gpg_packet_decrypt (char *packet, size_t packet_size, char **new_packet, char *homedir){
 
114
  const char *priority;
 
115
} mandos_context;
 
116
 
 
117
/*
 
118
 * Make room in "buffer" for at least BUFFER_SIZE additional bytes.
 
119
 * "buffer_capacity" is how much is currently allocated,
 
120
 * "buffer_length" is how much is already used.
 
121
 */
 
122
size_t adjustbuffer(char **buffer, size_t buffer_length,
 
123
                  size_t buffer_capacity){
 
124
  if (buffer_length + BUFFER_SIZE > buffer_capacity){
 
125
    *buffer = realloc(*buffer, buffer_capacity + BUFFER_SIZE);
 
126
    if (buffer == NULL){
 
127
      return 0;
 
128
    }
 
129
    buffer_capacity += BUFFER_SIZE;
 
130
  }
 
131
  return buffer_capacity;
 
132
}
 
133
 
 
134
/* 
 
135
 * Decrypt OpenPGP data using keyrings in HOMEDIR.
 
136
 * Returns -1 on error
 
137
 */
 
138
static ssize_t pgp_packet_decrypt (const char *cryptotext,
 
139
                                   size_t crypto_size,
 
140
                                   char **plaintext,
 
141
                                   const char *homedir){
83
142
  gpgme_data_t dh_crypto, dh_plain;
84
143
  gpgme_ctx_t ctx;
85
144
  gpgme_error_t rc;
86
145
  ssize_t ret;
87
 
  size_t new_packet_capacity = 0;
88
 
  size_t new_packet_length = 0;
 
146
  size_t plaintext_capacity = 0;
 
147
  ssize_t plaintext_length = 0;
89
148
  gpgme_engine_info_t engine_info;
90
 
 
 
149
  
 
150
  if (debug){
 
151
    fprintf(stderr, "Trying to decrypt OpenPGP data\n");
 
152
  }
 
153
  
91
154
  /* Init GPGME */
92
155
  gpgme_check_version(NULL);
93
 
  gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
 
156
  rc = gpgme_engine_check_version(GPGME_PROTOCOL_OpenPGP);
 
157
  if (rc != GPG_ERR_NO_ERROR){
 
158
    fprintf(stderr, "bad gpgme_engine_check_version: %s: %s\n",
 
159
            gpgme_strsource(rc), gpgme_strerror(rc));
 
160
    return -1;
 
161
  }
94
162
  
95
 
  /* Set GPGME home directory */
 
163
  /* Set GPGME home directory for the OpenPGP engine only */
96
164
  rc = gpgme_get_engine_info (&engine_info);
97
165
  if (rc != GPG_ERR_NO_ERROR){
98
166
    fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
108
176
    engine_info = engine_info->next;
109
177
  }
110
178
  if(engine_info == NULL){
111
 
    fprintf(stderr, "Could not set home dir to %s\n", homedir);
 
179
    fprintf(stderr, "Could not set GPGME home dir to %s\n", homedir);
112
180
    return -1;
113
181
  }
114
182
  
115
 
  /* Create new GPGME data buffer from packet buffer */
116
 
  rc = gpgme_data_new_from_mem(&dh_crypto, packet, packet_size, 0);
 
183
  /* Create new GPGME data buffer from memory cryptotext */
 
184
  rc = gpgme_data_new_from_mem(&dh_crypto, cryptotext, crypto_size,
 
185
                               0);
117
186
  if (rc != GPG_ERR_NO_ERROR){
118
187
    fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
119
188
            gpgme_strsource(rc), gpgme_strerror(rc));
125
194
  if (rc != GPG_ERR_NO_ERROR){
126
195
    fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
127
196
            gpgme_strsource(rc), gpgme_strerror(rc));
 
197
    gpgme_data_release(dh_crypto);
128
198
    return -1;
129
199
  }
130
200
  
133
203
  if (rc != GPG_ERR_NO_ERROR){
134
204
    fprintf(stderr, "bad gpgme_new: %s: %s\n",
135
205
            gpgme_strsource(rc), gpgme_strerror(rc));
136
 
    return -1;
 
206
    plaintext_length = -1;
 
207
    goto decrypt_end;
137
208
  }
138
209
  
139
 
  /* Decrypt data from the FILE pointer to the plaintext data buffer */
 
210
  /* Decrypt data from the cryptotext data buffer to the plaintext
 
211
     data buffer */
140
212
  rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
141
213
  if (rc != GPG_ERR_NO_ERROR){
142
214
    fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
143
215
            gpgme_strsource(rc), gpgme_strerror(rc));
144
 
    return -1;
 
216
    plaintext_length = -1;
 
217
    if (debug){
 
218
      gpgme_decrypt_result_t result;
 
219
      result = gpgme_op_decrypt_result(ctx);
 
220
      if (result == NULL){
 
221
        fprintf(stderr, "gpgme_op_decrypt_result failed\n");
 
222
      } else {
 
223
        fprintf(stderr, "Unsupported algorithm: %s\n",
 
224
                result->unsupported_algorithm);
 
225
        fprintf(stderr, "Wrong key usage: %u\n",
 
226
                result->wrong_key_usage);
 
227
        if(result->file_name != NULL){
 
228
          fprintf(stderr, "File name: %s\n", result->file_name);
 
229
        }
 
230
        gpgme_recipient_t recipient;
 
231
        recipient = result->recipients;
 
232
        if(recipient){
 
233
          while(recipient != NULL){
 
234
            fprintf(stderr, "Public key algorithm: %s\n",
 
235
                    gpgme_pubkey_algo_name(recipient->pubkey_algo));
 
236
            fprintf(stderr, "Key ID: %s\n", recipient->keyid);
 
237
            fprintf(stderr, "Secret key available: %s\n",
 
238
                    recipient->status == GPG_ERR_NO_SECKEY
 
239
                    ? "No" : "Yes");
 
240
            recipient = recipient->next;
 
241
          }
 
242
        }
 
243
      }
 
244
    }
 
245
    goto decrypt_end;
145
246
  }
146
247
  
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); */
153
 
/*   } */
154
 
/*   gpgme_recipient_t recipient; */
155
 
/*   recipient = result->recipients; */
156
 
/*   if(recipient){ */
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; */
164
 
/*     } */
165
 
/*   } */
166
 
 
167
 
  /* Delete the GPGME FILE pointer cryptotext data buffer */
168
 
  gpgme_data_release(dh_crypto);
 
248
  if(debug){
 
249
    fprintf(stderr, "Decryption of OpenPGP data succeeded\n");
 
250
  }
169
251
  
170
252
  /* Seek back to the beginning of the GPGME plaintext data buffer */
171
 
  gpgme_data_seek(dh_plain, 0, SEEK_SET);
172
 
 
173
 
  *new_packet = 0;
 
253
  if (gpgme_data_seek(dh_plain, (off_t) 0, SEEK_SET) == -1){
 
254
    perror("pgpme_data_seek");
 
255
    plaintext_length = -1;
 
256
    goto decrypt_end;
 
257
  }
 
258
  
 
259
  *plaintext = NULL;
174
260
  while(true){
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){
178
 
        perror("realloc");
179
 
        return -1;
180
 
      }
181
 
      new_packet_capacity += BUFFER_SIZE;
 
261
    plaintext_capacity = adjustbuffer(plaintext,
 
262
                                      (size_t)plaintext_length,
 
263
                                      plaintext_capacity);
 
264
    if (plaintext_capacity == 0){
 
265
        perror("adjustbuffer");
 
266
        plaintext_length = -1;
 
267
        goto decrypt_end;
182
268
    }
183
269
    
184
 
    ret = gpgme_data_read(dh_plain, *new_packet + new_packet_length, BUFFER_SIZE);
 
270
    ret = gpgme_data_read(dh_plain, *plaintext + plaintext_length,
 
271
                          BUFFER_SIZE);
185
272
    /* Print the data, if any */
186
273
    if (ret == 0){
187
 
      /* If password is empty, then a incorrect error will be printed */
 
274
      /* EOF */
188
275
      break;
189
276
    }
190
277
    if(ret < 0){
191
278
      perror("gpgme_data_read");
192
 
      return -1;
 
279
      plaintext_length = -1;
 
280
      goto decrypt_end;
193
281
    }
194
 
    new_packet_length += ret;
 
282
    plaintext_length += ret;
195
283
  }
196
284
 
197
 
   /* Delete the GPGME plaintext data buffer */
 
285
  if(debug){
 
286
    fprintf(stderr, "Decrypted password is: ");
 
287
    for(ssize_t i = 0; i < plaintext_length; i++){
 
288
      fprintf(stderr, "%02hhX ", (*plaintext)[i]);
 
289
    }
 
290
    fprintf(stderr, "\n");
 
291
  }
 
292
  
 
293
 decrypt_end:
 
294
  
 
295
  /* Delete the GPGME cryptotext data buffer */
 
296
  gpgme_data_release(dh_crypto);
 
297
  
 
298
  /* Delete the GPGME plaintext data buffer */
198
299
  gpgme_data_release(dh_plain);
199
 
  return new_packet_length;
 
300
  return plaintext_length;
200
301
}
201
302
 
202
303
static const char * safer_gnutls_strerror (int value) {
203
 
  const char *ret = gnutls_strerror (value);
 
304
  const char *ret = gnutls_strerror (value); /* Spurious warning */
204
305
  if (ret == NULL)
205
306
    ret = "(unknown)";
206
307
  return ret;
207
308
}
208
309
 
209
 
void debuggnutls(int level, const char* string){
210
 
  fprintf(stderr, "%s", string);
 
310
/* GnuTLS log function callback */
 
311
static void debuggnutls(__attribute__((unused)) int level,
 
312
                        const char* string){
 
313
  fprintf(stderr, "GnuTLS: %s", string);
211
314
}
212
315
 
213
 
int initgnutls(encrypted_session *es){
214
 
  const char *err;
 
316
static int init_gnutls_global(mandos_context *mc,
 
317
                              const char *pubkeyfilename,
 
318
                              const char *seckeyfilename){
215
319
  int ret;
216
320
  
217
 
  if ((ret = gnutls_global_init ())
218
 
      != GNUTLS_E_SUCCESS) {
219
 
    fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
220
 
    return -1;
221
 
  }
222
 
 
223
 
  /* Uncomment to enable full debuggin on the gnutls library */
224
 
  /*   gnutls_global_set_log_level(11); */
225
 
  /*   gnutls_global_set_log_function(debuggnutls); */
226
 
 
227
 
 
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));
232
 
    return -1;
233
 
  }
234
 
 
 
321
  if(debug){
 
322
    fprintf(stderr, "Initializing GnuTLS\n");
 
323
  }
 
324
  
 
325
  ret = gnutls_global_init();
 
326
  if (ret != GNUTLS_E_SUCCESS) {
 
327
    fprintf (stderr, "GnuTLS global_init: %s\n",
 
328
             safer_gnutls_strerror(ret));
 
329
    return -1;
 
330
  }
 
331
  
 
332
  if (debug){
 
333
    /* "Use a log level over 10 to enable all debugging options."
 
334
     * - GnuTLS manual
 
335
     */
 
336
    gnutls_global_set_log_level(11);
 
337
    gnutls_global_set_log_function(debuggnutls);
 
338
  }
 
339
  
 
340
  /* OpenPGP credentials */
 
341
  gnutls_certificate_allocate_credentials(&mc->cred);
 
342
  if (ret != GNUTLS_E_SUCCESS){
 
343
    fprintf (stderr, "GnuTLS memory error: %s\n", /* Spurious
 
344
                                                     warning */
 
345
             safer_gnutls_strerror(ret));
 
346
    gnutls_global_deinit ();
 
347
    return -1;
 
348
  }
 
349
  
 
350
  if(debug){
 
351
    fprintf(stderr, "Attempting to use OpenPGP certificate %s"
 
352
            " and keyfile %s as GnuTLS credentials\n", pubkeyfilename,
 
353
            seckeyfilename);
 
354
  }
 
355
  
235
356
  ret = gnutls_certificate_set_openpgp_key_file
236
 
    (es->cred, CERTFILE, KEYFILE, GNUTLS_OPENPGP_FMT_BASE64);
237
 
  if (ret != GNUTLS_E_SUCCESS) {
238
 
    fprintf
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));
243
 
    return -1;
244
 
  }
245
 
 
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));
251
 
    return -1;
252
 
  }
253
 
 
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));
258
 
    return -1;
259
 
  }
260
 
 
261
 
  gnutls_certificate_set_dh_params (es->cred, es->dh_params);
262
 
 
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));
268
 
  }
269
 
 
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));
275
 
    return -1;
276
 
  }
277
 
 
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));
283
 
    return -1;
284
 
  }
285
 
 
 
357
    (mc->cred, pubkeyfilename, seckeyfilename,
 
358
     GNUTLS_OPENPGP_FMT_BASE64);
 
359
  if (ret != GNUTLS_E_SUCCESS) {
 
360
    fprintf(stderr,
 
361
            "Error[%d] while reading the OpenPGP key pair ('%s',"
 
362
            " '%s')\n", ret, pubkeyfilename, seckeyfilename);
 
363
    fprintf(stdout, "The GnuTLS error is: %s\n",
 
364
            safer_gnutls_strerror(ret));
 
365
    goto globalfail;
 
366
  }
 
367
  
 
368
  /* GnuTLS server initialization */
 
369
  ret = gnutls_dh_params_init(&mc->dh_params);
 
370
  if (ret != GNUTLS_E_SUCCESS) {
 
371
    fprintf (stderr, "Error in GnuTLS DH parameter initialization:"
 
372
             " %s\n", safer_gnutls_strerror(ret));
 
373
    goto globalfail;
 
374
  }
 
375
  ret = gnutls_dh_params_generate2(mc->dh_params, mc->dh_bits);
 
376
  if (ret != GNUTLS_E_SUCCESS) {
 
377
    fprintf (stderr, "Error in GnuTLS prime generation: %s\n",
 
378
             safer_gnutls_strerror(ret));
 
379
    goto globalfail;
 
380
  }
 
381
  
 
382
  gnutls_certificate_set_dh_params(mc->cred, mc->dh_params);
 
383
 
 
384
  return 0;
 
385
 
 
386
 globalfail:
 
387
 
 
388
  gnutls_certificate_free_credentials(mc->cred);
 
389
  gnutls_global_deinit();
 
390
  return -1;
 
391
 
 
392
}
 
393
 
 
394
static int init_gnutls_session(mandos_context *mc,
 
395
                               gnutls_session_t *session){
 
396
  int ret;
 
397
  /* GnuTLS session creation */
 
398
  ret = gnutls_init(session, GNUTLS_SERVER);
 
399
  if (ret != GNUTLS_E_SUCCESS){
 
400
    fprintf(stderr, "Error in GnuTLS session initialization: %s\n",
 
401
            safer_gnutls_strerror(ret));
 
402
  }
 
403
  
 
404
  {
 
405
    const char *err;
 
406
    ret = gnutls_priority_set_direct(*session, mc->priority, &err);
 
407
    if (ret != GNUTLS_E_SUCCESS) {
 
408
      fprintf(stderr, "Syntax error at: %s\n", err);
 
409
      fprintf(stderr, "GnuTLS error: %s\n",
 
410
              safer_gnutls_strerror(ret));
 
411
      gnutls_deinit (*session);
 
412
      return -1;
 
413
    }
 
414
  }
 
415
  
 
416
  ret = gnutls_credentials_set(*session, GNUTLS_CRD_CERTIFICATE,
 
417
                               mc->cred);
 
418
  if (ret != GNUTLS_E_SUCCESS) {
 
419
    fprintf(stderr, "Error setting GnuTLS credentials: %s\n",
 
420
            safer_gnutls_strerror(ret));
 
421
    gnutls_deinit (*session);
 
422
    return -1;
 
423
  }
 
424
  
286
425
  /* ignore client certificate if any. */
287
 
  gnutls_certificate_server_set_request (es->session, GNUTLS_CERT_IGNORE);
 
426
  gnutls_certificate_server_set_request (*session,
 
427
                                         GNUTLS_CERT_IGNORE);
288
428
  
289
 
  gnutls_dh_set_prime_bits (es->session, DH_BITS);
 
429
  gnutls_dh_set_prime_bits (*session, mc->dh_bits);
290
430
  
291
431
  return 0;
292
432
}
293
433
 
294
 
void empty_log(AvahiLogLevel level, const char *txt){}
 
434
/* Avahi log function callback */
 
435
static void empty_log(__attribute__((unused)) AvahiLogLevel level,
 
436
                      __attribute__((unused)) const char *txt){}
295
437
 
296
 
int start_mandos_communcation(char *ip, uint16_t port){
 
438
/* Called when a Mandos server is found */
 
439
static int start_mandos_communication(const char *ip, uint16_t port,
 
440
                                      AvahiIfIndex if_index,
 
441
                                      mandos_context *mc){
297
442
  int ret, tcp_sd;
298
 
  struct sockaddr_in6 to;
299
 
  struct in6_addr ip_addr;
300
 
  encrypted_session es;
 
443
  union { struct sockaddr in; struct sockaddr_in6 in6; } to;
301
444
  char *buffer = NULL;
302
445
  char *decrypted_buffer;
303
446
  size_t buffer_length = 0;
304
447
  size_t buffer_capacity = 0;
305
448
  ssize_t decrypted_buffer_size;
 
449
  size_t written;
306
450
  int retval = 0;
307
 
 
 
451
  char interface[IF_NAMESIZE];
 
452
  gnutls_session_t session;
 
453
  
 
454
  ret = init_gnutls_session (mc, &session);
 
455
  if (ret != 0){
 
456
    return -1;
 
457
  }
 
458
  
 
459
  if(debug){
 
460
    fprintf(stderr, "Setting up a tcp connection to %s, port %" PRIu16
 
461
            "\n", ip, port);
 
462
  }
308
463
  
309
464
  tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
310
465
  if(tcp_sd < 0) {
311
466
    perror("socket");
312
467
    return -1;
313
468
  }
314
 
  
315
 
  ret = setsockopt(tcp_sd, SOL_SOCKET, SO_BINDTODEVICE, "eth0", 5);
316
 
  if(tcp_sd < 0) {
317
 
    perror("setsockopt bindtodevice");
318
 
    return -1;
 
469
 
 
470
  if(debug){
 
471
    if(if_indextoname((unsigned int)if_index, interface) == NULL){
 
472
      perror("if_indextoname");
 
473
      return -1;
 
474
    }
 
475
    fprintf(stderr, "Binding to interface %s\n", interface);
319
476
  }
320
477
  
321
 
  memset(&to,0,sizeof(to));
322
 
  to.sin6_family = AF_INET6;
323
 
  ret = inet_pton(AF_INET6, ip, &ip_addr);
 
478
  memset(&to, 0, sizeof(to));
 
479
  to.in6.sin6_family = AF_INET6;
 
480
  /* It would be nice to have a way to detect if we were passed an
 
481
     IPv4 address here.   Now we assume an IPv6 address. */
 
482
  ret = inet_pton(AF_INET6, ip, &to.in6.sin6_addr);
324
483
  if (ret < 0 ){
325
484
    perror("inet_pton");
326
485
    return -1;
327
 
  }  
 
486
  }
328
487
  if(ret == 0){
329
488
    fprintf(stderr, "Bad address: %s\n", ip);
330
489
    return -1;
331
490
  }
332
 
  to.sin6_port = htons(port);
333
 
  to.sin6_scope_id = if_nametoindex("eth0");
334
 
  
335
 
  ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
 
491
  to.in6.sin6_port = htons(port); /* Spurious warning */
 
492
  
 
493
  to.in6.sin6_scope_id = (uint32_t)if_index;
 
494
  
 
495
  if(debug){
 
496
    fprintf(stderr, "Connection to: %s, port %" PRIu16 "\n", ip,
 
497
            port);
 
498
    char addrstr[INET6_ADDRSTRLEN] = "";
 
499
    if(inet_ntop(to.in6.sin6_family, &(to.in6.sin6_addr), addrstr,
 
500
                 sizeof(addrstr)) == NULL){
 
501
      perror("inet_ntop");
 
502
    } else {
 
503
      if(strcmp(addrstr, ip) != 0){
 
504
        fprintf(stderr, "Canonical address form: %s\n", addrstr);
 
505
      }
 
506
    }
 
507
  }
 
508
  
 
509
  ret = connect(tcp_sd, &to.in, sizeof(to));
336
510
  if (ret < 0){
337
511
    perror("connect");
338
512
    return -1;
339
513
  }
340
 
  
341
 
  ret = initgnutls (&es);
342
 
  if (ret != 0){
343
 
    retval = -1;
344
 
    return -1;
345
 
  }
346
 
    
347
 
  
348
 
  gnutls_transport_set_ptr (es.session, (gnutls_transport_ptr_t) tcp_sd);
349
 
 
350
 
  ret = gnutls_handshake (es.session);
 
514
 
 
515
  const char *out = mandos_protocol_version;
 
516
  written = 0;
 
517
  while (true){
 
518
    size_t out_size = strlen(out);
 
519
    ret = TEMP_FAILURE_RETRY(write(tcp_sd, out + written,
 
520
                                   out_size - written));
 
521
    if (ret == -1){
 
522
      perror("write");
 
523
      retval = -1;
 
524
      goto mandos_end;
 
525
    }
 
526
    written += (size_t)ret;
 
527
    if(written < out_size){
 
528
      continue;
 
529
    } else {
 
530
      if (out == mandos_protocol_version){
 
531
        written = 0;
 
532
        out = "\r\n";
 
533
      } else {
 
534
        break;
 
535
      }
 
536
    }
 
537
  }
 
538
 
 
539
  if(debug){
 
540
    fprintf(stderr, "Establishing TLS session with %s\n", ip);
 
541
  }
 
542
  
 
543
  gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) tcp_sd);
 
544
 
 
545
  do{
 
546
    ret = gnutls_handshake (session);
 
547
  } while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
351
548
  
352
549
  if (ret != GNUTLS_E_SUCCESS){
353
 
    fprintf(stderr, "\n*** Handshake failed ***\n");
354
 
    gnutls_perror (ret);
 
550
    if(debug){
 
551
      fprintf(stderr, "*** GnuTLS Handshake failed ***\n");
 
552
      gnutls_perror (ret);
 
553
    }
355
554
    retval = -1;
356
 
    goto exit;
 
555
    goto mandos_end;
 
556
  }
 
557
  
 
558
  /* Read OpenPGP packet that contains the wanted password */
 
559
  
 
560
  if(debug){
 
561
    fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
 
562
            ip);
357
563
  }
358
564
 
359
 
  //retrive password
360
565
  while(true){
361
 
    if (buffer_length + BUFFER_SIZE > buffer_capacity){
362
 
      buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
363
 
      if (buffer == NULL){
364
 
        perror("realloc");
365
 
        goto exit;
366
 
      }
367
 
      buffer_capacity += BUFFER_SIZE;
 
566
    buffer_capacity = adjustbuffer(&buffer, buffer_length,
 
567
                                   buffer_capacity);
 
568
    if (buffer_capacity == 0){
 
569
      perror("adjustbuffer");
 
570
      retval = -1;
 
571
      goto mandos_end;
368
572
    }
369
573
    
370
 
    ret = gnutls_record_recv
371
 
      (es.session, buffer+buffer_length, BUFFER_SIZE);
 
574
    ret = gnutls_record_recv(session, buffer+buffer_length,
 
575
                             BUFFER_SIZE);
372
576
    if (ret == 0){
373
577
      break;
374
578
    }
378
582
      case GNUTLS_E_AGAIN:
379
583
        break;
380
584
      case GNUTLS_E_REHANDSHAKE:
381
 
        ret = gnutls_handshake (es.session);
 
585
        do{
 
586
          ret = gnutls_handshake (session);
 
587
        } while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
382
588
        if (ret < 0){
383
 
          fprintf(stderr, "\n*** Handshake failed ***\n");
 
589
          fprintf(stderr, "*** GnuTLS Re-handshake failed ***\n");
384
590
          gnutls_perror (ret);
385
591
          retval = -1;
386
 
          goto exit;
 
592
          goto mandos_end;
387
593
        }
388
594
        break;
389
595
      default:
390
 
        fprintf(stderr, "Unknown error while reading data from encrypted session with mandos server\n");
 
596
        fprintf(stderr, "Unknown error while reading data from"
 
597
                " encrypted session with Mandos server\n");
391
598
        retval = -1;
392
 
        gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
393
 
        goto exit;
 
599
        gnutls_bye (session, GNUTLS_SHUT_RDWR);
 
600
        goto mandos_end;
394
601
      }
395
602
    } else {
396
 
      buffer_length += ret;
 
603
      buffer_length += (size_t) ret;
397
604
    }
398
605
  }
399
 
 
 
606
  
 
607
  if(debug){
 
608
    fprintf(stderr, "Closing TLS session\n");
 
609
  }
 
610
  
 
611
  gnutls_bye (session, GNUTLS_SHUT_RDWR);
 
612
  
400
613
  if (buffer_length > 0){
401
 
    if ((decrypted_buffer_size = gpg_packet_decrypt(buffer, buffer_length, &decrypted_buffer, CERT_ROOT)) == 0){
402
 
      retval = -1;
403
 
    } else {
404
 
      fwrite (decrypted_buffer, 1, decrypted_buffer_size, stdout);
 
614
    decrypted_buffer_size = pgp_packet_decrypt(buffer,
 
615
                                               buffer_length,
 
616
                                               &decrypted_buffer,
 
617
                                               keydir);
 
618
    if (decrypted_buffer_size >= 0){
 
619
      written = 0;
 
620
      while(written < (size_t) decrypted_buffer_size){
 
621
        ret = (int)fwrite (decrypted_buffer + written, 1,
 
622
                           (size_t)decrypted_buffer_size - written,
 
623
                           stdout);
 
624
        if(ret == 0 and ferror(stdout)){
 
625
          if(debug){
 
626
            fprintf(stderr, "Error writing encrypted data: %s\n",
 
627
                    strerror(errno));
 
628
          }
 
629
          retval = -1;
 
630
          break;
 
631
        }
 
632
        written += (size_t)ret;
 
633
      }
405
634
      free(decrypted_buffer);
 
635
    } else {
 
636
      retval = -1;
406
637
    }
 
638
  } else {
 
639
    retval = -1;
407
640
  }
408
 
 
 
641
  
 
642
  /* Shutdown procedure */
 
643
  
 
644
 mandos_end:
409
645
  free(buffer);
410
 
 
411
 
  //shutdown procedure
412
 
  gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
413
 
 exit:
414
646
  close(tcp_sd);
415
 
  gnutls_deinit (es.session);
416
 
  gnutls_certificate_free_credentials (es.cred);
417
 
  gnutls_global_deinit ();
 
647
  gnutls_deinit (session);
418
648
  return retval;
419
649
}
420
650
 
421
 
static AvahiSimplePoll *simple_poll = NULL;
422
 
static AvahiServer *server = NULL;
423
 
 
424
 
static void resolve_callback(
425
 
    AvahiSServiceResolver *r,
426
 
    AVAHI_GCC_UNUSED AvahiIfIndex interface,
427
 
    AVAHI_GCC_UNUSED AvahiProtocol protocol,
428
 
    AvahiResolverEvent event,
429
 
    const char *name,
430
 
    const char *type,
431
 
    const char *domain,
432
 
    const char *host_name,
433
 
    const AvahiAddress *address,
434
 
    uint16_t port,
435
 
    AvahiStringList *txt,
436
 
    AvahiLookupResultFlags flags,
437
 
    AVAHI_GCC_UNUSED void* userdata) {
438
 
    
439
 
    assert(r);
440
 
 
441
 
    /* Called whenever a service has been resolved successfully or timed out */
442
 
 
443
 
    switch (event) {
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)));
446
 
            break;
447
 
 
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);
452
 
            if (ret == 0){
453
 
              exit(EXIT_SUCCESS);
454
 
            } else {
455
 
              exit(EXIT_FAILURE);
456
 
            }
457
 
        }
458
 
    }
459
 
    avahi_s_service_resolver_free(r);
460
 
}
461
 
 
462
 
static void browse_callback(
463
 
    AvahiSServiceBrowser *b,
464
 
    AvahiIfIndex interface,
465
 
    AvahiProtocol protocol,
466
 
    AvahiBrowserEvent event,
467
 
    const char *name,
468
 
    const char *type,
469
 
    const char *domain,
470
 
    AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
471
 
    void* userdata) {
472
 
    
473
 
    AvahiServer *s = userdata;
474
 
    assert(b);
475
 
 
476
 
    /* Called whenever a new services becomes available on the LAN or is removed from the LAN */
477
 
 
478
 
    switch (event) {
479
 
 
480
 
        case AVAHI_BROWSER_FAILURE:
481
 
            
482
 
            fprintf(stderr, "(Browser) %s\n", avahi_strerror(avahi_server_errno(server)));
483
 
            avahi_simple_poll_quit(simple_poll);
484
 
            return;
485
 
 
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. */
491
 
            
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)));
494
 
            
495
 
            break;
496
 
 
497
 
        case AVAHI_BROWSER_REMOVE:
498
 
            break;
499
 
 
500
 
        case AVAHI_BROWSER_ALL_FOR_NOW:
501
 
        case AVAHI_BROWSER_CACHE_EXHAUSTED:
502
 
            break;
503
 
    }
504
 
}
505
 
 
506
 
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
507
 
    AvahiServerConfig config;
 
651
static void resolve_callback(AvahiSServiceResolver *r,
 
652
                             AvahiIfIndex interface,
 
653
                             AVAHI_GCC_UNUSED AvahiProtocol protocol,
 
654
                             AvahiResolverEvent event,
 
655
                             const char *name,
 
656
                             const char *type,
 
657
                             const char *domain,
 
658
                             const char *host_name,
 
659
                             const AvahiAddress *address,
 
660
                             uint16_t port,
 
661
                             AVAHI_GCC_UNUSED AvahiStringList *txt,
 
662
                             AVAHI_GCC_UNUSED AvahiLookupResultFlags
 
663
                             flags,
 
664
                             void* userdata) {
 
665
  mandos_context *mc = userdata;
 
666
  assert(r);
 
667
  
 
668
  /* Called whenever a service has been resolved successfully or
 
669
     timed out */
 
670
  
 
671
  switch (event) {
 
672
  default:
 
673
  case AVAHI_RESOLVER_FAILURE:
 
674
    fprintf(stderr, "(Avahi Resolver) Failed to resolve service '%s'"
 
675
            " of type '%s' in domain '%s': %s\n", name, type, domain,
 
676
            avahi_strerror(avahi_server_errno(mc->server)));
 
677
    break;
 
678
    
 
679
  case AVAHI_RESOLVER_FOUND:
 
680
    {
 
681
      char ip[AVAHI_ADDRESS_STR_MAX];
 
682
      avahi_address_snprint(ip, sizeof(ip), address);
 
683
      if(debug){
 
684
        fprintf(stderr, "Mandos server \"%s\" found on %s (%s, %"
 
685
                PRIu16 ") on port %d\n", name, host_name, ip,
 
686
                interface, port);
 
687
      }
 
688
      int ret = start_mandos_communication(ip, port, interface, mc);
 
689
      if (ret == 0){
 
690
        avahi_simple_poll_quit(mc->simple_poll);
 
691
      }
 
692
    }
 
693
  }
 
694
  avahi_s_service_resolver_free(r);
 
695
}
 
696
 
 
697
static void browse_callback( AvahiSServiceBrowser *b,
 
698
                             AvahiIfIndex interface,
 
699
                             AvahiProtocol protocol,
 
700
                             AvahiBrowserEvent event,
 
701
                             const char *name,
 
702
                             const char *type,
 
703
                             const char *domain,
 
704
                             AVAHI_GCC_UNUSED AvahiLookupResultFlags
 
705
                             flags,
 
706
                             void* userdata) {
 
707
  mandos_context *mc = userdata;
 
708
  assert(b);
 
709
  
 
710
  /* Called whenever a new services becomes available on the LAN or
 
711
     is removed from the LAN */
 
712
  
 
713
  switch (event) {
 
714
  default:
 
715
  case AVAHI_BROWSER_FAILURE:
 
716
    
 
717
    fprintf(stderr, "(Avahi browser) %s\n",
 
718
            avahi_strerror(avahi_server_errno(mc->server)));
 
719
    avahi_simple_poll_quit(mc->simple_poll);
 
720
    return;
 
721
    
 
722
  case AVAHI_BROWSER_NEW:
 
723
    /* We ignore the returned Avahi resolver object. In the callback
 
724
       function we free it. If the Avahi server is terminated before
 
725
       the callback function is called the Avahi server will free the
 
726
       resolver for us. */
 
727
    
 
728
    if (!(avahi_s_service_resolver_new(mc->server, interface,
 
729
                                       protocol, name, type, domain,
 
730
                                       AVAHI_PROTO_INET6, 0,
 
731
                                       resolve_callback, mc)))
 
732
      fprintf(stderr, "Avahi: Failed to resolve service '%s': %s\n",
 
733
              name, avahi_strerror(avahi_server_errno(mc->server)));
 
734
    break;
 
735
    
 
736
  case AVAHI_BROWSER_REMOVE:
 
737
    break;
 
738
    
 
739
  case AVAHI_BROWSER_ALL_FOR_NOW:
 
740
  case AVAHI_BROWSER_CACHE_EXHAUSTED:
 
741
    if(debug){
 
742
      fprintf(stderr, "No Mandos server found, still searching...\n");
 
743
    }
 
744
    break;
 
745
  }
 
746
}
 
747
 
 
748
/* Combines file name and path and returns the malloced new
 
749
   string. some sane checks could/should be added */
 
750
static char *combinepath(const char *first, const char *second){
 
751
  char *tmp;
 
752
  int ret = asprintf(&tmp, "%s/%s", first, second);
 
753
  if(ret < 0){
 
754
    return NULL;
 
755
  }
 
756
  return tmp;
 
757
}
 
758
 
 
759
 
 
760
int main(int argc, char *argv[]){
508
761
    AvahiSServiceBrowser *sb = NULL;
509
762
    int error;
510
 
    int ret = 1;
511
 
 
512
 
    avahi_set_log_function(empty_log);
513
 
    
514
 
    /* Initialize the psuedo-RNG */
515
 
    srand(time(NULL));
516
 
 
517
 
    /* Allocate main loop object */
518
 
    if (!(simple_poll = avahi_simple_poll_new())) {
519
 
        fprintf(stderr, "Failed to create simple poll object.\n");
520
 
        goto fail;
521
 
    }
522
 
 
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;
529
 
 
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; */
534
 
    
535
 
    /* Allocate a new server */
536
 
    server = avahi_server_new(avahi_simple_poll_get(simple_poll), &config, NULL, NULL, &error);
537
 
 
538
 
    /* Free the configuration data */
539
 
    avahi_server_config_free(&config);
540
 
 
541
 
    /* Check wether creating the server object succeeded */
542
 
    if (!server) {
543
 
        fprintf(stderr, "Failed to create server: %s\n", avahi_strerror(error));
544
 
        goto fail;
545
 
    }
546
 
    
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)));
550
 
        goto fail;
 
763
    int ret;
 
764
    int exitcode = EXIT_SUCCESS;
 
765
    const char *interface = "eth0";
 
766
    struct ifreq network;
 
767
    int sd;
 
768
    uid_t uid;
 
769
    gid_t gid;
 
770
    char *connect_to = NULL;
 
771
    AvahiIfIndex if_index = AVAHI_IF_UNSPEC;
 
772
    char *pubkeyfilename = NULL;
 
773
    char *seckeyfilename = NULL;
 
774
    const char *pubkeyname = "pubkey.txt";
 
775
    const char *seckeyname = "seckey.txt";
 
776
    mandos_context mc = { .simple_poll = NULL, .server = NULL,
 
777
                          .dh_bits = 1024, .priority = "SECURE256"
 
778
                          ":!CTYPE-X.509:+CTYPE-OPENPGP" };
 
779
    bool gnutls_initalized = false;
 
780
    
 
781
    {
 
782
      struct argp_option options[] = {
 
783
        { .name = "debug", .key = 128,
 
784
          .doc = "Debug mode", .group = 3 },
 
785
        { .name = "connect", .key = 'c',
 
786
          .arg = "IP",
 
787
          .doc = "Connect directly to a sepcified mandos server",
 
788
          .group = 1 },
 
789
        { .name = "interface", .key = 'i',
 
790
          .arg = "INTERFACE",
 
791
          .doc = "Interface that Avahi will conntect through",
 
792
          .group = 1 },
 
793
        { .name = "keydir", .key = 'd',
 
794
          .arg = "KEYDIR",
 
795
          .doc = "Directory where the openpgp keyring is",
 
796
          .group = 1 },
 
797
        { .name = "seckey", .key = 's',
 
798
          .arg = "SECKEY",
 
799
          .doc = "Secret openpgp key for gnutls authentication",
 
800
          .group = 1 },
 
801
        { .name = "pubkey", .key = 'p',
 
802
          .arg = "PUBKEY",
 
803
          .doc = "Public openpgp key for gnutls authentication",
 
804
          .group = 2 },
 
805
        { .name = "dh-bits", .key = 129,
 
806
          .arg = "BITS",
 
807
          .doc = "dh-bits to use in gnutls communication",
 
808
          .group = 2 },
 
809
        { .name = "priority", .key = 130,
 
810
          .arg = "PRIORITY",
 
811
          .doc = "GNUTLS priority", .group = 1 },
 
812
        { .name = NULL }
 
813
      };
 
814
 
 
815
      
 
816
      error_t parse_opt (int key, char *arg,
 
817
                         struct argp_state *state) {
 
818
        /* Get the INPUT argument from `argp_parse', which we know is
 
819
           a pointer to our plugin list pointer. */
 
820
        switch (key) {
 
821
        case 128:               /* --debug */
 
822
          debug = true;
 
823
          break;
 
824
        case 'c':               /* --connect */
 
825
          connect_to = arg;
 
826
          break;
 
827
        case 'i':               /* --interface */
 
828
          interface = arg;
 
829
          break;
 
830
        case 'd':               /* --keydir */
 
831
          keydir = arg;
 
832
          break;
 
833
        case 's':               /* --seckey */
 
834
          seckeyname = arg;
 
835
          break;
 
836
        case 'p':               /* --pubkey */
 
837
          pubkeyname = arg;
 
838
          break;
 
839
        case 129:               /* --dh-bits */
 
840
          errno = 0;
 
841
          mc.dh_bits = (unsigned int) strtol(arg, NULL, 10);
 
842
          if (errno){
 
843
            perror("strtol");
 
844
            exit(EXIT_FAILURE);
 
845
          }
 
846
          break;
 
847
        case 130:               /* --priority */
 
848
          mc.priority = arg;
 
849
          break;
 
850
        case ARGP_KEY_ARG:
 
851
          argp_usage (state);
 
852
        case ARGP_KEY_END:
 
853
          break;
 
854
        default:
 
855
          return ARGP_ERR_UNKNOWN;
 
856
        }
 
857
        return 0;
 
858
      }
 
859
 
 
860
      struct argp argp = { .options = options, .parser = parse_opt,
 
861
                           .args_doc = "",
 
862
                           .doc = "Mandos client -- Get and decrypt"
 
863
                           " passwords from mandos server" };
 
864
      ret = argp_parse (&argp, argc, argv, 0, 0, NULL);
 
865
      if (ret == ARGP_ERR_UNKNOWN){
 
866
        fprintf(stderr, "Unknown error while parsing arguments\n");
 
867
        exitcode = EXIT_FAILURE;
 
868
        goto end;
 
869
      }
 
870
    }
 
871
      
 
872
    pubkeyfilename = combinepath(keydir, pubkeyname);
 
873
    if (pubkeyfilename == NULL){
 
874
      perror("combinepath");
 
875
      exitcode = EXIT_FAILURE;
 
876
      goto end;
 
877
    }
 
878
    
 
879
    seckeyfilename = combinepath(keydir, seckeyname);
 
880
    if (seckeyfilename == NULL){
 
881
      perror("combinepath");
 
882
      exitcode = EXIT_FAILURE;
 
883
      goto end;
 
884
    }
 
885
 
 
886
    ret = init_gnutls_global(&mc, pubkeyfilename, seckeyfilename);
 
887
    if (ret == -1){
 
888
      fprintf(stderr, "init_gnutls_global failed\n");
 
889
      exitcode = EXIT_FAILURE;
 
890
      goto end;
 
891
    } else {
 
892
      gnutls_initalized = true;
 
893
    }
 
894
    
 
895
    /* If the interface is down, bring it up */
 
896
    {
 
897
      sd = socket(PF_INET6, SOCK_DGRAM, IPPROTO_IP);
 
898
      if(sd < 0) {
 
899
        perror("socket");
 
900
        exitcode = EXIT_FAILURE;
 
901
        goto end;
 
902
      }
 
903
      strcpy(network.ifr_name, interface);
 
904
      ret = ioctl(sd, SIOCGIFFLAGS, &network);
 
905
      if(ret == -1){
 
906
        perror("ioctl SIOCGIFFLAGS");
 
907
        exitcode = EXIT_FAILURE;
 
908
        goto end;
 
909
      }
 
910
      if((network.ifr_flags & IFF_UP) == 0){
 
911
        network.ifr_flags |= IFF_UP;
 
912
        ret = ioctl(sd, SIOCSIFFLAGS, &network);
 
913
        if(ret == -1){
 
914
          perror("ioctl SIOCSIFFLAGS");
 
915
          exitcode = EXIT_FAILURE;
 
916
          goto end;
 
917
        }
 
918
      }
 
919
      close(sd);
 
920
    }
 
921
    
 
922
    uid = getuid();
 
923
    gid = getgid();
 
924
    
 
925
    ret = setuid(uid);
 
926
    if (ret == -1){
 
927
      perror("setuid");
 
928
    }
 
929
    
 
930
    setgid(gid);
 
931
    if (ret == -1){
 
932
      perror("setgid");
 
933
    }
 
934
    
 
935
    if_index = (AvahiIfIndex) if_nametoindex(interface);
 
936
    if(if_index == 0){
 
937
      fprintf(stderr, "No such interface: \"%s\"\n", interface);
 
938
      exit(EXIT_FAILURE);
 
939
    }
 
940
    
 
941
    if(connect_to != NULL){
 
942
      /* Connect directly, do not use Zeroconf */
 
943
      /* (Mainly meant for debugging) */
 
944
      char *address = strrchr(connect_to, ':');
 
945
      if(address == NULL){
 
946
        fprintf(stderr, "No colon in address\n");
 
947
        exitcode = EXIT_FAILURE;
 
948
        goto end;
 
949
      }
 
950
      errno = 0;
 
951
      uint16_t port = (uint16_t) strtol(address+1, NULL, 10);
 
952
      if(errno){
 
953
        perror("Bad port number");
 
954
        exitcode = EXIT_FAILURE;
 
955
        goto end;
 
956
      }
 
957
      *address = '\0';
 
958
      address = connect_to;
 
959
      ret = start_mandos_communication(address, port, if_index, &mc);
 
960
      if(ret < 0){
 
961
        exitcode = EXIT_FAILURE;
 
962
      } else {
 
963
        exitcode = EXIT_SUCCESS;
 
964
      }
 
965
      goto end;
 
966
    }
 
967
    
 
968
    if (not debug){
 
969
      avahi_set_log_function(empty_log);
 
970
    }
 
971
    
 
972
    /* Initialize the pseudo-RNG for Avahi */
 
973
    srand((unsigned int) time(NULL));
 
974
    
 
975
    /* Allocate main Avahi loop object */
 
976
    mc.simple_poll = avahi_simple_poll_new();
 
977
    if (mc.simple_poll == NULL) {
 
978
        fprintf(stderr, "Avahi: Failed to create simple poll"
 
979
                " object.\n");
 
980
        exitcode = EXIT_FAILURE;
 
981
        goto end;
 
982
    }
 
983
 
 
984
    {
 
985
      AvahiServerConfig config;
 
986
      /* Do not publish any local Zeroconf records */
 
987
      avahi_server_config_init(&config);
 
988
      config.publish_hinfo = 0;
 
989
      config.publish_addresses = 0;
 
990
      config.publish_workstation = 0;
 
991
      config.publish_domain = 0;
 
992
 
 
993
      /* Allocate a new server */
 
994
      mc.server = avahi_server_new(avahi_simple_poll_get
 
995
                                   (mc.simple_poll), &config, NULL,
 
996
                                   NULL, &error);
 
997
    
 
998
      /* Free the Avahi configuration data */
 
999
      avahi_server_config_free(&config);
 
1000
    }
 
1001
    
 
1002
    /* Check if creating the Avahi server object succeeded */
 
1003
    if (mc.server == NULL) {
 
1004
        fprintf(stderr, "Failed to create Avahi server: %s\n",
 
1005
                avahi_strerror(error));
 
1006
        exitcode = EXIT_FAILURE;
 
1007
        goto end;
 
1008
    }
 
1009
    
 
1010
    /* Create the Avahi service browser */
 
1011
    sb = avahi_s_service_browser_new(mc.server, if_index,
 
1012
                                     AVAHI_PROTO_INET6,
 
1013
                                     "_mandos._tcp", NULL, 0,
 
1014
                                     browse_callback, &mc);
 
1015
    if (sb == NULL) {
 
1016
        fprintf(stderr, "Failed to create service browser: %s\n",
 
1017
                avahi_strerror(avahi_server_errno(mc.server)));
 
1018
        exitcode = EXIT_FAILURE;
 
1019
        goto end;
551
1020
    }
552
1021
    
553
1022
    /* Run the main loop */
554
 
    avahi_simple_poll_loop(simple_poll);
555
 
    
556
 
    ret = 0;
557
 
    
558
 
fail:
 
1023
 
 
1024
    if (debug){
 
1025
      fprintf(stderr, "Starting Avahi loop search\n");
 
1026
    }
 
1027
    
 
1028
    avahi_simple_poll_loop(mc.simple_poll);
 
1029
    
 
1030
 end:
 
1031
 
 
1032
    if (debug){
 
1033
      fprintf(stderr, "%s exiting\n", argv[0]);
 
1034
    }
559
1035
    
560
1036
    /* Cleanup things */
561
 
    if (sb)
 
1037
    if (sb != NULL)
562
1038
        avahi_s_service_browser_free(sb);
563
1039
    
564
 
    if (server)
565
 
        avahi_server_free(server);
566
 
 
567
 
    if (simple_poll)
568
 
        avahi_simple_poll_free(simple_poll);
569
 
 
570
 
    return ret;
 
1040
    if (mc.server != NULL)
 
1041
        avahi_server_free(mc.server);
 
1042
 
 
1043
    if (mc.simple_poll != NULL)
 
1044
        avahi_simple_poll_free(mc.simple_poll);
 
1045
    free(pubkeyfilename);
 
1046
    free(seckeyfilename);
 
1047
 
 
1048
    if (gnutls_initalized){
 
1049
      gnutls_certificate_free_credentials(mc.cred);
 
1050
      gnutls_global_deinit ();
 
1051
    }
 
1052
    
 
1053
    return exitcode;
571
1054
}