Finished the bulk of changes for gnupg 1.9. This included switching
[gnupg.git] / g10 / call-agent.c
1 /* call-agent.c - divert operations to the agent
2  *      Copyright (C) 2001, 2002, 2003 Free Software Foundation, Inc.
3  *
4  * This file is part of GnuPG.
5  *
6  * GnuPG is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * GnuPG is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
19  */
20
21 #if 0  /* lety Emacs display a red warning */
22 #error fixme: this shares a lof of code with the file in ../sm
23 #endif
24
25 #include <config.h>
26 #include <stdio.h>
27 #include <stdlib.h>
28 #include <string.h>
29 #include <errno.h>
30 #include <unistd.h> 
31 #include <time.h>
32 #include <assert.h>
33 #ifdef HAVE_LOCALE_H
34 #include <locale.h>
35 #endif
36 #include <assuan.h>
37
38 #include "gpg.h"
39 #include "util.h"
40 #include "membuf.h"
41 #include "options.h"
42 #include "i18n.h"
43 #include "call-agent.h"
44
45 #ifndef DBG_ASSUAN
46 # define DBG_ASSUAN 1
47 #endif
48
49 static ASSUAN_CONTEXT agent_ctx = NULL;
50 static int force_pipe_server = 0;
51
52 struct cipher_parm_s {
53   ASSUAN_CONTEXT ctx;
54   const char *ciphertext;
55   size_t ciphertextlen;
56 };
57
58 struct genkey_parm_s {
59   ASSUAN_CONTEXT ctx;
60   const char *sexp;
61   size_t sexplen;
62 };
63
64 struct learn_parm_s {
65   int error;
66   ASSUAN_CONTEXT ctx;
67   struct membuf *data;
68 };
69
70 \f
71 /* Try to connect to the agent via socket or fork it off and work by
72    pipes.  Handle the server's initial greeting */
73 static int
74 start_agent (void)
75 {
76   int rc = 0;
77   char *infostr, *p;
78   ASSUAN_CONTEXT ctx;
79   char *dft_display = NULL;
80   char *dft_ttyname = NULL;
81   char *dft_ttytype = NULL;
82   char *old_lc = NULL;
83   char *dft_lc = NULL;
84
85   if (agent_ctx)
86     return 0; /* fixme: We need a context for each thread or serialize
87                  the access to the agent. */
88
89   infostr = force_pipe_server? NULL : getenv ("GPG_AGENT_INFO");
90   if (!infostr)
91     {
92       const char *pgmname;
93       const char *argv[3];
94       int no_close_list[3];
95       int i;
96
97       if (opt.verbose)
98         log_info (_("no running gpg-agent - starting one\n"));
99
100       if (fflush (NULL))
101         {
102           gpg_error_t tmperr = gpg_error_from_errno (errno);
103           log_error ("error flushing pending output: %s\n", strerror (errno));
104           return tmperr;
105         }
106
107       if (!opt.agent_program || !*opt.agent_program)
108         opt.agent_program = GNUPG_DEFAULT_AGENT;
109       if ( !(pgmname = strrchr (opt.agent_program, '/')))
110         pgmname = opt.agent_program;
111       else
112         pgmname++;
113
114       argv[0] = pgmname;
115       argv[1] = "--server";
116       argv[2] = NULL;
117
118       i=0;
119       if (log_get_fd () != -1)
120         no_close_list[i++] = log_get_fd ();
121       no_close_list[i++] = fileno (stderr);
122       no_close_list[i] = -1;
123
124       /* connect to the agent and perform initial handshaking */
125       rc = assuan_pipe_connect (&ctx, opt.agent_program, (char**)argv,
126                                 no_close_list);
127     }
128   else
129     {
130       int prot;
131       int pid;
132
133       infostr = xstrdup (infostr);
134       if ( !(p = strchr (infostr, ':')) || p == infostr)
135         {
136           log_error (_("malformed GPG_AGENT_INFO environment variable\n"));
137           xfree (infostr);
138           force_pipe_server = 1;
139           return start_agent ();
140         }
141       *p++ = 0;
142       pid = atoi (p);
143       while (*p && *p != ':')
144         p++;
145       prot = *p? atoi (p+1) : 0;
146       if (prot != 1)
147         {
148           log_error (_("gpg-agent protocol version %d is not supported\n"),
149                      prot);
150           xfree (infostr);
151           force_pipe_server = 1;
152           return start_agent ();
153         }
154
155       rc = assuan_socket_connect (&ctx, infostr, pid);
156       xfree (infostr);
157       if (rc == ASSUAN_Connect_Failed)
158         {
159           log_error (_("can't connect to the agent - trying fall back\n"));
160           force_pipe_server = 1;
161           return start_agent ();
162         }
163     }
164
165   if (rc)
166     {
167       log_error ("can't connect to the agent: %s\n", assuan_strerror (rc));
168       return gpg_error (GPG_ERR_NO_AGENT);
169     }
170   agent_ctx = ctx;
171
172   if (DBG_ASSUAN)
173     log_debug ("connection to agent established\n");
174
175   rc = assuan_transact (agent_ctx, "RESET", NULL, NULL, NULL, NULL, NULL, NULL);
176   if (rc)
177     return map_assuan_err (rc);
178
179   dft_display = getenv ("DISPLAY");
180   if (opt.display || dft_display)
181     {
182       char *optstr;
183       if (asprintf (&optstr, "OPTION display=%s",
184                     opt.display ? opt.display : dft_display) < 0)
185         return gpg_error_from_errno (errno);
186       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
187                             NULL);
188       free (optstr);
189       if (rc)
190         return map_assuan_err (rc);
191     }
192   if (!opt.ttyname)
193     {
194       dft_ttyname = getenv ("GPG_TTY");
195       if ((!dft_ttyname || !*dft_ttyname) && ttyname (0))
196         dft_ttyname = ttyname (0);
197     }
198   if (opt.ttyname || dft_ttyname)
199     {
200       char *optstr;
201       if (asprintf (&optstr, "OPTION ttyname=%s",
202                     opt.ttyname ? opt.ttyname : dft_ttyname) < 0)
203         return gpg_error_from_errno (errno);
204       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
205                             NULL);
206       free (optstr);
207       if (rc)
208         return map_assuan_err (rc);
209     }
210   dft_ttytype = getenv ("TERM");
211   if (opt.ttytype || (dft_ttyname && dft_ttytype))
212     {
213       char *optstr;
214       if (asprintf (&optstr, "OPTION ttytype=%s",
215                     opt.ttyname ? opt.ttytype : dft_ttytype) < 0)
216         return gpg_error_from_errno (errno);
217       rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
218                             NULL);
219       free (optstr);
220       if (rc)
221         return map_assuan_err (rc);
222     }
223 #if defined(HAVE_SETLOCALE) && defined(LC_CTYPE)
224   old_lc = setlocale (LC_CTYPE, NULL);
225   if (old_lc)
226     {
227       old_lc = strdup (old_lc);
228       if (!old_lc)
229         return gpg_error_from_errno (errno);
230
231     }
232   dft_lc = setlocale (LC_CTYPE, "");
233 #endif
234   if (opt.lc_ctype || (dft_ttyname && dft_lc))
235     {
236       char *optstr;
237       if (asprintf (&optstr, "OPTION lc-ctype=%s",
238                     opt.lc_ctype ? opt.lc_ctype : dft_lc) < 0)
239         rc = gpg_error_from_errno (errno);
240       else
241         {
242           rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
243                                 NULL);
244           free (optstr);
245           if (rc)
246             rc = map_assuan_err (rc);
247         }
248     }
249 #if defined(HAVE_SETLOCALE) && defined(LC_CTYPE)
250   if (old_lc)
251     {
252       setlocale (LC_CTYPE, old_lc);
253       free (old_lc);
254     }
255 #endif
256   if (rc)
257     return rc;
258 #if defined(HAVE_SETLOCALE) && defined(LC_MESSAGES)
259   old_lc = setlocale (LC_MESSAGES, NULL);
260   if (old_lc)
261     {
262       old_lc = strdup (old_lc);
263       if (!old_lc)
264         return gpg_error_from_errno (errno);
265     }
266   dft_lc = setlocale (LC_MESSAGES, "");
267 #endif
268   if (opt.lc_messages || (dft_ttyname && dft_lc))
269     {
270       char *optstr;
271       if (asprintf (&optstr, "OPTION lc-messages=%s",
272                     opt.lc_messages ? opt.lc_messages : dft_lc) < 0)
273         rc = gpg_error_from_errno (errno);
274       else
275         {
276           rc = assuan_transact (agent_ctx, optstr, NULL, NULL, NULL, NULL, NULL,
277                                 NULL);
278           free (optstr);
279           if (rc)
280             rc = map_assuan_err (rc);
281         }
282     }
283 #if defined(HAVE_SETLOCALE) && defined(LC_MESSAGES)
284   if (old_lc)
285     {
286       setlocale (LC_MESSAGES, old_lc);
287       free (old_lc);
288     }
289 #endif
290
291   return rc;
292 }
293
294
295 static AssuanError
296 membuf_data_cb (void *opaque, const void *buffer, size_t length)
297 {
298   membuf_t *data = opaque;
299
300   if (buffer)
301     put_membuf (data, buffer, length);
302   return 0;
303 }
304   
305
306 \f
307 #if 0
308 /* Handle a KEYPARMS inquiry.  Note, we only send the data,
309    assuan_transact takes care of flushing and writing the end */
310 static AssuanError
311 inq_genkey_parms (void *opaque, const char *keyword)
312 {
313   struct genkey_parm_s *parm = opaque; 
314   AssuanError rc;
315
316   rc = assuan_send_data (parm->ctx, parm->sexp, parm->sexplen);
317   return rc; 
318 }
319
320
321 \f
322 /* Call the agent to generate a new key */
323 int
324 agent_genkey (KsbaConstSexp keyparms, KsbaSexp *r_pubkey)
325 {
326   int rc;
327   struct genkey_parm_s gk_parm;
328   membuf_t data;
329   size_t len;
330   char *buf;
331
332   *r_pubkey = NULL;
333   rc = start_agent ();
334   if (rc)
335     return rc;
336
337   rc = assuan_transact (agent_ctx, "RESET", NULL, NULL,
338                         NULL, NULL, NULL, NULL);
339   if (rc)
340     return map_assuan_err (rc);
341
342   init_membuf (&data, 1024);
343   gk_parm.ctx = agent_ctx;
344   gk_parm.sexp = keyparms;
345   gk_parm.sexplen = gcry_sexp_canon_len (keyparms, 0, NULL, NULL);
346   if (!gk_parm.sexplen)
347     return gpg_error (GPG_ERR_INV_VALUE);
348   rc = assuan_transact (agent_ctx, "GENKEY",
349                         membuf_data_cb, &data, 
350                         inq_genkey_parms, &gk_parm, NULL, NULL);
351   if (rc)
352     {
353       xfree (get_membuf (&data, &len));
354       return map_assuan_err (rc);
355     }
356   buf = get_membuf (&data, &len);
357   if (!buf)
358     return gpg_error (GPG_ERR_ENOMEM);
359   if (!gcry_sexp_canon_len (buf, len, NULL, NULL))
360     {
361       xfree (buf);
362       return gpg_error (GPG_ERR_INV_SEXP);
363     }
364   *r_pubkey = buf;
365   return 0;
366 }
367 #endif /*0*/
368
369
370 \f
371 /* Ask the agent whether the corresponding secret key is available for
372    the given keygrip. */
373 int
374 agent_havekey (const char *hexkeygrip)
375 {
376   int rc;
377   char line[ASSUAN_LINELENGTH];
378
379   rc = start_agent ();
380   if (rc)
381     return rc;
382
383   if (!hexkeygrip || strlen (hexkeygrip) != 40)
384     return gpg_error (GPG_ERR_INV_VALUE);
385
386   snprintf (line, DIM(line)-1, "HAVEKEY %s", hexkeygrip);
387   line[DIM(line)-1] = 0;
388
389   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
390   return map_assuan_err (rc);
391 }
392
393 \f
394 /* Ask the agent to change the passphrase of the key identified by
395    HEXKEYGRIP. */
396 int
397 agent_passwd (const char *hexkeygrip)
398 {
399   int rc;
400   char line[ASSUAN_LINELENGTH];
401
402   rc = start_agent ();
403   if (rc)
404     return rc;
405
406   if (!hexkeygrip || strlen (hexkeygrip) != 40)
407     return gpg_error (GPG_ERR_INV_VALUE);
408
409   snprintf (line, DIM(line)-1, "PASSWD %s", hexkeygrip);
410   line[DIM(line)-1] = 0;
411
412   rc = assuan_transact (agent_ctx, line, NULL, NULL, NULL, NULL, NULL, NULL);
413   return map_assuan_err (rc);
414 }
415