mirror of https://gitee.com/openkylin/gnupg2.git
496 lines
12 KiB
C
496 lines
12 KiB
C
|
/* simple-pwquery.c - A simple password query client for gpg-agent
|
|||
|
* Copyright (C) 2002, 2004, 2007 Free Software Foundation, Inc.
|
|||
|
*
|
|||
|
* This file is part of GnuPG.
|
|||
|
*
|
|||
|
* GnuPG is free software; you can redistribute it and/or modify
|
|||
|
* it under the terms of the GNU General Public License as published by
|
|||
|
* the Free Software Foundation; either version 3 of the License, or
|
|||
|
* (at your option) any later version.
|
|||
|
*
|
|||
|
* GnuPG is distributed in the hope that it will be useful,
|
|||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|||
|
* GNU General Public License for more details.
|
|||
|
*
|
|||
|
* You should have received a copy of the GNU General Public License
|
|||
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
|||
|
*/
|
|||
|
|
|||
|
/* This module is intended as a simple client implementation to
|
|||
|
gpg-agent's GET_PASSPHRASE command. It can only cope with an
|
|||
|
already running gpg-agent. Some stuff is configurable in the
|
|||
|
header file. */
|
|||
|
|
|||
|
#ifdef HAVE_CONFIG_H
|
|||
|
#include <config.h>
|
|||
|
#endif
|
|||
|
#include <stdlib.h>
|
|||
|
#include <stddef.h>
|
|||
|
#include <string.h>
|
|||
|
#include <errno.h>
|
|||
|
#include <unistd.h>
|
|||
|
#include <assuan.h>
|
|||
|
#ifdef HAVE_W32_SYSTEM
|
|||
|
#include <winsock2.h>
|
|||
|
#else
|
|||
|
#include <sys/socket.h>
|
|||
|
#include <sys/un.h>
|
|||
|
#endif
|
|||
|
#ifdef HAVE_LOCALE_H
|
|||
|
#include <locale.h>
|
|||
|
#endif
|
|||
|
|
|||
|
#define GNUPG_COMMON_NEED_AFLOCAL
|
|||
|
#include "../common/mischelp.h"
|
|||
|
#include "sysutils.h"
|
|||
|
#include "membuf.h"
|
|||
|
|
|||
|
|
|||
|
#define SIMPLE_PWQUERY_IMPLEMENTATION 1
|
|||
|
#include "simple-pwquery.h"
|
|||
|
|
|||
|
#define SPWQ_OUT_OF_CORE gpg_error_from_errno (ENOMEM)
|
|||
|
#define SPWQ_IO_ERROR gpg_error_from_errno (EIO)
|
|||
|
#define SPWQ_PROTOCOL_ERROR gpg_error (GPG_ERR_PROTOCOL_VIOLATION)
|
|||
|
#define SPWQ_ERR_RESPONSE gpg_error (GPG_ERR_INV_RESPONSE)
|
|||
|
#define SPWQ_NO_AGENT gpg_error (GPG_ERR_NO_AGENT)
|
|||
|
#define SPWQ_SYS_ERROR gpg_error_from_syserror ()
|
|||
|
#define SPWQ_GENERAL_ERROR gpg_error (GPG_ERR_GENERAL)
|
|||
|
#define SPWQ_NO_PIN_ENTRY gpg_error (GPG_ERR_NO_PIN_ENTRY)
|
|||
|
|
|||
|
#ifndef _
|
|||
|
#define _(a) (a)
|
|||
|
#endif
|
|||
|
|
|||
|
#if !defined (hexdigitp) && !defined (xtoi_2)
|
|||
|
#define digitp(p) (*(p) >= '0' && *(p) <= '9')
|
|||
|
#define hexdigitp(a) (digitp (a) \
|
|||
|
|| (*(a) >= 'A' && *(a) <= 'F') \
|
|||
|
|| (*(a) >= 'a' && *(a) <= 'f'))
|
|||
|
#define xtoi_1(p) (*(p) <= '9'? (*(p)- '0'): \
|
|||
|
*(p) <= 'F'? (*(p)-'A'+10):(*(p)-'a'+10))
|
|||
|
#define xtoi_2(p) ((xtoi_1(p) * 16) + xtoi_1((p)+1))
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
/* Name of the socket to be used. This is a kludge to keep on using
|
|||
|
the existsing code despite that we only support a standard socket. */
|
|||
|
static char *default_gpg_agent_info;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
#ifndef HAVE_STPCPY
|
|||
|
static char *
|
|||
|
my_stpcpy(char *a,const char *b)
|
|||
|
{
|
|||
|
while( *b )
|
|||
|
*a++ = *b++;
|
|||
|
*a = 0;
|
|||
|
|
|||
|
return (char*)a;
|
|||
|
}
|
|||
|
#define stpcpy(a,b) my_stpcpy((a), (b))
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
/* Send an option to the agent */
|
|||
|
static int
|
|||
|
agent_send_option (assuan_context_t ctx, const char *name, const char *value)
|
|||
|
{
|
|||
|
int err;
|
|||
|
char *line;
|
|||
|
|
|||
|
line = spwq_malloc (7 + strlen (name) + 1 + strlen (value) + 2);
|
|||
|
if (!line)
|
|||
|
return SPWQ_OUT_OF_CORE;
|
|||
|
strcpy (stpcpy (stpcpy (stpcpy (
|
|||
|
stpcpy (line, "OPTION "), name), "="), value), "\n");
|
|||
|
|
|||
|
err = assuan_transact (ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
|
|||
|
|
|||
|
spwq_free (line);
|
|||
|
return err;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Send all available options to the agent. */
|
|||
|
static int
|
|||
|
agent_send_all_options (assuan_context_t ctx)
|
|||
|
{
|
|||
|
char *dft_display = NULL;
|
|||
|
char *dft_ttyname = NULL;
|
|||
|
char *dft_ttytype = NULL;
|
|||
|
char *dft_xauthority = NULL;
|
|||
|
char *dft_pinentry_user_data = NULL;
|
|||
|
int rc = 0;
|
|||
|
|
|||
|
dft_display = getenv ("DISPLAY");
|
|||
|
if (dft_display)
|
|||
|
{
|
|||
|
if ((rc = agent_send_option (ctx, "display", dft_display)))
|
|||
|
return rc;
|
|||
|
}
|
|||
|
|
|||
|
dft_ttyname = getenv ("GPG_TTY");
|
|||
|
#if !defined(HAVE_W32_SYSTEM) && !defined(HAVE_BROKEN_TTYNAME)
|
|||
|
if ((!dft_ttyname || !*dft_ttyname) && ttyname (0))
|
|||
|
dft_ttyname = ttyname (0);
|
|||
|
#endif
|
|||
|
if (dft_ttyname && *dft_ttyname)
|
|||
|
{
|
|||
|
if ((rc=agent_send_option (ctx, "ttyname", dft_ttyname)))
|
|||
|
return rc;
|
|||
|
}
|
|||
|
|
|||
|
dft_ttytype = getenv ("TERM");
|
|||
|
if (dft_ttyname && dft_ttytype)
|
|||
|
{
|
|||
|
if ((rc = agent_send_option (ctx, "ttytype", dft_ttytype)))
|
|||
|
return rc;
|
|||
|
}
|
|||
|
|
|||
|
#if defined(HAVE_SETLOCALE)
|
|||
|
{
|
|||
|
char *old_lc = NULL;
|
|||
|
char *dft_lc = NULL;
|
|||
|
|
|||
|
#if defined(LC_CTYPE)
|
|||
|
old_lc = setlocale (LC_CTYPE, NULL);
|
|||
|
if (old_lc)
|
|||
|
{
|
|||
|
char *p = spwq_malloc (strlen (old_lc)+1);
|
|||
|
if (!p)
|
|||
|
return SPWQ_OUT_OF_CORE;
|
|||
|
strcpy (p, old_lc);
|
|||
|
old_lc = p;
|
|||
|
}
|
|||
|
dft_lc = setlocale (LC_CTYPE, "");
|
|||
|
if (dft_ttyname && dft_lc)
|
|||
|
rc = agent_send_option (ctx, "lc-ctype", dft_lc);
|
|||
|
if (old_lc)
|
|||
|
{
|
|||
|
setlocale (LC_CTYPE, old_lc);
|
|||
|
spwq_free (old_lc);
|
|||
|
}
|
|||
|
if (rc)
|
|||
|
return rc;
|
|||
|
#endif
|
|||
|
|
|||
|
#if defined(LC_MESSAGES)
|
|||
|
old_lc = setlocale (LC_MESSAGES, NULL);
|
|||
|
if (old_lc)
|
|||
|
{
|
|||
|
char *p = spwq_malloc (strlen (old_lc)+1);
|
|||
|
if (!p)
|
|||
|
return SPWQ_OUT_OF_CORE;
|
|||
|
strcpy (p, old_lc);
|
|||
|
old_lc = p;
|
|||
|
}
|
|||
|
dft_lc = setlocale (LC_MESSAGES, "");
|
|||
|
if (dft_ttyname && dft_lc)
|
|||
|
rc = agent_send_option (ctx, "lc-messages", dft_lc);
|
|||
|
if (old_lc)
|
|||
|
{
|
|||
|
setlocale (LC_MESSAGES, old_lc);
|
|||
|
spwq_free (old_lc);
|
|||
|
}
|
|||
|
if (rc)
|
|||
|
return rc;
|
|||
|
#endif
|
|||
|
}
|
|||
|
#endif /*HAVE_SETLOCALE*/
|
|||
|
|
|||
|
/* Send the XAUTHORITY variable. */
|
|||
|
dft_xauthority = getenv ("XAUTHORITY");
|
|||
|
if (dft_xauthority)
|
|||
|
{
|
|||
|
/* We ignore errors here because older gpg-agents don't support
|
|||
|
this option. */
|
|||
|
agent_send_option (ctx, "xauthority", dft_xauthority);
|
|||
|
}
|
|||
|
|
|||
|
/* Send the PINENTRY_USER_DATA variable. */
|
|||
|
dft_pinentry_user_data = getenv ("PINENTRY_USER_DATA");
|
|||
|
if (dft_pinentry_user_data)
|
|||
|
{
|
|||
|
/* We ignore errors here because older gpg-agents don't support
|
|||
|
this option. */
|
|||
|
agent_send_option (ctx, "pinentry-user-data", dft_pinentry_user_data);
|
|||
|
}
|
|||
|
|
|||
|
/* Tell the agent that we support Pinentry notifications. No
|
|||
|
error checking so that it will work with older agents. */
|
|||
|
assuan_transact (ctx, "OPTION allow-pinentry-notify",
|
|||
|
NULL, NULL, NULL, NULL, NULL, NULL);
|
|||
|
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/* Try to open a connection to the agent, send all options and return
|
|||
|
the file descriptor for the connection. Return -1 in case of
|
|||
|
error. */
|
|||
|
static int
|
|||
|
agent_open (assuan_context_t *ctx)
|
|||
|
{
|
|||
|
int rc;
|
|||
|
char *infostr;
|
|||
|
|
|||
|
infostr = default_gpg_agent_info;
|
|||
|
if ( !infostr || !*infostr )
|
|||
|
{
|
|||
|
#ifdef SPWQ_USE_LOGGING
|
|||
|
log_error (_("no gpg-agent running in this session\n"));
|
|||
|
#endif
|
|||
|
*ctx = NULL;
|
|||
|
return SPWQ_NO_AGENT;
|
|||
|
}
|
|||
|
|
|||
|
rc = assuan_new (ctx);
|
|||
|
if (rc)
|
|||
|
return rc;
|
|||
|
|
|||
|
rc = assuan_socket_connect (*ctx, infostr, 0, 0);
|
|||
|
if (rc)
|
|||
|
{
|
|||
|
#ifdef SPWQ_USE_LOGGING
|
|||
|
log_error (_("can't connect to '%s': %s\n"),
|
|||
|
infostr, gpg_strerror (rc));
|
|||
|
#endif
|
|||
|
goto errout;
|
|||
|
}
|
|||
|
|
|||
|
rc = agent_send_all_options (*ctx);
|
|||
|
if (rc)
|
|||
|
{
|
|||
|
#ifdef SPWQ_USE_LOGGING
|
|||
|
log_error (_("problem setting the gpg-agent options\n"));
|
|||
|
#endif
|
|||
|
goto errout;
|
|||
|
}
|
|||
|
|
|||
|
return 0;
|
|||
|
|
|||
|
errout:
|
|||
|
assuan_release (*ctx);
|
|||
|
*ctx = NULL;
|
|||
|
return rc;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Copy text to BUFFER and escape as required. Return a pointer to
|
|||
|
the end of the new buffer. Note that BUFFER must be large enough
|
|||
|
to keep the entire text; allocataing it 3 times the size of TEXT
|
|||
|
is sufficient. */
|
|||
|
static char *
|
|||
|
copy_and_escape (char *buffer, const char *text)
|
|||
|
{
|
|||
|
int i;
|
|||
|
const unsigned char *s = (unsigned char *)text;
|
|||
|
char *p = buffer;
|
|||
|
|
|||
|
|
|||
|
for (i=0; s[i]; i++)
|
|||
|
{
|
|||
|
if (s[i] < ' ' || s[i] == '+')
|
|||
|
{
|
|||
|
sprintf (p, "%%%02X", s[i]);
|
|||
|
p += 3;
|
|||
|
}
|
|||
|
else if (s[i] == ' ')
|
|||
|
*p++ = '+';
|
|||
|
else
|
|||
|
*p++ = s[i];
|
|||
|
}
|
|||
|
return p;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Set the name of the default socket to NAME. */
|
|||
|
int
|
|||
|
simple_pw_set_socket (const char *name)
|
|||
|
{
|
|||
|
spwq_free (default_gpg_agent_info);
|
|||
|
default_gpg_agent_info = NULL;
|
|||
|
if (name)
|
|||
|
{
|
|||
|
default_gpg_agent_info = spwq_malloc (strlen (name) + 1);
|
|||
|
if (!default_gpg_agent_info)
|
|||
|
return SPWQ_OUT_OF_CORE;
|
|||
|
strcpy (default_gpg_agent_info, name);
|
|||
|
}
|
|||
|
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* This is the default inquiry callback. It merely handles the
|
|||
|
Pinentry notification. */
|
|||
|
static gpg_error_t
|
|||
|
default_inq_cb (void *opaque, const char *line)
|
|||
|
{
|
|||
|
(void)opaque;
|
|||
|
|
|||
|
if (!strncmp (line, "PINENTRY_LAUNCHED", 17) && (line[17]==' '||!line[17]))
|
|||
|
{
|
|||
|
gnupg_allow_set_foregound_window ((pid_t)strtoul (line+17, NULL, 10));
|
|||
|
/* We do not return errors to avoid breaking other code. */
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
#ifdef SPWQ_USE_LOGGING
|
|||
|
log_debug ("ignoring gpg-agent inquiry '%s'\n", line);
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Ask the gpg-agent for a passphrase and present the user with a
|
|||
|
DESCRIPTION, a PROMPT and optionally with a TRYAGAIN extra text.
|
|||
|
If a CACHEID is not NULL it is used to locate the passphrase in
|
|||
|
the cache and store it under this ID. If OPT_CHECK is true
|
|||
|
gpg-agent is asked to apply some checks on the passphrase security.
|
|||
|
If ERRORCODE is not NULL it should point a variable receiving an
|
|||
|
errorcode; this error code might be 0 if the user canceled the
|
|||
|
operation. The function returns NULL to indicate an error. */
|
|||
|
char *
|
|||
|
simple_pwquery (const char *cacheid,
|
|||
|
const char *tryagain,
|
|||
|
const char *prompt,
|
|||
|
const char *description,
|
|||
|
int opt_check,
|
|||
|
int *errorcode)
|
|||
|
{
|
|||
|
int rc;
|
|||
|
assuan_context_t ctx;
|
|||
|
membuf_t data;
|
|||
|
char *result = NULL;
|
|||
|
char *pw = NULL;
|
|||
|
char *p;
|
|||
|
size_t n;
|
|||
|
|
|||
|
|
|||
|
rc = agent_open (&ctx);
|
|||
|
if (rc)
|
|||
|
goto leave;
|
|||
|
|
|||
|
if (!cacheid)
|
|||
|
cacheid = "X";
|
|||
|
if (!tryagain)
|
|||
|
tryagain = "X";
|
|||
|
if (!prompt)
|
|||
|
prompt = "X";
|
|||
|
if (!description)
|
|||
|
description = "X";
|
|||
|
|
|||
|
{
|
|||
|
char *line;
|
|||
|
/* We allocate 3 times the needed space so that there is enough
|
|||
|
space for escaping. */
|
|||
|
line = spwq_malloc (15 + 10
|
|||
|
+ 3*strlen (cacheid) + 1
|
|||
|
+ 3*strlen (tryagain) + 1
|
|||
|
+ 3*strlen (prompt) + 1
|
|||
|
+ 3*strlen (description) + 1
|
|||
|
+ 2);
|
|||
|
if (!line)
|
|||
|
{
|
|||
|
rc = SPWQ_OUT_OF_CORE;
|
|||
|
goto leave;
|
|||
|
}
|
|||
|
strcpy (line, "GET_PASSPHRASE ");
|
|||
|
p = line+15;
|
|||
|
if (opt_check)
|
|||
|
p = stpcpy (p, "--check ");
|
|||
|
p = copy_and_escape (p, cacheid);
|
|||
|
*p++ = ' ';
|
|||
|
p = copy_and_escape (p, tryagain);
|
|||
|
*p++ = ' ';
|
|||
|
p = copy_and_escape (p, prompt);
|
|||
|
*p++ = ' ';
|
|||
|
p = copy_and_escape (p, description);
|
|||
|
*p++ = '\n';
|
|||
|
|
|||
|
init_membuf_secure (&data, 64);
|
|||
|
|
|||
|
rc = assuan_transact (ctx, line, put_membuf_cb, &data,
|
|||
|
default_inq_cb, NULL, NULL, NULL);
|
|||
|
spwq_free (line);
|
|||
|
|
|||
|
/* Older Pinentries return the old assuan error code for canceled
|
|||
|
which gets translated by libassuan to GPG_ERR_ASS_CANCELED and
|
|||
|
not to the code for a user cancel. Fix this here. */
|
|||
|
if (rc && gpg_err_source (rc)
|
|||
|
&& gpg_err_code (rc) == GPG_ERR_ASS_CANCELED)
|
|||
|
rc = gpg_err_make (gpg_err_source (rc), GPG_ERR_CANCELED);
|
|||
|
|
|||
|
if (rc)
|
|||
|
{
|
|||
|
p = get_membuf (&data, &n);
|
|||
|
if (p)
|
|||
|
wipememory (p, n);
|
|||
|
spwq_free (p);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
put_membuf (&data, "", 1);
|
|||
|
result = get_membuf (&data, NULL);
|
|||
|
if (pw == NULL)
|
|||
|
rc = gpg_error_from_syserror ();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
leave:
|
|||
|
if (errorcode)
|
|||
|
*errorcode = rc;
|
|||
|
assuan_release (ctx);
|
|||
|
return result;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Ask the gpg-agent to clear the passphrase for the cache ID CACHEID. */
|
|||
|
int
|
|||
|
simple_pwclear (const char *cacheid)
|
|||
|
{
|
|||
|
char line[500];
|
|||
|
char *p;
|
|||
|
|
|||
|
/* We need not more than 50 characters for the command and the
|
|||
|
terminating nul. */
|
|||
|
if (strlen (cacheid) * 3 > sizeof (line) - 50)
|
|||
|
return SPWQ_PROTOCOL_ERROR;
|
|||
|
|
|||
|
strcpy (line, "CLEAR_PASSPHRASE ");
|
|||
|
p = line + 17;
|
|||
|
p = copy_and_escape (p, cacheid);
|
|||
|
*p++ = '\n';
|
|||
|
*p++ = '\0';
|
|||
|
|
|||
|
return simple_query (line);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Perform the simple query QUERY (which must be new-line and 0
|
|||
|
terminated) and return the error code. */
|
|||
|
int
|
|||
|
simple_query (const char *query)
|
|||
|
{
|
|||
|
assuan_context_t ctx;
|
|||
|
int rc;
|
|||
|
|
|||
|
rc = agent_open (&ctx);
|
|||
|
if (rc)
|
|||
|
return rc;
|
|||
|
|
|||
|
rc = assuan_transact (ctx, query, NULL, NULL, NULL, NULL, NULL, NULL);
|
|||
|
|
|||
|
assuan_release (ctx);
|
|||
|
return rc;
|
|||
|
}
|