Blame tests/run-verify.c

Packit d7e8d0
/* run-verify.c  - Helper to perform a verify operation
Packit d7e8d0
   Copyright (C) 2009 g10 Code GmbH
Packit d7e8d0
Packit d7e8d0
   This file is part of GPGME.
Packit d7e8d0
Packit d7e8d0
   GPGME is free software; you can redistribute it and/or modify it
Packit d7e8d0
   under the terms of the GNU Lesser General Public License as
Packit d7e8d0
   published by the Free Software Foundation; either version 2.1 of
Packit d7e8d0
   the License, or (at your option) any later version.
Packit d7e8d0
Packit d7e8d0
   GPGME is distributed in the hope that it will be useful, but
Packit d7e8d0
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit d7e8d0
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit d7e8d0
   Lesser General Public License for more details.
Packit d7e8d0
Packit d7e8d0
   You should have received a copy of the GNU Lesser General Public
Packit d7e8d0
   License along with this program; if not, see <https://www.gnu.org/licenses/>.
Packit d7e8d0
*/
Packit d7e8d0
Packit d7e8d0
/* We need to include config.h so that we know whether we are building
Packit d7e8d0
   with large file system (LFS) support. */
Packit d7e8d0
#ifdef HAVE_CONFIG_H
Packit d7e8d0
#include <config.h>
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
#include <stdlib.h>
Packit d7e8d0
#include <stdio.h>
Packit d7e8d0
#include <string.h>
Packit d7e8d0
#include <time.h>
Packit d7e8d0
Packit d7e8d0
#include <gpgme.h>
Packit d7e8d0
Packit d7e8d0
#define PGM "run-verify"
Packit d7e8d0
Packit d7e8d0
#include "run-support.h"
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static int verbose;
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static const char *
Packit d7e8d0
isotimestr (unsigned long value)
Packit d7e8d0
{
Packit d7e8d0
  time_t t;
Packit d7e8d0
  static char buffer[25+5];
Packit d7e8d0
  struct tm *tp;
Packit d7e8d0
Packit d7e8d0
  if (!value)
Packit d7e8d0
    return "none";
Packit d7e8d0
  t = value;
Packit d7e8d0
Packit d7e8d0
  tp = gmtime (&t);
Packit d7e8d0
  snprintf (buffer, sizeof buffer, "%04d-%02d-%02d %02d:%02d:%02d",
Packit d7e8d0
            1900+tp->tm_year, tp->tm_mon+1, tp->tm_mday,
Packit d7e8d0
            tp->tm_hour, tp->tm_min, tp->tm_sec);
Packit d7e8d0
  return buffer;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpg_error_t
Packit d7e8d0
status_cb (void *opaque, const char *keyword, const char *value)
Packit d7e8d0
{
Packit d7e8d0
  (void)opaque;
Packit d7e8d0
  fprintf (stderr, "status_cb: %s %s\n", keyword, value);
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
print_summary (gpgme_sigsum_t summary)
Packit d7e8d0
{
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_VALID      ))
Packit d7e8d0
    fputs (" valid", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_GREEN      ))
Packit d7e8d0
    fputs (" green", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_RED        ))
Packit d7e8d0
    fputs (" red", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_KEY_REVOKED))
Packit d7e8d0
    fputs (" revoked", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_KEY_EXPIRED))
Packit d7e8d0
    fputs (" key-expired", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_SIG_EXPIRED))
Packit d7e8d0
    fputs (" sig-expired", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_KEY_MISSING))
Packit d7e8d0
    fputs (" key-missing", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_CRL_MISSING))
Packit d7e8d0
    fputs (" crl-missing", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_CRL_TOO_OLD))
Packit d7e8d0
    fputs (" crl-too-old", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_BAD_POLICY ))
Packit d7e8d0
    fputs (" bad-policy", stdout);
Packit d7e8d0
  if ( (summary & GPGME_SIGSUM_SYS_ERROR  ))
Packit d7e8d0
    fputs (" sys-error", stdout);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
print_validity (gpgme_validity_t val)
Packit d7e8d0
{
Packit d7e8d0
  const char *s = NULL;
Packit d7e8d0
Packit d7e8d0
  switch (val)
Packit d7e8d0
    {
Packit d7e8d0
    case GPGME_VALIDITY_UNKNOWN:  s = "unknown"; break;
Packit d7e8d0
    case GPGME_VALIDITY_UNDEFINED:s = "undefined"; break;
Packit d7e8d0
    case GPGME_VALIDITY_NEVER:    s = "never"; break;
Packit d7e8d0
    case GPGME_VALIDITY_MARGINAL: s = "marginal"; break;
Packit d7e8d0
    case GPGME_VALIDITY_FULL:     s = "full"; break;
Packit d7e8d0
    case GPGME_VALIDITY_ULTIMATE: s = "ultimate"; break;
Packit d7e8d0
    }
Packit d7e8d0
  if (s)
Packit d7e8d0
    fputs (s, stdout);
Packit d7e8d0
  else
Packit d7e8d0
    printf ("[bad validity value %u]", (unsigned int)val);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
print_description (const char *text, int indent)
Packit d7e8d0
{
Packit d7e8d0
  for (; *text; text++)
Packit d7e8d0
    {
Packit d7e8d0
      putchar (*text);
Packit d7e8d0
      if (*text == '\n')
Packit d7e8d0
        printf ("%*s", indent, "");
Packit d7e8d0
    }
Packit d7e8d0
  putchar ('\n');
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
print_result (gpgme_verify_result_t result)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_signature_t sig;
Packit d7e8d0
  gpgme_sig_notation_t nt;
Packit d7e8d0
  gpgme_user_id_t uid;
Packit d7e8d0
  gpgme_tofu_info_t ti;
Packit d7e8d0
  int count = 0;
Packit d7e8d0
Packit d7e8d0
  printf ("Original file name: %s\n", nonnull(result->file_name));
Packit d7e8d0
  for (sig = result->signatures; sig; sig = sig->next)
Packit d7e8d0
    {
Packit d7e8d0
      printf ("Signature %d\n", count++);
Packit d7e8d0
      printf ("  status ....: %s\n", gpgme_strerror (sig->status));
Packit d7e8d0
      printf ("  summary ...:"); print_summary (sig->summary); putchar ('\n');
Packit d7e8d0
      printf ("  fingerprint: %s\n", nonnull (sig->fpr));
Packit d7e8d0
      printf ("  created ...: %lu\n", sig->timestamp);
Packit d7e8d0
      printf ("  expires ...: %lu\n", sig->exp_timestamp);
Packit d7e8d0
      printf ("  validity ..: ");
Packit d7e8d0
      print_validity (sig->validity); putchar ('\n');
Packit d7e8d0
      printf ("  val.reason : %s\n", gpgme_strerror (sig->status));
Packit d7e8d0
      printf ("  pubkey algo: %d (%s)\n", sig->pubkey_algo,
Packit d7e8d0
              nonnull(gpgme_pubkey_algo_name (sig->pubkey_algo)));
Packit d7e8d0
      printf ("  digest algo: %d (%s)\n", sig->hash_algo,
Packit d7e8d0
              nonnull(gpgme_hash_algo_name (sig->hash_algo)));
Packit d7e8d0
      printf ("  pka address: %s\n", nonnull (sig->pka_address));
Packit d7e8d0
      printf ("  pka trust .: %s\n",
Packit d7e8d0
              sig->pka_trust == 0? "n/a" :
Packit d7e8d0
              sig->pka_trust == 1? "bad" :
Packit d7e8d0
              sig->pka_trust == 2? "okay": "RFU");
Packit d7e8d0
      printf ("  other flags:%s%s\n",
Packit d7e8d0
              sig->wrong_key_usage? " wrong-key-usage":"",
Packit d7e8d0
              sig->chain_model? " chain-model":""
Packit d7e8d0
              );
Packit d7e8d0
      for (nt = sig->notations; nt; nt = nt->next)
Packit d7e8d0
        {
Packit d7e8d0
          printf ("  notation ..: '%s'\n", nt->name);
Packit d7e8d0
          if (strlen (nt->name) != nt->name_len)
Packit d7e8d0
            printf ("    warning : name larger (%d)\n", nt->name_len);
Packit d7e8d0
          printf ("    flags ...:%s%s (0x%02x)\n",
Packit d7e8d0
                  nt->critical? " critical":"",
Packit d7e8d0
                  nt->human_readable? " human":"",
Packit d7e8d0
                  nt->flags);
Packit d7e8d0
          if (nt->value)
Packit d7e8d0
            printf ("    value ...: '%s'\n", nt->value);
Packit d7e8d0
          if ((nt->value?strlen (nt->value):0) != nt->value_len)
Packit d7e8d0
            printf ("    warning : value larger (%d)\n", nt->value_len);
Packit d7e8d0
        }
Packit d7e8d0
      if (sig->key)
Packit d7e8d0
        {
Packit d7e8d0
          printf ("  primary fpr: %s\n", nonnull (sig->key->fpr));
Packit d7e8d0
          for (uid = sig->key->uids; uid; uid = uid->next)
Packit d7e8d0
            {
Packit d7e8d0
              printf ("  tofu addr .: %s\n", nonnull (uid->address));
Packit d7e8d0
              ti = uid->tofu;
Packit d7e8d0
              if (!ti)
Packit d7e8d0
                continue;
Packit d7e8d0
              printf ("    validity : %u (%s)\n", ti->validity,
Packit d7e8d0
                      ti->validity == 0? "conflict" :
Packit d7e8d0
                      ti->validity == 1? "no history" :
Packit d7e8d0
                      ti->validity == 2? "little history" :
Packit d7e8d0
                      ti->validity == 3? "enough history" :
Packit d7e8d0
                      ti->validity == 4? "lot of history" : "?");
Packit d7e8d0
              printf ("    policy ..: %u (%s)\n", ti->policy,
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_NONE? "none" :
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_AUTO? "auto" :
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_GOOD? "good" :
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_UNKNOWN? "unknown" :
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_BAD? "bad" :
Packit d7e8d0
                      ti->policy == GPGME_TOFU_POLICY_ASK? "ask" : "?");
Packit d7e8d0
              printf ("    signcount: %hu\n", ti->signcount);
Packit d7e8d0
              printf ("      first..: %s\n", isotimestr (ti->signfirst));
Packit d7e8d0
              printf ("      last ..: %s\n", isotimestr (ti->signlast));
Packit d7e8d0
              printf ("    encrcount: %hu\n", ti->encrcount);
Packit d7e8d0
              printf ("      first..: %s\n", isotimestr (ti->encrfirst));
Packit d7e8d0
              printf ("      last ..: %s\n", isotimestr (ti->encrlast));
Packit d7e8d0
              printf ("    desc ....: ");
Packit d7e8d0
              print_description (nonnull (ti->description), 15);
Packit d7e8d0
            }
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static int
Packit d7e8d0
show_usage (int ex)
Packit d7e8d0
{
Packit d7e8d0
  fputs ("usage: " PGM " [options] [DETACHEDSIGFILE] FILE\n\n"
Packit d7e8d0
         "Options:\n"
Packit d7e8d0
         "  --verbose        run in verbose mode\n"
Packit d7e8d0
         "  --status         print status lines from the backend\n"
Packit d7e8d0
         "  --openpgp        use the OpenPGP protocol (default)\n"
Packit d7e8d0
         "  --cms            use the CMS protocol\n"
Packit d7e8d0
         "  --sender MBOX    use MBOX as sender address\n"
Packit d7e8d0
         "  --auto-key-retrieve\n"
Packit d7e8d0
         , stderr);
Packit d7e8d0
  exit (ex);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
int
Packit d7e8d0
main (int argc, char **argv)
Packit d7e8d0
{
Packit d7e8d0
  int last_argc = -1;
Packit d7e8d0
  const char *s;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  gpgme_ctx_t ctx;
Packit d7e8d0
  gpgme_protocol_t protocol = GPGME_PROTOCOL_OpenPGP;
Packit d7e8d0
  FILE *fp_sig = NULL;
Packit d7e8d0
  gpgme_data_t sig = NULL;
Packit d7e8d0
  FILE *fp_msg = NULL;
Packit d7e8d0
  gpgme_data_t msg = NULL;
Packit d7e8d0
  gpgme_verify_result_t result;
Packit d7e8d0
  int print_status = 0;
Packit d7e8d0
  const char *sender = NULL;
Packit d7e8d0
  int auto_key_retrieve = 0;
Packit d7e8d0
Packit d7e8d0
  if (argc)
Packit d7e8d0
    { argc--; argv++; }
Packit d7e8d0
Packit d7e8d0
  while (argc && last_argc != argc )
Packit d7e8d0
    {
Packit d7e8d0
      last_argc = argc;
Packit d7e8d0
      if (!strcmp (*argv, "--"))
Packit d7e8d0
        {
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
          break;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--help"))
Packit d7e8d0
        show_usage (0);
Packit d7e8d0
      else if (!strcmp (*argv, "--verbose"))
Packit d7e8d0
        {
Packit d7e8d0
          verbose = 1;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--status"))
Packit d7e8d0
        {
Packit d7e8d0
          print_status = 1;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--openpgp"))
Packit d7e8d0
        {
Packit d7e8d0
          protocol = GPGME_PROTOCOL_OpenPGP;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--cms"))
Packit d7e8d0
        {
Packit d7e8d0
          protocol = GPGME_PROTOCOL_CMS;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--sender"))
Packit d7e8d0
        {
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
          if (!argc)
Packit d7e8d0
            show_usage (1);
Packit d7e8d0
          sender = *argv;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
      else if (!strcmp (*argv, "--auto-key-retrieve"))
Packit d7e8d0
        {
Packit d7e8d0
          auto_key_retrieve = 1;
Packit d7e8d0
          argc--; argv++;
Packit d7e8d0
        }
Packit d7e8d0
Packit d7e8d0
      else if (!strncmp (*argv, "--", 2))
Packit d7e8d0
        show_usage (1);
Packit d7e8d0
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  if (argc < 1 || argc > 2)
Packit d7e8d0
    show_usage (1);
Packit d7e8d0
Packit d7e8d0
  fp_sig = fopen (argv[0], "rb");
Packit d7e8d0
  if (!fp_sig)
Packit d7e8d0
    {
Packit d7e8d0
      err = gpgme_error_from_syserror ();
Packit d7e8d0
      fprintf (stderr, PGM ": can't open `%s': %s\n",
Packit d7e8d0
               argv[0], gpgme_strerror (err));
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (argc > 1)
Packit d7e8d0
    {
Packit d7e8d0
      fp_msg = fopen (argv[1], "rb");
Packit d7e8d0
      if (!fp_msg)
Packit d7e8d0
        {
Packit d7e8d0
          err = gpgme_error_from_syserror ();
Packit d7e8d0
          fprintf (stderr, PGM ": can't open `%s': %s\n",
Packit d7e8d0
                   argv[1], gpgme_strerror (err));
Packit d7e8d0
          exit (1);
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  init_gpgme (protocol);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_new (&ctx;;
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  gpgme_set_protocol (ctx, protocol);
Packit d7e8d0
  if (print_status)
Packit d7e8d0
    {
Packit d7e8d0
      gpgme_set_status_cb (ctx, status_cb, NULL);
Packit d7e8d0
      gpgme_set_ctx_flag (ctx, "full-status", "1");
Packit d7e8d0
    }
Packit d7e8d0
  /* gpgme_set_ctx_flag (ctx, "raw-description", "1"); */
Packit d7e8d0
Packit d7e8d0
  if (auto_key_retrieve)
Packit d7e8d0
    {
Packit d7e8d0
      gpgme_set_ctx_flag (ctx, "auto-key-retrieve", "1");
Packit d7e8d0
      s = gpgme_get_ctx_flag (ctx, "auto-key-retrieve");
Packit d7e8d0
      if (!s || strcmp (s, "1"))
Packit d7e8d0
        {
Packit d7e8d0
          fprintf (stderr, PGM ": gpgme_get_ctx_flag failed for '%s'\n",
Packit d7e8d0
                   "auto-key-retrieve");
Packit d7e8d0
          exit (1);
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  if (sender)
Packit d7e8d0
    {
Packit d7e8d0
      err = gpgme_set_sender (ctx, sender);
Packit d7e8d0
      fail_if_err (err);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_data_new_from_stream (&sig, fp_sig);
Packit d7e8d0
  if (err)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, PGM ": error allocating data object: %s\n",
Packit d7e8d0
               gpgme_strerror (err));
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (fp_msg)
Packit d7e8d0
    {
Packit d7e8d0
      err = gpgme_data_new_from_stream (&msg, fp_msg);
Packit d7e8d0
      if (err)
Packit d7e8d0
        {
Packit d7e8d0
          fprintf (stderr, PGM ": error allocating data object: %s\n",
Packit d7e8d0
                   gpgme_strerror (err));
Packit d7e8d0
          exit (1);
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_op_verify (ctx, sig, msg, NULL);
Packit d7e8d0
  result = gpgme_op_verify_result (ctx);
Packit d7e8d0
  if (result)
Packit d7e8d0
    print_result (result);
Packit d7e8d0
  if (err)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, PGM ": verify failed: %s\n", gpgme_strerror (err));
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  gpgme_data_release (msg);
Packit d7e8d0
  gpgme_data_release (sig);
Packit d7e8d0
Packit d7e8d0
  gpgme_release (ctx);
Packit d7e8d0
  return 0;
Packit d7e8d0
}