2016-06-29 12:00:22 +02:00
|
|
|
|
/* gpg-wks-client.c - A client for the Web Key Service protocols.
|
|
|
|
|
* Copyright (C) 2016 Werner Koch
|
2017-06-19 12:37:52 +02:00
|
|
|
|
* Copyright (C) 2016 Bundesamt für Sicherheit in der Informationstechnik
|
2016-06-29 12:00:22 +02:00
|
|
|
|
*
|
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
|
*
|
2017-06-19 12:37:52 +02:00
|
|
|
|
* This file is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU Lesser General Public License as
|
|
|
|
|
* published by the Free Software Foundation; either version 2.1 of
|
|
|
|
|
* the License, or (at your option) any later version.
|
2016-06-29 12:00:22 +02:00
|
|
|
|
*
|
2017-06-19 12:37:52 +02:00
|
|
|
|
* This file is distributed in the hope that it will be useful,
|
2016-06-29 12:00:22 +02:00
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2017-06-19 12:37:52 +02:00
|
|
|
|
* GNU Lesser General Public License for more details.
|
2016-06-29 12:00:22 +02:00
|
|
|
|
*
|
2017-06-19 12:37:52 +02:00
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
2016-11-05 12:02:19 +01:00
|
|
|
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
2016-06-29 12:00:22 +02:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
2017-03-07 12:21:23 +01:00
|
|
|
|
#include "../common/util.h"
|
|
|
|
|
#include "../common/status.h"
|
|
|
|
|
#include "../common/i18n.h"
|
|
|
|
|
#include "../common/sysutils.h"
|
|
|
|
|
#include "../common/init.h"
|
|
|
|
|
#include "../common/asshelp.h"
|
|
|
|
|
#include "../common/userids.h"
|
|
|
|
|
#include "../common/ccparray.h"
|
|
|
|
|
#include "../common/exectool.h"
|
|
|
|
|
#include "../common/mbox-util.h"
|
|
|
|
|
#include "../common/name-value.h"
|
2016-06-29 12:00:22 +02:00
|
|
|
|
#include "call-dirmngr.h"
|
|
|
|
|
#include "mime-maker.h"
|
2016-07-03 00:41:30 +02:00
|
|
|
|
#include "send-mail.h"
|
2016-06-29 12:00:22 +02:00
|
|
|
|
#include "gpg-wks.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Constants to identify the commands and options. */
|
|
|
|
|
enum cmd_and_opt_values
|
|
|
|
|
{
|
|
|
|
|
aNull = 0,
|
|
|
|
|
|
|
|
|
|
oQuiet = 'q',
|
|
|
|
|
oVerbose = 'v',
|
2016-07-03 00:41:30 +02:00
|
|
|
|
oOutput = 'o',
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
oDebug = 500,
|
|
|
|
|
|
2016-08-24 15:48:21 +02:00
|
|
|
|
aSupported,
|
2016-12-08 16:57:21 +01:00
|
|
|
|
aCheck,
|
2016-07-03 00:41:30 +02:00
|
|
|
|
aCreate,
|
2016-06-29 12:00:22 +02:00
|
|
|
|
aReceive,
|
2016-07-04 11:28:58 +02:00
|
|
|
|
aRead,
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
oGpgProgram,
|
2016-07-03 00:41:30 +02:00
|
|
|
|
oSend,
|
2016-10-05 11:51:32 +02:00
|
|
|
|
oFakeSubmissionAddr,
|
2016-12-08 17:55:36 +01:00
|
|
|
|
oStatusFD,
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
oDummy
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The list of commands and options. */
|
|
|
|
|
static ARGPARSE_OPTS opts[] = {
|
|
|
|
|
ARGPARSE_group (300, ("@Commands:\n ")),
|
|
|
|
|
|
2016-08-24 15:48:21 +02:00
|
|
|
|
ARGPARSE_c (aSupported, "supported",
|
|
|
|
|
("check whether provider supports WKS")),
|
2016-12-08 16:57:21 +01:00
|
|
|
|
ARGPARSE_c (aCheck, "check",
|
|
|
|
|
("check whether a key is available")),
|
2016-07-03 00:41:30 +02:00
|
|
|
|
ARGPARSE_c (aCreate, "create",
|
|
|
|
|
("create a publication request")),
|
2016-06-29 12:00:22 +02:00
|
|
|
|
ARGPARSE_c (aReceive, "receive",
|
2016-07-04 11:28:58 +02:00
|
|
|
|
("receive a MIME confirmation request")),
|
|
|
|
|
ARGPARSE_c (aRead, "read",
|
|
|
|
|
("receive a plain text confirmation request")),
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
ARGPARSE_group (301, ("@\nOptions:\n ")),
|
|
|
|
|
|
|
|
|
|
ARGPARSE_s_n (oVerbose, "verbose", ("verbose")),
|
|
|
|
|
ARGPARSE_s_n (oQuiet, "quiet", ("be somewhat more quiet")),
|
|
|
|
|
ARGPARSE_s_s (oDebug, "debug", "@"),
|
|
|
|
|
ARGPARSE_s_s (oGpgProgram, "gpg", "@"),
|
2016-07-03 00:41:30 +02:00
|
|
|
|
ARGPARSE_s_n (oSend, "send", "send the mail using sendmail"),
|
|
|
|
|
ARGPARSE_s_s (oOutput, "output", "|FILE|write the mail to FILE"),
|
2016-12-08 17:55:36 +01:00
|
|
|
|
ARGPARSE_s_i (oStatusFD, "status-fd", N_("|FD|write status info to this FD")),
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
2016-10-05 11:51:32 +02:00
|
|
|
|
ARGPARSE_s_s (oFakeSubmissionAddr, "fake-submission-addr", "@"),
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
ARGPARSE_end ()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The list of supported debug flags. */
|
|
|
|
|
static struct debug_flags_s debug_flags [] =
|
|
|
|
|
{
|
2016-09-29 17:55:32 +02:00
|
|
|
|
{ DBG_MIME_VALUE , "mime" },
|
|
|
|
|
{ DBG_PARSER_VALUE , "parser" },
|
2016-06-29 12:00:22 +02:00
|
|
|
|
{ DBG_CRYPTO_VALUE , "crypto" },
|
|
|
|
|
{ DBG_MEMORY_VALUE , "memory" },
|
|
|
|
|
{ DBG_MEMSTAT_VALUE, "memstat" },
|
|
|
|
|
{ DBG_IPC_VALUE , "ipc" },
|
|
|
|
|
{ DBG_EXTPROG_VALUE, "extprog" },
|
|
|
|
|
{ 0, NULL }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2016-10-05 11:51:32 +02:00
|
|
|
|
|
|
|
|
|
/* Value of the option --fake-submission-addr. */
|
|
|
|
|
const char *fake_submission_addr;
|
|
|
|
|
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static void wrong_args (const char *text) GPGRT_ATTR_NORETURN;
|
2016-08-24 15:48:21 +02:00
|
|
|
|
static gpg_error_t command_supported (char *userid);
|
2016-12-08 16:57:21 +01:00
|
|
|
|
static gpg_error_t command_check (char *userid);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static gpg_error_t command_send (const char *fingerprint, char *userid);
|
2016-10-05 14:38:37 +02:00
|
|
|
|
static gpg_error_t encrypt_response (estream_t *r_output, estream_t input,
|
2016-11-07 14:04:47 +01:00
|
|
|
|
const char *addrspec,
|
|
|
|
|
const char *fingerprint);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
static gpg_error_t read_confirmation_request (estream_t msg);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static gpg_error_t command_receive_cb (void *opaque,
|
2016-09-29 17:55:32 +02:00
|
|
|
|
const char *mediatype, estream_t fp,
|
|
|
|
|
unsigned int flags);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-02-20 22:19:50 +01:00
|
|
|
|
/* Print usage information and provide strings for help. */
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static const char *
|
|
|
|
|
my_strusage( int level )
|
|
|
|
|
{
|
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
|
|
switch (level)
|
|
|
|
|
{
|
2017-07-26 17:45:28 +02:00
|
|
|
|
case 11: p = "gpg-wks-client"; break;
|
|
|
|
|
case 12: p = "@GNUPG@"; break;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
case 13: p = VERSION; break;
|
|
|
|
|
case 17: p = PRINTABLE_OS_NAME; break;
|
|
|
|
|
case 19: p = ("Please report bugs to <@EMAIL@>.\n"); break;
|
|
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
|
case 40:
|
2016-07-03 00:41:30 +02:00
|
|
|
|
p = ("Usage: gpg-wks-client [command] [options] [args] (-h for help)");
|
2016-06-29 12:00:22 +02:00
|
|
|
|
break;
|
|
|
|
|
case 41:
|
2016-07-03 00:41:30 +02:00
|
|
|
|
p = ("Syntax: gpg-wks-client [command] [options] [args]\n"
|
2016-06-29 12:00:22 +02:00
|
|
|
|
"Client for the Web Key Service\n");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default: p = NULL; break;
|
|
|
|
|
}
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wrong_args (const char *text)
|
|
|
|
|
{
|
|
|
|
|
es_fprintf (es_stderr, _("usage: %s [options] %s\n"), strusage (11), text);
|
|
|
|
|
exit (2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Command line parsing. */
|
|
|
|
|
static enum cmd_and_opt_values
|
|
|
|
|
parse_arguments (ARGPARSE_ARGS *pargs, ARGPARSE_OPTS *popts)
|
|
|
|
|
{
|
|
|
|
|
enum cmd_and_opt_values cmd = 0;
|
|
|
|
|
int no_more_options = 0;
|
|
|
|
|
|
|
|
|
|
while (!no_more_options && optfile_parse (NULL, NULL, NULL, pargs, popts))
|
|
|
|
|
{
|
|
|
|
|
switch (pargs->r_opt)
|
|
|
|
|
{
|
|
|
|
|
case oQuiet: opt.quiet = 1; break;
|
|
|
|
|
case oVerbose: opt.verbose++; break;
|
|
|
|
|
case oDebug:
|
|
|
|
|
if (parse_debug_flag (pargs->r.ret_str, &opt.debug, debug_flags))
|
|
|
|
|
{
|
|
|
|
|
pargs->r_opt = ARGPARSE_INVALID_ARG;
|
|
|
|
|
pargs->err = ARGPARSE_PRINT_ERROR;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case oGpgProgram:
|
|
|
|
|
opt.gpg_program = pargs->r.ret_str;
|
|
|
|
|
break;
|
2016-07-03 00:41:30 +02:00
|
|
|
|
case oSend:
|
|
|
|
|
opt.use_sendmail = 1;
|
|
|
|
|
break;
|
|
|
|
|
case oOutput:
|
|
|
|
|
opt.output = pargs->r.ret_str;
|
|
|
|
|
break;
|
2016-10-05 11:51:32 +02:00
|
|
|
|
case oFakeSubmissionAddr:
|
|
|
|
|
fake_submission_addr = pargs->r.ret_str;
|
|
|
|
|
break;
|
2016-12-08 17:55:36 +01:00
|
|
|
|
case oStatusFD:
|
|
|
|
|
wks_set_status_fd (translate_sys2libc_fd_int (pargs->r.ret_int, 1));
|
|
|
|
|
break;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
2016-08-24 15:48:21 +02:00
|
|
|
|
case aSupported:
|
2016-07-03 00:41:30 +02:00
|
|
|
|
case aCreate:
|
2016-06-29 12:00:22 +02:00
|
|
|
|
case aReceive:
|
2016-07-04 11:28:58 +02:00
|
|
|
|
case aRead:
|
2016-12-08 16:57:21 +01:00
|
|
|
|
case aCheck:
|
2016-06-29 12:00:22 +02:00
|
|
|
|
cmd = pargs->r_opt;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default: pargs->err = 2; break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cmd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* gpg-wks-client main. */
|
|
|
|
|
int
|
|
|
|
|
main (int argc, char **argv)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
ARGPARSE_ARGS pargs;
|
|
|
|
|
enum cmd_and_opt_values cmd;
|
|
|
|
|
|
|
|
|
|
gnupg_reopen_std ("gpg-wks-client");
|
|
|
|
|
set_strusage (my_strusage);
|
|
|
|
|
log_set_prefix ("gpg-wks-client", GPGRT_LOG_WITH_PREFIX);
|
|
|
|
|
|
|
|
|
|
/* Make sure that our subsystems are ready. */
|
|
|
|
|
i18n_init();
|
|
|
|
|
init_common_subsystems (&argc, &argv);
|
|
|
|
|
|
|
|
|
|
assuan_set_gpg_err_source (GPG_ERR_SOURCE_DEFAULT);
|
2016-09-05 10:55:10 +02:00
|
|
|
|
setup_libassuan_logging (&opt.debug, NULL);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
/* Parse the command line. */
|
|
|
|
|
pargs.argc = &argc;
|
|
|
|
|
pargs.argv = &argv;
|
|
|
|
|
pargs.flags = ARGPARSE_FLAG_KEEP;
|
|
|
|
|
cmd = parse_arguments (&pargs, opts);
|
|
|
|
|
|
|
|
|
|
if (log_get_errorcount (0))
|
|
|
|
|
exit (2);
|
|
|
|
|
|
|
|
|
|
/* Print a warning if an argument looks like an option. */
|
|
|
|
|
if (!opt.quiet && !(pargs.flags & ARGPARSE_FLAG_STOP_SEEN))
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i=0; i < argc; i++)
|
|
|
|
|
if (argv[i][0] == '-' && argv[i][1] == '-')
|
|
|
|
|
log_info (("NOTE: '%s' is not considered an option\n"), argv[i]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set defaults for non given options. */
|
|
|
|
|
if (!opt.gpg_program)
|
|
|
|
|
opt.gpg_program = gnupg_module_name (GNUPG_MODULE_NAME_GPG);
|
|
|
|
|
|
|
|
|
|
/* Tell call-dirmngr what options we want. */
|
|
|
|
|
set_dirmngr_options (opt.verbose, (opt.debug & DBG_IPC_VALUE), 1);
|
|
|
|
|
|
|
|
|
|
/* Run the selected command. */
|
|
|
|
|
switch (cmd)
|
|
|
|
|
{
|
2016-08-24 15:48:21 +02:00
|
|
|
|
case aSupported:
|
|
|
|
|
if (argc != 1)
|
|
|
|
|
wrong_args ("--supported USER-ID");
|
|
|
|
|
err = command_supported (argv[0]);
|
|
|
|
|
if (err && gpg_err_code (err) != GPG_ERR_FALSE)
|
|
|
|
|
log_error ("checking support failed: %s\n", gpg_strerror (err));
|
|
|
|
|
break;
|
|
|
|
|
|
2016-07-03 00:41:30 +02:00
|
|
|
|
case aCreate:
|
2016-06-29 12:00:22 +02:00
|
|
|
|
if (argc != 2)
|
2016-07-03 00:41:30 +02:00
|
|
|
|
wrong_args ("--create FINGERPRINT USER-ID");
|
2016-06-29 12:00:22 +02:00
|
|
|
|
err = command_send (argv[0], argv[1]);
|
|
|
|
|
if (err)
|
2016-07-03 00:41:30 +02:00
|
|
|
|
log_error ("creating request failed: %s\n", gpg_strerror (err));
|
2016-06-29 12:00:22 +02:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case aReceive:
|
|
|
|
|
if (argc)
|
2016-07-04 11:28:58 +02:00
|
|
|
|
wrong_args ("--receive < MIME-DATA");
|
2016-06-29 12:00:22 +02:00
|
|
|
|
err = wks_receive (es_stdin, command_receive_cb, NULL);
|
|
|
|
|
if (err)
|
2016-07-03 00:41:30 +02:00
|
|
|
|
log_error ("processing mail failed: %s\n", gpg_strerror (err));
|
2016-06-29 12:00:22 +02:00
|
|
|
|
break;
|
|
|
|
|
|
2016-07-04 11:28:58 +02:00
|
|
|
|
case aRead:
|
|
|
|
|
if (argc)
|
|
|
|
|
wrong_args ("--read < WKS-DATA");
|
2016-09-29 17:55:32 +02:00
|
|
|
|
err = read_confirmation_request (es_stdin);
|
2016-07-04 11:28:58 +02:00
|
|
|
|
if (err)
|
|
|
|
|
log_error ("processing mail failed: %s\n", gpg_strerror (err));
|
|
|
|
|
break;
|
|
|
|
|
|
2016-12-08 16:57:21 +01:00
|
|
|
|
case aCheck:
|
|
|
|
|
if (argc != 1)
|
|
|
|
|
wrong_args ("--check USER-ID");
|
2016-12-08 17:55:36 +01:00
|
|
|
|
err = command_check (argv[0]);
|
2016-12-08 16:57:21 +01:00
|
|
|
|
break;
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
default:
|
|
|
|
|
usage (1);
|
2016-12-08 17:55:36 +01:00
|
|
|
|
err = 0;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-08 17:55:36 +01:00
|
|
|
|
if (err)
|
|
|
|
|
wks_write_status (STATUS_FAILURE, "- %u", err);
|
|
|
|
|
else if (log_get_errorcount (0))
|
|
|
|
|
wks_write_status (STATUS_FAILURE, "- %u", GPG_ERR_GENERAL);
|
|
|
|
|
else
|
|
|
|
|
wks_write_status (STATUS_SUCCESS, NULL);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
return log_get_errorcount (0)? 1:0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct get_key_status_parm_s
|
|
|
|
|
{
|
|
|
|
|
const char *fpr;
|
|
|
|
|
int found;
|
|
|
|
|
int count;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
get_key_status_cb (void *opaque, const char *keyword, char *args)
|
|
|
|
|
{
|
|
|
|
|
struct get_key_status_parm_s *parm = opaque;
|
|
|
|
|
|
|
|
|
|
/*log_debug ("%s: %s\n", keyword, args);*/
|
|
|
|
|
if (!strcmp (keyword, "EXPORTED"))
|
|
|
|
|
{
|
|
|
|
|
parm->count++;
|
|
|
|
|
if (!ascii_strcasecmp (args, parm->fpr))
|
|
|
|
|
parm->found = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Get a key by fingerprint from gpg's keyring and make sure that the
|
|
|
|
|
* mail address ADDRSPEC is included in the key. The key is returned
|
|
|
|
|
* as a new memory stream at R_KEY.
|
|
|
|
|
*
|
|
|
|
|
* Fixme: After we have implemented import and export filters for gpg
|
|
|
|
|
* this function shall only return a key with just this user id. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
get_key (estream_t *r_key, const char *fingerprint, const char *addrspec)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
ccparray_t ccp;
|
2016-07-03 10:27:39 +02:00
|
|
|
|
const char **argv = NULL;
|
|
|
|
|
estream_t key = NULL;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
struct get_key_status_parm_s parm;
|
2016-07-03 10:27:39 +02:00
|
|
|
|
char *filterexp = NULL;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
memset (&parm, 0, sizeof parm);
|
|
|
|
|
|
|
|
|
|
*r_key = NULL;
|
|
|
|
|
|
|
|
|
|
key = es_fopenmem (0, "w+b");
|
|
|
|
|
if (!key)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error allocating memory buffer: %s\n", gpg_strerror (err));
|
2016-07-03 10:27:39 +02:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2017-02-23 20:10:59 +01:00
|
|
|
|
|
2016-10-05 14:38:37 +02:00
|
|
|
|
/* Prefix the key with the MIME content type. */
|
|
|
|
|
es_fputs ("Content-Type: application/pgp-keys\n"
|
|
|
|
|
"\n", key);
|
2016-07-03 10:27:39 +02:00
|
|
|
|
|
|
|
|
|
filterexp = es_bsprintf ("keep-uid=mbox = %s", addrspec);
|
|
|
|
|
if (!filterexp)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error allocating memory buffer: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ccparray_init (&ccp, 0);
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, "--no-options");
|
|
|
|
|
if (!opt.verbose)
|
|
|
|
|
ccparray_put (&ccp, "--quiet");
|
|
|
|
|
else if (opt.verbose > 1)
|
|
|
|
|
ccparray_put (&ccp, "--verbose");
|
|
|
|
|
ccparray_put (&ccp, "--batch");
|
|
|
|
|
ccparray_put (&ccp, "--status-fd=2");
|
|
|
|
|
ccparray_put (&ccp, "--always-trust");
|
|
|
|
|
ccparray_put (&ccp, "--armor");
|
|
|
|
|
ccparray_put (&ccp, "--export-options=export-minimal");
|
2016-07-03 10:27:39 +02:00
|
|
|
|
ccparray_put (&ccp, "--export-filter");
|
|
|
|
|
ccparray_put (&ccp, filterexp);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
ccparray_put (&ccp, "--export");
|
|
|
|
|
ccparray_put (&ccp, "--");
|
|
|
|
|
ccparray_put (&ccp, fingerprint);
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, NULL);
|
|
|
|
|
argv = ccparray_get (&ccp, NULL);
|
|
|
|
|
if (!argv)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
parm.fpr = fingerprint;
|
|
|
|
|
err = gnupg_exec_tool_stream (opt.gpg_program, argv, NULL,
|
|
|
|
|
NULL, key,
|
|
|
|
|
get_key_status_cb, &parm);
|
|
|
|
|
if (!err && parm.count > 1)
|
|
|
|
|
err = gpg_error (GPG_ERR_TOO_MANY);
|
|
|
|
|
else if (!err && !parm.found)
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("export failed: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
es_rewind (key);
|
|
|
|
|
*r_key = key;
|
|
|
|
|
key = NULL;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
es_fclose (key);
|
|
|
|
|
xfree (argv);
|
2016-07-03 10:27:39 +02:00
|
|
|
|
xfree (filterexp);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-09-29 17:55:32 +02:00
|
|
|
|
|
2017-02-23 20:10:59 +01:00
|
|
|
|
struct decrypt_stream_parm_s
|
|
|
|
|
{
|
|
|
|
|
char *fpr;
|
|
|
|
|
char *mainfpr;
|
|
|
|
|
int otrust;
|
|
|
|
|
};
|
|
|
|
|
|
2016-09-29 17:55:32 +02:00
|
|
|
|
static void
|
|
|
|
|
decrypt_stream_status_cb (void *opaque, const char *keyword, char *args)
|
|
|
|
|
{
|
2017-02-23 20:10:59 +01:00
|
|
|
|
struct decrypt_stream_parm_s *decinfo = opaque;
|
2016-09-29 17:55:32 +02:00
|
|
|
|
|
|
|
|
|
if (DBG_CRYPTO)
|
|
|
|
|
log_debug ("gpg status: %s %s\n", keyword, args);
|
2017-02-23 20:10:59 +01:00
|
|
|
|
if (!strcmp (keyword, "DECRYPTION_KEY") && !decinfo->fpr)
|
|
|
|
|
{
|
|
|
|
|
char *fields[3];
|
2016-09-29 17:55:32 +02:00
|
|
|
|
|
2017-02-23 20:10:59 +01:00
|
|
|
|
if (split_fields (args, fields, DIM (fields)) >= 3)
|
|
|
|
|
{
|
|
|
|
|
decinfo->fpr = xstrdup (fields[0]);
|
|
|
|
|
decinfo->mainfpr = xstrdup (fields[1]);
|
|
|
|
|
decinfo->otrust = *fields[2];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-09-29 17:55:32 +02:00
|
|
|
|
|
|
|
|
|
/* Decrypt the INPUT stream to a new stream which is stored at success
|
|
|
|
|
* at R_OUTPUT. */
|
|
|
|
|
static gpg_error_t
|
2017-02-23 20:10:59 +01:00
|
|
|
|
decrypt_stream (estream_t *r_output, struct decrypt_stream_parm_s *decinfo,
|
|
|
|
|
estream_t input)
|
2016-09-29 17:55:32 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
ccparray_t ccp;
|
|
|
|
|
const char **argv;
|
|
|
|
|
estream_t output;
|
|
|
|
|
|
|
|
|
|
*r_output = NULL;
|
2017-02-23 20:10:59 +01:00
|
|
|
|
memset (decinfo, 0, sizeof *decinfo);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
|
|
|
|
|
output = es_fopenmem (0, "w+b");
|
|
|
|
|
if (!output)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error allocating memory buffer: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ccparray_init (&ccp, 0);
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, "--no-options");
|
|
|
|
|
/* We limit the output to 64 KiB to avoid DoS using compression
|
|
|
|
|
* tricks. A regular client will anyway only send a minimal key;
|
|
|
|
|
* that is one w/o key signatures and attribute packets. */
|
|
|
|
|
ccparray_put (&ccp, "--max-output=0x10000");
|
|
|
|
|
if (!opt.verbose)
|
|
|
|
|
ccparray_put (&ccp, "--quiet");
|
|
|
|
|
else if (opt.verbose > 1)
|
|
|
|
|
ccparray_put (&ccp, "--verbose");
|
|
|
|
|
ccparray_put (&ccp, "--batch");
|
|
|
|
|
ccparray_put (&ccp, "--status-fd=2");
|
|
|
|
|
ccparray_put (&ccp, "--decrypt");
|
|
|
|
|
ccparray_put (&ccp, "--");
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, NULL);
|
|
|
|
|
argv = ccparray_get (&ccp, NULL);
|
|
|
|
|
if (!argv)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
err = gnupg_exec_tool_stream (opt.gpg_program, argv, input,
|
|
|
|
|
NULL, output,
|
2017-02-23 20:10:59 +01:00
|
|
|
|
decrypt_stream_status_cb, decinfo);
|
|
|
|
|
if (!err && (!decinfo->fpr || !decinfo->mainfpr || !decinfo->otrust))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ENGINE);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("decryption failed: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else if (opt.verbose)
|
|
|
|
|
log_info ("decryption succeeded\n");
|
|
|
|
|
|
|
|
|
|
es_rewind (output);
|
|
|
|
|
*r_output = output;
|
|
|
|
|
output = NULL;
|
|
|
|
|
|
|
|
|
|
leave:
|
2017-02-23 20:10:59 +01:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
xfree (decinfo->fpr);
|
|
|
|
|
xfree (decinfo->mainfpr);
|
|
|
|
|
memset (decinfo, 0, sizeof *decinfo);
|
|
|
|
|
}
|
2016-09-29 17:55:32 +02:00
|
|
|
|
es_fclose (output);
|
|
|
|
|
xfree (argv);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-08-24 15:48:21 +02:00
|
|
|
|
|
|
|
|
|
/* Check whether the provider supports the WKS protocol. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
command_supported (char *userid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *addrspec = NULL;
|
|
|
|
|
char *submission_to = NULL;
|
|
|
|
|
|
2017-08-03 21:16:22 +02:00
|
|
|
|
if (!strchr (userid, '@'))
|
|
|
|
|
{
|
|
|
|
|
char *tmp = xstrconcat ("foo@", userid, NULL);
|
|
|
|
|
addrspec = mailbox_from_userid (tmp);
|
|
|
|
|
xfree (tmp);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
addrspec = mailbox_from_userid (userid);
|
2016-08-24 15:48:21 +02:00
|
|
|
|
if (!addrspec)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("\"%s\" is not a proper mail address\n"), userid);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_USER_ID);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Get the submission address. */
|
|
|
|
|
err = wkd_get_submission_address (addrspec, &submission_to);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA
|
|
|
|
|
|| gpg_err_code (err) == GPG_ERR_UNKNOWN_HOST)
|
|
|
|
|
{
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("provider for '%s' does NOT support WKS (%s)\n",
|
|
|
|
|
addrspec, gpg_strerror (err));
|
|
|
|
|
err = gpg_error (GPG_ERR_FALSE);
|
|
|
|
|
log_inc_errorcount ();
|
|
|
|
|
}
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("provider for '%s' supports WKS\n", addrspec);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (submission_to);
|
|
|
|
|
xfree (addrspec);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-12-08 16:57:21 +01:00
|
|
|
|
|
|
|
|
|
/* Check whether the key for USERID is available in the WKD. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
command_check (char *userid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *addrspec = NULL;
|
|
|
|
|
estream_t key = NULL;
|
|
|
|
|
char *fpr = NULL;
|
|
|
|
|
strlist_t mboxes = NULL;
|
|
|
|
|
strlist_t sl;
|
|
|
|
|
int found = 0;
|
|
|
|
|
|
|
|
|
|
addrspec = mailbox_from_userid (userid);
|
|
|
|
|
if (!addrspec)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("\"%s\" is not a proper mail address\n"), userid);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_USER_ID);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Get the submission address. */
|
|
|
|
|
err = wkd_get_key (addrspec, &key);
|
|
|
|
|
switch (gpg_err_code (err))
|
|
|
|
|
{
|
|
|
|
|
case 0:
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("public key for '%s' found via WKD\n", addrspec);
|
|
|
|
|
/* Fixme: Check that the key contains the user id. */
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GPG_ERR_NO_DATA: /* No such key. */
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("public key for '%s' NOT found via WKD\n", addrspec);
|
|
|
|
|
err = gpg_error (GPG_ERR_NO_PUBKEY);
|
|
|
|
|
log_inc_errorcount ();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GPG_ERR_UNKNOWN_HOST:
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("error looking up '%s' via WKD: %s\n",
|
|
|
|
|
addrspec, gpg_strerror (err));
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
log_error ("error looking up '%s' via WKD: %s\n",
|
|
|
|
|
addrspec, gpg_strerror (err));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Look closer at the key. */
|
|
|
|
|
err = wks_list_key (key, &fpr, &mboxes);
|
|
|
|
|
if (err || !fpr)
|
|
|
|
|
{
|
|
|
|
|
log_error ("error parsing key: %s\n",
|
|
|
|
|
err? gpg_strerror (err) : "no fingerprint found");
|
|
|
|
|
err = gpg_error (GPG_ERR_NO_PUBKEY);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("fingerprint: %s\n", fpr);
|
|
|
|
|
|
|
|
|
|
for (sl = mboxes; sl; sl = sl->next)
|
|
|
|
|
{
|
|
|
|
|
if (!strcmp (sl->d, addrspec))
|
|
|
|
|
found = 1;
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info (" addr-spec: %s\n", sl->d);
|
|
|
|
|
}
|
|
|
|
|
if (!found)
|
|
|
|
|
{
|
|
|
|
|
log_error ("public key for '%s' has no user id with the mail address\n",
|
|
|
|
|
addrspec);
|
|
|
|
|
err = gpg_error (GPG_ERR_CERT_REVOKED);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (fpr);
|
|
|
|
|
free_strlist (mboxes);
|
|
|
|
|
es_fclose (key);
|
|
|
|
|
xfree (addrspec);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
/* Locate the key by fingerprint and userid and send a publication
|
|
|
|
|
* request. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
command_send (const char *fingerprint, char *userid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
KEYDB_SEARCH_DESC desc;
|
|
|
|
|
char *addrspec = NULL;
|
|
|
|
|
estream_t key = NULL;
|
2016-10-05 14:38:37 +02:00
|
|
|
|
estream_t keyenc = NULL;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
char *submission_to = NULL;
|
|
|
|
|
mime_maker_t mime = NULL;
|
2016-09-02 16:54:42 +02:00
|
|
|
|
struct policy_flags_s policy;
|
|
|
|
|
|
|
|
|
|
memset (&policy, 0, sizeof policy);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
if (classify_user_id (fingerprint, &desc, 1)
|
|
|
|
|
|| !(desc.mode == KEYDB_SEARCH_MODE_FPR
|
|
|
|
|
|| desc.mode == KEYDB_SEARCH_MODE_FPR20))
|
|
|
|
|
{
|
|
|
|
|
log_error (_("\"%s\" is not a fingerprint\n"), fingerprint);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_NAME);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
addrspec = mailbox_from_userid (userid);
|
|
|
|
|
if (!addrspec)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("\"%s\" is not a proper mail address\n"), userid);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_USER_ID);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
err = get_key (&key, fingerprint, addrspec);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Get the submission address. */
|
2016-10-05 11:51:32 +02:00
|
|
|
|
if (fake_submission_addr)
|
|
|
|
|
{
|
|
|
|
|
submission_to = xstrdup (fake_submission_addr);
|
|
|
|
|
err = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
err = wkd_get_submission_address (addrspec, &submission_to);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
if (err)
|
2016-12-22 16:06:13 +01:00
|
|
|
|
{
|
|
|
|
|
char *domain = strchr (addrspec, '@');
|
|
|
|
|
if (domain)
|
|
|
|
|
domain = domain + 1;
|
|
|
|
|
log_error (_("looking up WKS submission address for %s: %s\n"),
|
|
|
|
|
domain ? domain : addrspec, gpg_strerror (err));
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
log_error (_("this domain probably doesn't support WKS.\n"));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
log_info ("submitting request to '%s'\n", submission_to);
|
|
|
|
|
|
2016-09-02 16:54:42 +02:00
|
|
|
|
/* Get the policy flags. */
|
2016-10-05 11:51:32 +02:00
|
|
|
|
if (!fake_submission_addr)
|
|
|
|
|
{
|
|
|
|
|
estream_t mbuf;
|
|
|
|
|
|
|
|
|
|
err = wkd_get_policy_flags (addrspec, &mbuf);
|
2016-12-22 14:39:11 +01:00
|
|
|
|
if (err && gpg_err_code (err) != GPG_ERR_NO_DATA)
|
2016-10-05 11:51:32 +02:00
|
|
|
|
{
|
|
|
|
|
log_error ("error reading policy flags for '%s': %s\n",
|
|
|
|
|
submission_to, gpg_strerror (err));
|
2016-09-02 16:54:42 +02:00
|
|
|
|
goto leave;
|
2016-12-22 16:06:13 +01:00
|
|
|
|
}
|
2016-10-05 11:51:32 +02:00
|
|
|
|
if (mbuf)
|
|
|
|
|
{
|
|
|
|
|
err = wks_parse_policy (&policy, mbuf, 1);
|
|
|
|
|
es_fclose (mbuf);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-09-02 16:54:42 +02:00
|
|
|
|
|
|
|
|
|
if (policy.auth_submit)
|
|
|
|
|
log_info ("no confirmation required for '%s'\n", addrspec);
|
|
|
|
|
|
2016-10-05 14:38:37 +02:00
|
|
|
|
/* Encrypt the key part. */
|
|
|
|
|
es_rewind (key);
|
2016-11-07 14:04:47 +01:00
|
|
|
|
err = encrypt_response (&keyenc, key, submission_to, fingerprint);
|
2016-10-05 14:38:37 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
es_fclose (key);
|
|
|
|
|
key = NULL;
|
|
|
|
|
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
/* Send the key. */
|
|
|
|
|
err = mime_maker_new (&mime, NULL);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "From", addrspec);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "To", submission_to);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "Subject", "Key publishing request");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2017-02-23 20:10:59 +01:00
|
|
|
|
/* Tell server which draft we support. */
|
|
|
|
|
err = mime_maker_add_header (mime, "Wks-Draft-Version",
|
|
|
|
|
STR2(WKS_DRAFT_VERSION));
|
2016-06-29 12:00:22 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2016-10-05 14:38:37 +02:00
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"multipart/encrypted; "
|
|
|
|
|
"protocol=\"application/pgp-encrypted\"");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_container (mime);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2016-10-05 14:38:37 +02:00
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"application/pgp-encrypted");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_body (mime, "Version: 1\n");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"application/octet-stream");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = mime_maker_add_stream (mime, &keyenc);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2016-07-03 00:41:30 +02:00
|
|
|
|
err = wks_send_mime (mime);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
mime_maker_release (mime);
|
|
|
|
|
xfree (submission_to);
|
2016-10-05 14:38:37 +02:00
|
|
|
|
es_fclose (keyenc);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
es_fclose (key);
|
|
|
|
|
xfree (addrspec);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-07-12 17:27:15 +02:00
|
|
|
|
static void
|
|
|
|
|
encrypt_response_status_cb (void *opaque, const char *keyword, char *args)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t *failure = opaque;
|
|
|
|
|
char *fields[2];
|
|
|
|
|
|
2016-09-29 17:55:32 +02:00
|
|
|
|
if (DBG_CRYPTO)
|
|
|
|
|
log_debug ("gpg status: %s %s\n", keyword, args);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
|
|
|
|
|
if (!strcmp (keyword, "FAILURE"))
|
|
|
|
|
{
|
|
|
|
|
if (split_fields (args, fields, DIM (fields)) >= 2
|
|
|
|
|
&& !strcmp (fields[0], "encrypt"))
|
|
|
|
|
*failure = strtoul (fields[1], NULL, 10);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Encrypt the INPUT stream to a new stream which is stored at success
|
2016-11-07 14:04:47 +01:00
|
|
|
|
* at R_OUTPUT. Encryption is done for ADDRSPEC and for FINGERPRINT
|
|
|
|
|
* (so that the sent message may later be inspected by the user). We
|
|
|
|
|
* currently retrieve that key from the WKD, DANE, or from "local".
|
|
|
|
|
* "local" is last to prefer the latest key version but use a local
|
|
|
|
|
* copy in case we are working offline. It might be useful for the
|
|
|
|
|
* server to send the fingerprint of its encryption key - or even the
|
|
|
|
|
* entire key back. */
|
2016-07-12 17:27:15 +02:00
|
|
|
|
static gpg_error_t
|
2016-11-07 14:04:47 +01:00
|
|
|
|
encrypt_response (estream_t *r_output, estream_t input, const char *addrspec,
|
|
|
|
|
const char *fingerprint)
|
2016-07-12 17:27:15 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
ccparray_t ccp;
|
|
|
|
|
const char **argv;
|
|
|
|
|
estream_t output;
|
|
|
|
|
gpg_error_t gpg_err = 0;
|
|
|
|
|
|
|
|
|
|
*r_output = NULL;
|
|
|
|
|
|
|
|
|
|
output = es_fopenmem (0, "w+b");
|
|
|
|
|
if (!output)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error allocating memory buffer: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ccparray_init (&ccp, 0);
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, "--no-options");
|
|
|
|
|
if (!opt.verbose)
|
|
|
|
|
ccparray_put (&ccp, "--quiet");
|
|
|
|
|
else if (opt.verbose > 1)
|
|
|
|
|
ccparray_put (&ccp, "--verbose");
|
|
|
|
|
ccparray_put (&ccp, "--batch");
|
|
|
|
|
ccparray_put (&ccp, "--status-fd=2");
|
|
|
|
|
ccparray_put (&ccp, "--always-trust");
|
|
|
|
|
ccparray_put (&ccp, "--armor");
|
2016-10-05 14:38:37 +02:00
|
|
|
|
if (fake_submission_addr)
|
|
|
|
|
ccparray_put (&ccp, "--auto-key-locate=clear,local");
|
|
|
|
|
else
|
|
|
|
|
ccparray_put (&ccp, "--auto-key-locate=clear,wkd,dane,local");
|
2016-07-12 17:27:15 +02:00
|
|
|
|
ccparray_put (&ccp, "--recipient");
|
|
|
|
|
ccparray_put (&ccp, addrspec);
|
2016-11-07 14:04:47 +01:00
|
|
|
|
ccparray_put (&ccp, "--recipient");
|
|
|
|
|
ccparray_put (&ccp, fingerprint);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
ccparray_put (&ccp, "--encrypt");
|
|
|
|
|
ccparray_put (&ccp, "--");
|
|
|
|
|
|
|
|
|
|
ccparray_put (&ccp, NULL);
|
|
|
|
|
argv = ccparray_get (&ccp, NULL);
|
|
|
|
|
if (!argv)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
err = gnupg_exec_tool_stream (opt.gpg_program, argv, input,
|
|
|
|
|
NULL, output,
|
|
|
|
|
encrypt_response_status_cb, &gpg_err);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
if (gpg_err)
|
|
|
|
|
err = gpg_err;
|
|
|
|
|
log_error ("encryption failed: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
es_rewind (output);
|
|
|
|
|
*r_output = output;
|
|
|
|
|
output = NULL;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
es_fclose (output);
|
|
|
|
|
xfree (argv);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
send_confirmation_response (const char *sender, const char *address,
|
2016-11-07 14:04:47 +01:00
|
|
|
|
const char *nonce, int encrypt,
|
|
|
|
|
const char *fingerprint)
|
2016-06-29 12:00:22 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
estream_t body = NULL;
|
2016-07-12 17:27:15 +02:00
|
|
|
|
estream_t bodyenc = NULL;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
mime_maker_t mime = NULL;
|
|
|
|
|
|
|
|
|
|
body = es_fopenmem (0, "w+b");
|
|
|
|
|
if (!body)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error allocating memory buffer: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2016-07-12 17:27:15 +02:00
|
|
|
|
|
|
|
|
|
/* It is fine to use 8 bit encoding because that is encrypted and
|
2016-06-29 12:00:22 +02:00
|
|
|
|
* only our client will see it. */
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (encrypt)
|
|
|
|
|
{
|
|
|
|
|
es_fputs ("Content-Type: application/vnd.gnupg.wks\n"
|
|
|
|
|
"Content-Transfer-Encoding: 8bit\n"
|
|
|
|
|
"\n",
|
|
|
|
|
body);
|
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
es_fprintf (body, ("type: confirmation-response\n"
|
|
|
|
|
"sender: %s\n"
|
|
|
|
|
"address: %s\n"
|
|
|
|
|
"nonce: %s\n"),
|
|
|
|
|
sender,
|
|
|
|
|
address,
|
|
|
|
|
nonce);
|
|
|
|
|
|
|
|
|
|
es_rewind (body);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (encrypt)
|
|
|
|
|
{
|
2016-11-07 14:04:47 +01:00
|
|
|
|
err = encrypt_response (&bodyenc, body, sender, fingerprint);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
es_fclose (body);
|
|
|
|
|
body = NULL;
|
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
err = mime_maker_new (&mime, NULL);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "From", address);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "To", sender);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "Subject", "Key publication confirmation");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2017-02-23 20:10:59 +01:00
|
|
|
|
err = mime_maker_add_header (mime, "Wks-Draft-Version",
|
|
|
|
|
STR2(WKS_DRAFT_VERSION));
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (encrypt)
|
|
|
|
|
{
|
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"multipart/encrypted; "
|
|
|
|
|
"protocol=\"application/pgp-encrypted\"");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2016-09-29 10:20:38 +02:00
|
|
|
|
err = mime_maker_add_container (mime);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
2016-07-12 17:27:15 +02:00
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"application/pgp-encrypted");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_body (mime, "Version: 1\n");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"application/octet-stream");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = mime_maker_add_stream (mime, &bodyenc);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = mime_maker_add_header (mime, "Content-Type",
|
|
|
|
|
"application/vnd.gnupg.wks");
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
err = mime_maker_add_stream (mime, &body);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
2016-07-03 00:41:30 +02:00
|
|
|
|
err = wks_send_mime (mime);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
mime_maker_release (mime);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
es_fclose (bodyenc);
|
|
|
|
|
es_fclose (body);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Reply to a confirmation request. The MSG has already been
|
2017-02-23 20:10:59 +01:00
|
|
|
|
* decrypted and we only need to send the nonce back. MAINFPR is
|
|
|
|
|
* either NULL or the primary key fingerprint of the key used to
|
|
|
|
|
* decrypt the request. */
|
2016-06-29 12:00:22 +02:00
|
|
|
|
static gpg_error_t
|
2017-02-23 20:10:59 +01:00
|
|
|
|
process_confirmation_request (estream_t msg, const char *mainfpr)
|
2016-06-29 12:00:22 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
nvc_t nvc;
|
|
|
|
|
nve_t item;
|
2016-11-07 14:04:47 +01:00
|
|
|
|
const char *value, *sender, *address, *fingerprint, *nonce;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
err = nvc_parse (&nvc, NULL, msg);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("parsing the WKS message failed: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-29 17:55:32 +02:00
|
|
|
|
if (DBG_MIME)
|
2016-06-29 12:00:22 +02:00
|
|
|
|
{
|
|
|
|
|
log_debug ("request follows:\n");
|
|
|
|
|
nvc_write (nvc, log_get_stream ());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check that this is a confirmation request. */
|
|
|
|
|
if (!((item = nvc_lookup (nvc, "type:")) && (value = nve_value (item))
|
|
|
|
|
&& !strcmp (value, "confirmation-request")))
|
|
|
|
|
{
|
|
|
|
|
if (item && value)
|
|
|
|
|
log_error ("received unexpected wks message '%s'\n", value);
|
|
|
|
|
else
|
|
|
|
|
log_error ("received invalid wks message: %s\n", "'type' missing");
|
|
|
|
|
err = gpg_error (GPG_ERR_UNEXPECTED_MSG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-07 14:04:47 +01:00
|
|
|
|
/* Get the fingerprint. */
|
|
|
|
|
if (!((item = nvc_lookup (nvc, "fingerprint:"))
|
|
|
|
|
&& (value = nve_value (item))
|
|
|
|
|
&& strlen (value) >= 40))
|
|
|
|
|
{
|
|
|
|
|
log_error ("received invalid wks message: %s\n",
|
|
|
|
|
"'fingerprint' missing or invalid");
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
fingerprint = value;
|
|
|
|
|
|
2017-02-23 20:10:59 +01:00
|
|
|
|
/* Check that the fingerprint matches the key used to decrypt the
|
|
|
|
|
* message. In --read mode or with the old format we don't have the
|
|
|
|
|
* decryption key; thus we can't bail out. */
|
|
|
|
|
if (!mainfpr || ascii_strcasecmp (mainfpr, fingerprint))
|
|
|
|
|
{
|
|
|
|
|
log_info ("target fingerprint: %s\n", fingerprint);
|
|
|
|
|
log_info ("but decrypted with: %s\n", mainfpr);
|
|
|
|
|
log_error ("confirmation request not decrypted with target key\n");
|
|
|
|
|
if (mainfpr)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
/* Get the address. */
|
|
|
|
|
if (!((item = nvc_lookup (nvc, "address:")) && (value = nve_value (item))
|
|
|
|
|
&& is_valid_mailbox (value)))
|
|
|
|
|
{
|
|
|
|
|
log_error ("received invalid wks message: %s\n",
|
|
|
|
|
"'address' missing or invalid");
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
address = value;
|
|
|
|
|
/* FIXME: Check that the "address" matches the User ID we want to
|
2017-02-23 20:10:59 +01:00
|
|
|
|
* publish. */
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
/* Get the sender. */
|
|
|
|
|
if (!((item = nvc_lookup (nvc, "sender:")) && (value = nve_value (item))
|
|
|
|
|
&& is_valid_mailbox (value)))
|
|
|
|
|
{
|
|
|
|
|
log_error ("received invalid wks message: %s\n",
|
|
|
|
|
"'sender' missing or invalid");
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
sender = value;
|
|
|
|
|
/* FIXME: Check that the "sender" matches the From: address. */
|
|
|
|
|
|
|
|
|
|
/* Get the nonce. */
|
|
|
|
|
if (!((item = nvc_lookup (nvc, "nonce:")) && (value = nve_value (item))
|
|
|
|
|
&& strlen (value) > 16))
|
|
|
|
|
{
|
|
|
|
|
log_error ("received invalid wks message: %s\n",
|
|
|
|
|
"'nonce' missing or too short");
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
nonce = value;
|
|
|
|
|
|
2016-07-12 17:27:15 +02:00
|
|
|
|
/* Send the confirmation. If no key was found, try again without
|
|
|
|
|
* encryption. */
|
2016-11-07 14:04:47 +01:00
|
|
|
|
err = send_confirmation_response (sender, address, nonce, 1, fingerprint);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_PUBKEY)
|
|
|
|
|
{
|
|
|
|
|
log_info ("no encryption key found - sending response in the clear\n");
|
2016-11-07 14:04:47 +01:00
|
|
|
|
err = send_confirmation_response (sender, address, nonce, 0, NULL);
|
2016-07-12 17:27:15 +02:00
|
|
|
|
}
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
nvc_release (nvc);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-09-29 17:55:32 +02:00
|
|
|
|
/* Read a confirmation request and decrypt it if needed. This
|
|
|
|
|
* function may not be used with a mail or MIME message but only with
|
|
|
|
|
* the actual encrypted or plaintext WKS data. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_confirmation_request (estream_t msg)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int c;
|
|
|
|
|
estream_t plaintext = NULL;
|
|
|
|
|
|
|
|
|
|
/* We take a really simple approach to check whether MSG is
|
|
|
|
|
* encrypted: We know that an encrypted message is always armored
|
|
|
|
|
* and thus starts with a few dashes. It is even sufficient to
|
|
|
|
|
* check for a single dash, because that can never be a proper first
|
|
|
|
|
* WKS data octet. We need to skip leading spaces, though. */
|
|
|
|
|
while ((c = es_fgetc (msg)) == ' ' || c == '\t' || c == '\r' || c == '\n')
|
|
|
|
|
;
|
|
|
|
|
if (c == EOF)
|
|
|
|
|
{
|
|
|
|
|
log_error ("can't process an empty message\n");
|
|
|
|
|
return gpg_error (GPG_ERR_INV_DATA);
|
|
|
|
|
}
|
|
|
|
|
if (es_ungetc (c, msg) != c)
|
|
|
|
|
{
|
|
|
|
|
log_error ("error ungetting octet from message\n");
|
|
|
|
|
return gpg_error (GPG_ERR_INTERNAL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c != '-')
|
2017-02-23 20:10:59 +01:00
|
|
|
|
err = process_confirmation_request (msg, NULL);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
2017-02-23 20:10:59 +01:00
|
|
|
|
struct decrypt_stream_parm_s decinfo;
|
|
|
|
|
|
|
|
|
|
err = decrypt_stream (&plaintext, &decinfo, msg);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
if (err)
|
|
|
|
|
log_error ("decryption failed: %s\n", gpg_strerror (err));
|
2017-02-23 20:10:59 +01:00
|
|
|
|
else if (decinfo.otrust != 'u')
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_WRONG_SECKEY);
|
|
|
|
|
log_error ("key used to decrypt the confirmation request"
|
|
|
|
|
" was not generated by us\n");
|
|
|
|
|
}
|
2016-09-29 17:55:32 +02:00
|
|
|
|
else
|
2017-02-23 20:10:59 +01:00
|
|
|
|
err = process_confirmation_request (plaintext, decinfo.mainfpr);
|
|
|
|
|
xfree (decinfo.fpr);
|
|
|
|
|
xfree (decinfo.mainfpr);
|
2016-09-29 17:55:32 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
es_fclose (plaintext);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-06-29 12:00:22 +02:00
|
|
|
|
/* Called from the MIME receiver to process the plain text data in MSG. */
|
|
|
|
|
static gpg_error_t
|
2016-09-29 17:55:32 +02:00
|
|
|
|
command_receive_cb (void *opaque, const char *mediatype,
|
|
|
|
|
estream_t msg, unsigned int flags)
|
2016-06-29 12:00:22 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
|
|
|
|
(void)opaque;
|
2016-09-29 17:55:32 +02:00
|
|
|
|
(void)flags;
|
2016-06-29 12:00:22 +02:00
|
|
|
|
|
|
|
|
|
if (!strcmp (mediatype, "application/vnd.gnupg.wks"))
|
2016-09-29 17:55:32 +02:00
|
|
|
|
err = read_confirmation_request (msg);
|
2016-06-29 12:00:22 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_info ("ignoring unexpected message of type '%s'\n", mediatype);
|
|
|
|
|
err = gpg_error (GPG_ERR_UNEXPECTED_MSG);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|