/mandos/release

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

« 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
 
/***
2
 
  This file is part of avahi.
3
 
 
4
 
  avahi is free software; you can redistribute it and/or modify it
5
 
  under the terms of the GNU Lesser General Public License as
6
 
  published by the Free Software Foundation; either version 2.1 of the
7
 
  License, or (at your option) any later version.
8
 
 
9
 
  avahi is distributed in the hope that it will be useful, but WITHOUT
10
 
  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
11
 
  or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
12
 
  Public License for more details.
13
 
 
14
 
  You should have received a copy of the GNU Lesser General Public
15
 
  License along with avahi; if not, write to the Free Software
16
 
  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
17
 
  USA.
18
 
***/
 
1
/*  -*- coding: utf-8 -*- */
 
2
/*
 
3
 * Mandos client - get and decrypt data from a Mandos server
 
4
 *
 
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>.
 
29
 */
19
30
 
 
31
/* Needed by GPGME, specifically gpgme_data_seek() */
20
32
#define _LARGEFILE_SOURCE
21
33
#define _FILE_OFFSET_BITS 64
22
34
 
23
 
#include <stdio.h>
24
 
#include <assert.h>
25
 
#include <stdlib.h>
26
 
#include <time.h>
27
 
#include <net/if.h>             /* if_nametoindex */
28
 
 
 
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_* */
29
76
#include <avahi-core/core.h>
30
77
#include <avahi-core/lookup.h>
31
78
#include <avahi-core/log.h>
33
80
#include <avahi-common/malloc.h>
34
81
#include <avahi-common/error.h>
35
82
 
36
 
//mandos client part
37
 
#include <sys/types.h>          /* socket(), setsockopt(), inet_pton() */
38
 
#include <sys/socket.h>         /* socket(), setsockopt(), struct sockaddr_in6, struct in6_addr, inet_pton() */
39
 
#include <gnutls/gnutls.h>      /* ALL GNUTLS STUFF */
40
 
#include <gnutls/openpgp.h>     /* gnutls with openpgp stuff */
41
 
 
42
 
#include <unistd.h>             /* close() */
43
 
#include <netinet/in.h>
44
 
#include <stdbool.h>            /* true */
45
 
#include <string.h>             /* memset */
46
 
#include <arpa/inet.h>          /* inet_pton() */
47
 
#include <iso646.h>             /* not */
48
 
 
49
 
// gpgme
50
 
#include <errno.h>              /* perror() */
51
 
#include <gpgme.h>
52
 
 
53
 
// getopt long
54
 
#include <getopt.h>
55
 
 
56
 
#ifndef CERT_ROOT
57
 
#define CERT_ROOT "/conf/conf.d/cryptkeyreq/"
58
 
#endif
59
 
#define CERTFILE CERT_ROOT "openpgp-client.txt"
60
 
#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
 
61
99
#define BUFFER_SIZE 256
62
 
#define DH_BITS 1024
63
100
 
64
101
bool debug = false;
65
 
char *interface = "eth0";
 
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>";
66
106
 
 
107
/* Used for passing in values through the Avahi callback functions */
67
108
typedef struct {
68
 
  gnutls_session_t session;
 
109
  AvahiSimplePoll *simple_poll;
 
110
  AvahiServer *server;
69
111
  gnutls_certificate_credentials_t cred;
 
112
  unsigned int dh_bits;
70
113
  gnutls_dh_params_t dh_params;
71
 
} encrypted_session;
72
 
 
73
 
 
74
 
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){
75
142
  gpgme_data_t dh_crypto, dh_plain;
76
143
  gpgme_ctx_t ctx;
77
144
  gpgme_error_t rc;
78
145
  ssize_t ret;
79
 
  size_t new_packet_capacity = 0;
80
 
  size_t new_packet_length = 0;
 
146
  size_t plaintext_capacity = 0;
 
147
  ssize_t plaintext_length = 0;
81
148
  gpgme_engine_info_t engine_info;
82
 
 
 
149
  
83
150
  if (debug){
84
 
    fprintf(stderr, "Attempting to decrypt password from gpg packet\n");
 
151
    fprintf(stderr, "Trying to decrypt OpenPGP data\n");
85
152
  }
86
153
  
87
154
  /* Init GPGME */
88
155
  gpgme_check_version(NULL);
89
 
  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
  }
90
162
  
91
 
  /* Set GPGME home directory */
 
163
  /* Set GPGME home directory for the OpenPGP engine only */
92
164
  rc = gpgme_get_engine_info (&engine_info);
93
165
  if (rc != GPG_ERR_NO_ERROR){
94
166
    fprintf(stderr, "bad gpgme_get_engine_info: %s: %s\n",
104
176
    engine_info = engine_info->next;
105
177
  }
106
178
  if(engine_info == NULL){
107
 
    fprintf(stderr, "Could not set home dir to %s\n", homedir);
 
179
    fprintf(stderr, "Could not set GPGME home dir to %s\n", homedir);
108
180
    return -1;
109
181
  }
110
182
  
111
 
  /* Create new GPGME data buffer from packet buffer */
112
 
  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);
113
186
  if (rc != GPG_ERR_NO_ERROR){
114
187
    fprintf(stderr, "bad gpgme_data_new_from_mem: %s: %s\n",
115
188
            gpgme_strsource(rc), gpgme_strerror(rc));
121
194
  if (rc != GPG_ERR_NO_ERROR){
122
195
    fprintf(stderr, "bad gpgme_data_new: %s: %s\n",
123
196
            gpgme_strsource(rc), gpgme_strerror(rc));
 
197
    gpgme_data_release(dh_crypto);
124
198
    return -1;
125
199
  }
126
200
  
129
203
  if (rc != GPG_ERR_NO_ERROR){
130
204
    fprintf(stderr, "bad gpgme_new: %s: %s\n",
131
205
            gpgme_strsource(rc), gpgme_strerror(rc));
132
 
    return -1;
 
206
    plaintext_length = -1;
 
207
    goto decrypt_end;
133
208
  }
134
209
  
135
 
  /* 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 */
136
212
  rc = gpgme_op_decrypt(ctx, dh_crypto, dh_plain);
137
213
  if (rc != GPG_ERR_NO_ERROR){
138
214
    fprintf(stderr, "bad gpgme_op_decrypt: %s: %s\n",
139
215
            gpgme_strsource(rc), gpgme_strerror(rc));
140
 
    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;
141
246
  }
142
 
 
 
247
  
143
248
  if(debug){
144
 
    fprintf(stderr, "decryption of gpg packet succeeded\n");
145
 
  }
146
 
 
147
 
  if (debug){
148
 
    gpgme_decrypt_result_t result;
149
 
    result = gpgme_op_decrypt_result(ctx);
150
 
    if (result == NULL){
151
 
      fprintf(stderr, "gpgme_op_decrypt_result failed\n");
152
 
    } else {
153
 
      fprintf(stderr, "Unsupported algorithm: %s\n", result->unsupported_algorithm);
154
 
      fprintf(stderr, "Wrong key usage: %d\n", result->wrong_key_usage);
155
 
      if(result->file_name != NULL){
156
 
        fprintf(stderr, "File name: %s\n", result->file_name);
157
 
      }
158
 
      gpgme_recipient_t recipient;
159
 
      recipient = result->recipients;
160
 
      if(recipient){
161
 
        while(recipient != NULL){
162
 
          fprintf(stderr, "Public key algorithm: %s\n",
163
 
                  gpgme_pubkey_algo_name(recipient->pubkey_algo));
164
 
          fprintf(stderr, "Key ID: %s\n", recipient->keyid);
165
 
          fprintf(stderr, "Secret key available: %s\n",
166
 
                  recipient->status == GPG_ERR_NO_SECKEY ? "No" : "Yes");
167
 
          recipient = recipient->next;
168
 
        }
169
 
      }
170
 
    }
171
 
  }
172
 
  
173
 
  /* Delete the GPGME FILE pointer cryptotext data buffer */
174
 
  gpgme_data_release(dh_crypto);
 
249
    fprintf(stderr, "Decryption of OpenPGP data succeeded\n");
 
250
  }
175
251
  
176
252
  /* Seek back to the beginning of the GPGME plaintext data buffer */
177
 
  gpgme_data_seek(dh_plain, 0, SEEK_SET);
178
 
 
179
 
  *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;
180
260
  while(true){
181
 
    if (new_packet_length + BUFFER_SIZE > new_packet_capacity){
182
 
      *new_packet = realloc(*new_packet, new_packet_capacity + BUFFER_SIZE);
183
 
      if (*new_packet == NULL){
184
 
        perror("realloc");
185
 
        return -1;
186
 
      }
187
 
      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;
188
268
    }
189
269
    
190
 
    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);
191
272
    /* Print the data, if any */
192
273
    if (ret == 0){
193
 
      /* If password is empty, then a incorrect error will be printed */
 
274
      /* EOF */
194
275
      break;
195
276
    }
196
277
    if(ret < 0){
197
278
      perror("gpgme_data_read");
198
 
      return -1;
 
279
      plaintext_length = -1;
 
280
      goto decrypt_end;
199
281
    }
200
 
    new_packet_length += ret;
 
282
    plaintext_length += ret;
201
283
  }
202
284
 
203
 
  /* FIXME: check characters before printing to screen so to not print
204
 
     terminal control characters */
205
 
  /*   if(debug){ */
206
 
  /*     fprintf(stderr, "decrypted password is: "); */
207
 
  /*     fwrite(*new_packet, 1, new_packet_length, stderr); */
208
 
  /*     fprintf(stderr, "\n"); */
209
 
  /*   } */
 
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);
210
297
  
211
298
  /* Delete the GPGME plaintext data buffer */
212
299
  gpgme_data_release(dh_plain);
213
 
  return new_packet_length;
 
300
  return plaintext_length;
214
301
}
215
302
 
216
303
static const char * safer_gnutls_strerror (int value) {
217
 
  const char *ret = gnutls_strerror (value);
 
304
  const char *ret = gnutls_strerror (value); /* Spurious warning */
218
305
  if (ret == NULL)
219
306
    ret = "(unknown)";
220
307
  return ret;
221
308
}
222
309
 
223
 
void debuggnutls(int level, const char* string){
224
 
  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);
225
314
}
226
315
 
227
 
int initgnutls(encrypted_session *es){
228
 
  const char *err;
 
316
static int init_gnutls_global(mandos_context *mc,
 
317
                              const char *pubkeyfilename,
 
318
                              const char *seckeyfilename){
229
319
  int ret;
230
 
 
 
320
  
231
321
  if(debug){
232
 
    fprintf(stderr, "Initializing gnutls\n");
 
322
    fprintf(stderr, "Initializing GnuTLS\n");
233
323
  }
234
 
 
235
324
  
236
 
  if ((ret = gnutls_global_init ())
237
 
      != GNUTLS_E_SUCCESS) {
238
 
    fprintf (stderr, "global_init: %s\n", safer_gnutls_strerror(ret));
 
325
  ret = gnutls_global_init();
 
326
  if (ret != GNUTLS_E_SUCCESS) {
 
327
    fprintf (stderr, "GnuTLS global_init: %s\n",
 
328
             safer_gnutls_strerror(ret));
239
329
    return -1;
240
330
  }
241
 
 
 
331
  
242
332
  if (debug){
 
333
    /* "Use a log level over 10 to enable all debugging options."
 
334
     * - GnuTLS manual
 
335
     */
243
336
    gnutls_global_set_log_level(11);
244
337
    gnutls_global_set_log_function(debuggnutls);
245
338
  }
246
339
  
247
 
 
248
 
  /* openpgp credentials */
249
 
  if ((ret = gnutls_certificate_allocate_credentials (&es->cred))
250
 
      != GNUTLS_E_SUCCESS) {
251
 
    fprintf (stderr, "memory error: %s\n", safer_gnutls_strerror(ret));
 
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 ();
252
347
    return -1;
253
348
  }
254
 
 
 
349
  
255
350
  if(debug){
256
 
    fprintf(stderr, "Attempting to use openpgp certificate %s"
257
 
            " and keyfile %s as gnutls credentials\n", CERTFILE, KEYFILE);
 
351
    fprintf(stderr, "Attempting to use OpenPGP certificate %s"
 
352
            " and keyfile %s as GnuTLS credentials\n", pubkeyfilename,
 
353
            seckeyfilename);
258
354
  }
259
 
 
 
355
  
260
356
  ret = gnutls_certificate_set_openpgp_key_file
261
 
    (es->cred, CERTFILE, KEYFILE, GNUTLS_OPENPGP_FMT_BASE64);
262
 
  if (ret != GNUTLS_E_SUCCESS) {
263
 
    fprintf
264
 
      (stderr, "Error[%d] while reading the OpenPGP key pair ('%s', '%s')\n",
265
 
       ret, CERTFILE, KEYFILE);
266
 
    fprintf(stdout, "The Error is: %s\n",
267
 
            safer_gnutls_strerror(ret));
268
 
    return -1;
269
 
  }
270
 
 
271
 
  //Gnutls server initialization
272
 
  if ((ret = gnutls_dh_params_init (&es->dh_params))
273
 
      != GNUTLS_E_SUCCESS) {
274
 
    fprintf (stderr, "Error in dh parameter initialization: %s\n",
275
 
             safer_gnutls_strerror(ret));
276
 
    return -1;
277
 
  }
278
 
 
279
 
  if ((ret = gnutls_dh_params_generate2 (es->dh_params, DH_BITS))
280
 
      != GNUTLS_E_SUCCESS) {
281
 
    fprintf (stderr, "Error in prime generation: %s\n",
282
 
             safer_gnutls_strerror(ret));
283
 
    return -1;
284
 
  }
285
 
 
286
 
  gnutls_certificate_set_dh_params (es->cred, es->dh_params);
287
 
 
288
 
  // Gnutls session creation
289
 
  if ((ret = gnutls_init (&es->session, GNUTLS_SERVER))
290
 
      != GNUTLS_E_SUCCESS){
291
 
    fprintf(stderr, "Error in gnutls session initialization: %s\n",
292
 
            safer_gnutls_strerror(ret));
293
 
  }
294
 
 
295
 
  if ((ret = gnutls_priority_set_direct (es->session, "NORMAL", &err))
296
 
      != GNUTLS_E_SUCCESS) {
297
 
    fprintf(stderr, "Syntax error at: %s\n", err);
298
 
    fprintf(stderr, "Gnutls error: %s\n",
299
 
            safer_gnutls_strerror(ret));
300
 
    return -1;
301
 
  }
302
 
 
303
 
  if ((ret = gnutls_credentials_set
304
 
       (es->session, GNUTLS_CRD_CERTIFICATE, es->cred))
305
 
      != GNUTLS_E_SUCCESS) {
306
 
    fprintf(stderr, "Error setting a credentials set: %s\n",
307
 
            safer_gnutls_strerror(ret));
308
 
    return -1;
309
 
  }
310
 
 
 
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
  
311
425
  /* ignore client certificate if any. */
312
 
  gnutls_certificate_server_set_request (es->session, GNUTLS_CERT_IGNORE);
 
426
  gnutls_certificate_server_set_request (*session,
 
427
                                         GNUTLS_CERT_IGNORE);
313
428
  
314
 
  gnutls_dh_set_prime_bits (es->session, DH_BITS);
 
429
  gnutls_dh_set_prime_bits (*session, mc->dh_bits);
315
430
  
316
431
  return 0;
317
432
}
318
433
 
319
 
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){}
320
437
 
321
 
int start_mandos_communication(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){
322
442
  int ret, tcp_sd;
323
 
  struct sockaddr_in6 to;
324
 
  encrypted_session es;
 
443
  union { struct sockaddr in; struct sockaddr_in6 in6; } to;
325
444
  char *buffer = NULL;
326
445
  char *decrypted_buffer;
327
446
  size_t buffer_length = 0;
328
447
  size_t buffer_capacity = 0;
329
448
  ssize_t decrypted_buffer_size;
 
449
  size_t written;
330
450
  int retval = 0;
331
 
 
 
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
  
332
459
  if(debug){
333
 
    fprintf(stderr, "Setting up a tcp connection to %s\n", ip);
 
460
    fprintf(stderr, "Setting up a tcp connection to %s, port %" PRIu16
 
461
            "\n", ip, port);
334
462
  }
335
463
  
336
464
  tcp_sd = socket(PF_INET6, SOCK_STREAM, 0);
340
468
  }
341
469
 
342
470
  if(debug){
 
471
    if(if_indextoname((unsigned int)if_index, interface) == NULL){
 
472
      perror("if_indextoname");
 
473
      return -1;
 
474
    }
343
475
    fprintf(stderr, "Binding to interface %s\n", interface);
344
476
  }
345
 
 
346
 
  ret = setsockopt(tcp_sd, SOL_SOCKET, SO_BINDTODEVICE, interface, 5);
347
 
  if(tcp_sd < 0) {
348
 
    perror("setsockopt bindtodevice");
349
 
    return -1;
350
 
  }
351
477
  
352
 
  memset(&to,0,sizeof(to));
353
 
  to.sin6_family = AF_INET6;
354
 
  ret = inet_pton(AF_INET6, ip, &to.sin6_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);
355
483
  if (ret < 0 ){
356
484
    perror("inet_pton");
357
485
    return -1;
358
 
  }  
 
486
  }
359
487
  if(ret == 0){
360
488
    fprintf(stderr, "Bad address: %s\n", ip);
361
489
    return -1;
362
490
  }
363
 
  to.sin6_port = htons(port);
364
 
  to.sin6_scope_id = if_nametoindex(interface);
365
 
 
 
491
  to.in6.sin6_port = htons(port); /* Spurious warning */
 
492
  
 
493
  to.in6.sin6_scope_id = (uint32_t)if_index;
 
494
  
366
495
  if(debug){
367
 
    fprintf(stderr, "Connection to: %s\n", ip);
 
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
    }
368
507
  }
369
508
  
370
 
  ret = connect(tcp_sd, (struct sockaddr *) &to, sizeof(to));
 
509
  ret = connect(tcp_sd, &to.in, sizeof(to));
371
510
  if (ret < 0){
372
511
    perror("connect");
373
512
    return -1;
374
513
  }
375
 
  
376
 
  ret = initgnutls (&es);
377
 
  if (ret != 0){
378
 
    retval = -1;
379
 
    return -1;
380
 
  }
381
 
    
382
 
  
383
 
  gnutls_transport_set_ptr (es.session, (gnutls_transport_ptr_t) tcp_sd);
384
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
 
385
539
  if(debug){
386
 
    fprintf(stderr, "Establishing tls session with %s\n", ip);
 
540
    fprintf(stderr, "Establishing TLS session with %s\n", ip);
387
541
  }
 
542
  
 
543
  gnutls_transport_set_ptr (session, (gnutls_transport_ptr_t) tcp_sd);
388
544
 
389
 
  
390
 
  ret = gnutls_handshake (es.session);
 
545
  do{
 
546
    ret = gnutls_handshake (session);
 
547
  } while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
391
548
  
392
549
  if (ret != GNUTLS_E_SUCCESS){
393
 
    fprintf(stderr, "\n*** Handshake failed ***\n");
394
 
    gnutls_perror (ret);
 
550
    if(debug){
 
551
      fprintf(stderr, "*** GnuTLS Handshake failed ***\n");
 
552
      gnutls_perror (ret);
 
553
    }
395
554
    retval = -1;
396
 
    goto exit;
 
555
    goto mandos_end;
397
556
  }
398
 
 
399
 
  //Retrieve gpg packet that contains the wanted password
400
 
 
 
557
  
 
558
  /* Read OpenPGP packet that contains the wanted password */
 
559
  
401
560
  if(debug){
402
 
    fprintf(stderr, "Retrieving pgp encrypted password from %s\n", ip);
 
561
    fprintf(stderr, "Retrieving pgp encrypted password from %s\n",
 
562
            ip);
403
563
  }
404
564
 
405
565
  while(true){
406
 
    if (buffer_length + BUFFER_SIZE > buffer_capacity){
407
 
      buffer = realloc(buffer, buffer_capacity + BUFFER_SIZE);
408
 
      if (buffer == NULL){
409
 
        perror("realloc");
410
 
        goto exit;
411
 
      }
412
 
      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;
413
572
    }
414
573
    
415
 
    ret = gnutls_record_recv
416
 
      (es.session, buffer+buffer_length, BUFFER_SIZE);
 
574
    ret = gnutls_record_recv(session, buffer+buffer_length,
 
575
                             BUFFER_SIZE);
417
576
    if (ret == 0){
418
577
      break;
419
578
    }
423
582
      case GNUTLS_E_AGAIN:
424
583
        break;
425
584
      case GNUTLS_E_REHANDSHAKE:
426
 
        ret = gnutls_handshake (es.session);
 
585
        do{
 
586
          ret = gnutls_handshake (session);
 
587
        } while(ret == GNUTLS_E_AGAIN or ret == GNUTLS_E_INTERRUPTED);
427
588
        if (ret < 0){
428
 
          fprintf(stderr, "\n*** Handshake failed ***\n");
 
589
          fprintf(stderr, "*** GnuTLS Re-handshake failed ***\n");
429
590
          gnutls_perror (ret);
430
591
          retval = -1;
431
 
          goto exit;
 
592
          goto mandos_end;
432
593
        }
433
594
        break;
434
595
      default:
435
 
        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");
436
598
        retval = -1;
437
 
        gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
438
 
        goto exit;
 
599
        gnutls_bye (session, GNUTLS_SHUT_RDWR);
 
600
        goto mandos_end;
439
601
      }
440
602
    } else {
441
 
      buffer_length += ret;
 
603
      buffer_length += (size_t) ret;
442
604
    }
443
605
  }
444
606
  
 
607
  if(debug){
 
608
    fprintf(stderr, "Closing TLS session\n");
 
609
  }
 
610
  
 
611
  gnutls_bye (session, GNUTLS_SHUT_RDWR);
 
612
  
445
613
  if (buffer_length > 0){
446
 
    if ((decrypted_buffer_size = gpg_packet_decrypt(buffer, buffer_length, &decrypted_buffer, CERT_ROOT)) >= 0){
447
 
      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
      }
448
634
      free(decrypted_buffer);
449
635
    } else {
450
636
      retval = -1;
451
637
    }
452
 
  }
453
 
 
454
 
  //shutdown procedure
455
 
 
456
 
  if(debug){
457
 
    fprintf(stderr, "Closing tls session\n");
458
 
  }
459
 
 
 
638
  } else {
 
639
    retval = -1;
 
640
  }
 
641
  
 
642
  /* Shutdown procedure */
 
643
  
 
644
 mandos_end:
460
645
  free(buffer);
461
 
  gnutls_bye (es.session, GNUTLS_SHUT_RDWR);
462
 
 exit:
463
646
  close(tcp_sd);
464
 
  gnutls_deinit (es.session);
465
 
  gnutls_certificate_free_credentials (es.cred);
466
 
  gnutls_global_deinit ();
 
647
  gnutls_deinit (session);
467
648
  return retval;
468
649
}
469
650
 
470
 
static AvahiSimplePoll *simple_poll = NULL;
471
 
static AvahiServer *server = NULL;
472
 
 
473
 
static void resolve_callback(
474
 
    AvahiSServiceResolver *r,
475
 
    AVAHI_GCC_UNUSED AvahiIfIndex interface,
476
 
    AVAHI_GCC_UNUSED AvahiProtocol protocol,
477
 
    AvahiResolverEvent event,
478
 
    const char *name,
479
 
    const char *type,
480
 
    const char *domain,
481
 
    const char *host_name,
482
 
    const AvahiAddress *address,
483
 
    uint16_t port,
484
 
    AvahiStringList *txt,
485
 
    AvahiLookupResultFlags flags,
486
 
    AVAHI_GCC_UNUSED void* userdata) {
487
 
    
488
 
    assert(r);
489
 
 
490
 
    /* Called whenever a service has been resolved successfully or timed out */
491
 
 
492
 
    switch (event) {
493
 
        case AVAHI_RESOLVER_FAILURE:
494
 
            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)));
495
 
            break;
496
 
 
497
 
        case AVAHI_RESOLVER_FOUND: {
498
 
          char ip[AVAHI_ADDRESS_STR_MAX];
499
 
            avahi_address_snprint(ip, sizeof(ip), address);
500
 
            if(debug){
501
 
              fprintf(stderr, "Mandos server found at %s on port %d\n", ip, port);
502
 
            }
503
 
            int ret = start_mandos_communication(ip, port);
504
 
            if (ret == 0){
505
 
              exit(EXIT_SUCCESS);
506
 
            } else {
507
 
              exit(EXIT_FAILURE);
508
 
            }
509
 
        }
510
 
    }
511
 
    avahi_s_service_resolver_free(r);
512
 
}
513
 
 
514
 
static void browse_callback(
515
 
    AvahiSServiceBrowser *b,
516
 
    AvahiIfIndex interface,
517
 
    AvahiProtocol protocol,
518
 
    AvahiBrowserEvent event,
519
 
    const char *name,
520
 
    const char *type,
521
 
    const char *domain,
522
 
    AVAHI_GCC_UNUSED AvahiLookupResultFlags flags,
523
 
    void* userdata) {
524
 
    
525
 
    AvahiServer *s = userdata;
526
 
    assert(b);
527
 
 
528
 
    /* Called whenever a new services becomes available on the LAN or is removed from the LAN */
529
 
 
530
 
    switch (event) {
531
 
 
532
 
        case AVAHI_BROWSER_FAILURE:
533
 
            
534
 
            fprintf(stderr, "(Browser) %s\n", avahi_strerror(avahi_server_errno(server)));
535
 
            avahi_simple_poll_quit(simple_poll);
536
 
            return;
537
 
 
538
 
        case AVAHI_BROWSER_NEW:
539
 
            /* We ignore the returned resolver object. In the callback
540
 
               function we free it. If the server is terminated before
541
 
               the callback function is called the server will free
542
 
               the resolver for us. */
543
 
            
544
 
            if (!(avahi_s_service_resolver_new(s, interface, protocol, name, type, domain, AVAHI_PROTO_INET6, 0, resolve_callback, s)))
545
 
                fprintf(stderr, "Failed to resolve service '%s': %s\n", name, avahi_strerror(avahi_server_errno(s)));
546
 
            
547
 
            break;
548
 
 
549
 
        case AVAHI_BROWSER_REMOVE:
550
 
            break;
551
 
 
552
 
        case AVAHI_BROWSER_ALL_FOR_NOW:
553
 
        case AVAHI_BROWSER_CACHE_EXHAUSTED:
554
 
            break;
555
 
    }
556
 
}
557
 
 
558
 
int main(AVAHI_GCC_UNUSED int argc, AVAHI_GCC_UNUSED char*argv[]) {
559
 
    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[]){
560
761
    AvahiSServiceBrowser *sb = NULL;
561
762
    int error;
562
763
    int ret;
563
 
    int returncode = EXIT_SUCCESS;
564
 
 
565
 
    while (true){
566
 
      static struct option long_options[] = {
567
 
        {"debug", no_argument, (int *)&debug, 1},
568
 
        {"interface", required_argument, 0, 'i'},
569
 
        {0, 0, 0, 0} };
570
 
 
571
 
      int option_index = 0;
572
 
      ret = getopt_long (argc, argv, "i:", long_options, &option_index);
573
 
 
574
 
      if (ret == -1){
575
 
        break;
576
 
      }
577
 
      
578
 
      switch(ret){
579
 
      case 0:
580
 
        break;
581
 
      case 'i':
582
 
        interface = optarg;
583
 
        break;
584
 
      default:
585
 
        exit(EXIT_FAILURE);
586
 
      }
 
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;
587
966
    }
588
967
    
589
968
    if (not debug){
590
969
      avahi_set_log_function(empty_log);
591
970
    }
592
971
    
593
 
    /* Initialize the psuedo-RNG */
594
 
    srand(time(NULL));
595
 
 
596
 
    /* Allocate main loop object */
597
 
    if (!(simple_poll = avahi_simple_poll_new())) {
598
 
        fprintf(stderr, "Failed to create simple poll object.\n");
599
 
        
600
 
        goto exit;
601
 
    }
602
 
 
603
 
    /* Do not publish any local records */
604
 
    avahi_server_config_init(&config);
605
 
    config.publish_hinfo = 0;
606
 
    config.publish_addresses = 0;
607
 
    config.publish_workstation = 0;
608
 
    config.publish_domain = 0;
609
 
 
610
 
    /* Allocate a new server */
611
 
    server = avahi_server_new(avahi_simple_poll_get(simple_poll), &config, NULL, NULL, &error);
612
 
 
613
 
    /* Free the configuration data */
614
 
    avahi_server_config_free(&config);
615
 
 
616
 
    /* Check if creating the server object succeeded */
617
 
    if (!server) {
618
 
        fprintf(stderr, "Failed to create server: %s\n", avahi_strerror(error));
619
 
        returncode = EXIT_FAILURE;
620
 
        goto exit;
621
 
    }
622
 
    
623
 
    /* Create the service browser */
624
 
    if (!(sb = avahi_s_service_browser_new(server, if_nametoindex("eth0"), AVAHI_PROTO_INET6, "_mandos._tcp", NULL, 0, browse_callback, server))) {
625
 
        fprintf(stderr, "Failed to create service browser: %s\n", avahi_strerror(avahi_server_errno(server)));
626
 
        returncode = EXIT_FAILURE;
627
 
        goto exit;
 
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;
628
1020
    }
629
1021
    
630
1022
    /* Run the main loop */
631
1023
 
632
1024
    if (debug){
633
 
      fprintf(stderr, "Starting avahi loop search\n");
 
1025
      fprintf(stderr, "Starting Avahi loop search\n");
634
1026
    }
635
1027
    
636
 
    avahi_simple_poll_loop(simple_poll);
 
1028
    avahi_simple_poll_loop(mc.simple_poll);
637
1029
    
638
 
exit:
 
1030
 end:
639
1031
 
640
1032
    if (debug){
641
1033
      fprintf(stderr, "%s exiting\n", argv[0]);
642
1034
    }
643
1035
    
644
1036
    /* Cleanup things */
645
 
    if (sb)
 
1037
    if (sb != NULL)
646
1038
        avahi_s_service_browser_free(sb);
647
1039
    
648
 
    if (server)
649
 
        avahi_server_free(server);
650
 
 
651
 
    if (simple_poll)
652
 
        avahi_simple_poll_free(simple_poll);
653
 
 
654
 
    return returncode;
 
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;
655
1054
}