Blame tests/gpg/t-verify.c

Packit d7e8d0
/* t-verify.c - Regression test.
Packit d7e8d0
   Copyright (C) 2000 Werner Koch (dd9jn)
Packit d7e8d0
   Copyright (C) 2001, 2002, 2003, 2004 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, write to the Free Software
Packit d7e8d0
   Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
Packit d7e8d0
   02111-1307, USA.  */
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
Packit d7e8d0
#include <gpgme.h>
Packit d7e8d0
Packit d7e8d0
#include "t-support.h"
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
static const char test_text1[] = "Just GNU it!\n";
Packit d7e8d0
static const char test_text1f[]= "Just GNU it?\n";
Packit d7e8d0
static const char test_sig1[] =
Packit d7e8d0
#if 0
Packit d7e8d0
"-----BEGIN PGP SIGNATURE-----\n"
Packit d7e8d0
"\n"
Packit d7e8d0
"iEYEABECAAYFAjoKgjIACgkQLXJ8x2hpdzQMSwCeO/xUrhysZ7zJKPf/FyXA//u1\n"
Packit d7e8d0
"ZgIAn0204PBR7yxSdQx6CFxugstNqmRv\n"
Packit d7e8d0
"=yku6\n"
Packit d7e8d0
"-----END PGP SIGNATURE-----\n"
Packit d7e8d0
#elif 0
Packit d7e8d0
"-----BEGIN PGP SIGNATURE-----\n"
Packit d7e8d0
"Version: GnuPG v1.0.4-2 (GNU/Linux)\n"
Packit d7e8d0
"Comment: For info see http://www.gnupg.org\n"
Packit d7e8d0
"\n"
Packit d7e8d0
"iJcEABECAFcFAjoS8/E1FIAAAAAACAAkZm9vYmFyLjF0aGlzIGlzIGEgbm90YXRp\n"
Packit d7e8d0
"b24gZGF0YSB3aXRoIDIgbGluZXMaGmh0dHA6Ly93d3cuZ3Uub3JnL3BvbGljeS8A\n"
Packit d7e8d0
"CgkQLXJ8x2hpdzQLyQCbBW/fgU8ZeWSlWPM1F8umHX17bAAAoIfSNDSp5zM85XcG\n"
Packit d7e8d0
"iwxMrf+u8v4r\n"
Packit d7e8d0
"=88Zo\n"
Packit d7e8d0
"-----END PGP SIGNATURE-----\n"
Packit d7e8d0
#elif 1
Packit d7e8d0
"-----BEGIN PGP SIGNATURE-----\n"
Packit d7e8d0
"\n"
Packit d7e8d0
"iN0EABECAJ0FAjoS+i9FFIAAAAAAAwA5YmFyw7bDpMO8w58gZGFzIHdhcmVuIFVt\n"
Packit d7e8d0
"bGF1dGUgdW5kIGpldHp0IGVpbiBwcm96ZW50JS1aZWljaGVuNRSAAAAAAAgAJGZv\n"
Packit d7e8d0
"b2Jhci4xdGhpcyBpcyBhIG5vdGF0aW9uIGRhdGEgd2l0aCAyIGxpbmVzGhpodHRw\n"
Packit d7e8d0
"Oi8vd3d3Lmd1Lm9yZy9wb2xpY3kvAAoJEC1yfMdoaXc0JBIAoIiLlUsvpMDOyGEc\n"
Packit d7e8d0
"dADGKXF/Hcb+AKCJWPphZCphduxSvrzH0hgzHdeQaA==\n"
Packit d7e8d0
"=nts1\n"
Packit d7e8d0
"-----END PGP SIGNATURE-----\n"
Packit d7e8d0
#endif
Packit d7e8d0
;
Packit d7e8d0
static const char test_sig2[] =
Packit d7e8d0
"-----BEGIN PGP MESSAGE-----\n"
Packit d7e8d0
"\n"
Packit d7e8d0
"owGbwMvMwCSoW1RzPCOz3IRxjXQSR0lqcYleSUWJTZOvjVdpcYmCu1+oQmaJIleH\n"
Packit d7e8d0
"GwuDIBMDGysTSIqBi1MApi+nlGGuwDeHao53HBr+FoVGP3xX+kvuu9fCMJvl6IOf\n"
Packit d7e8d0
"y1kvP4y+8D5a11ang0udywsA\n"
Packit d7e8d0
"=Crq6\n"
Packit d7e8d0
"-----END PGP MESSAGE-----\n";
Packit d7e8d0
Packit d7e8d0
/* A message with a prepended but unsigned plaintext packet. */
Packit d7e8d0
static const char double_plaintext_sig[] =
Packit d7e8d0
"-----BEGIN PGP MESSAGE-----\n"
Packit d7e8d0
"\n"
Packit d7e8d0
"rDRiCmZvb2Jhci50eHRF4pxNVGhpcyBpcyBteSBzbmVha3kgcGxhaW50ZXh0IG1l\n"
Packit d7e8d0
"c3NhZ2UKowGbwMvMwCSoW1RzPCOz3IRxTWISa6JebnG666MFD1wzSzJSixQ81XMV\n"
Packit d7e8d0
"UlITUxTyixRyKxXKE0uSMxQyEosVikvyCwpSU/S4FNCArq6Ce1F+aXJGvoJvYlGF\n"
Packit d7e8d0
"erFCTmJxiUJ5flFKMVeHGwuDIBMDGysTyA4GLk4BmO036xgWzMgzt9V85jCtfDFn\n"
Packit d7e8d0
"UqVooWlGXHwNw/xg/fVzt9VNbtjtJ/fhUqYo0/LyCGEA\n"
Packit d7e8d0
"=6+AK\n"
Packit d7e8d0
"-----END PGP MESSAGE-----\n";
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
check_result (gpgme_verify_result_t result, unsigned int summary,
Packit d7e8d0
              const char *fpr,
Packit d7e8d0
	      gpgme_error_t status, int notation)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_signature_t sig;
Packit d7e8d0
Packit d7e8d0
  sig = result->signatures;
Packit d7e8d0
  if (!sig || sig->next)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected number of signatures\n",
Packit d7e8d0
	       __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (sig->summary != summary)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected signature summary: "
Packit d7e8d0
               "want=0x%x have=0x%x\n",
Packit d7e8d0
	       __FILE__, __LINE__, summary, sig->summary);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (strcmp (sig->fpr, fpr))
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected fingerprint: %s\n",
Packit d7e8d0
	       __FILE__, __LINE__, sig->fpr);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (gpgme_err_code (sig->status) != status)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected signature status: %s\n",
Packit d7e8d0
	       __FILE__, __LINE__, gpgme_strerror (sig->status));
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (notation)
Packit d7e8d0
    {
Packit d7e8d0
      static struct {
Packit d7e8d0
        const char *name;
Packit d7e8d0
        const char *value;
Packit d7e8d0
        int seen;
Packit d7e8d0
      } expected_notations[] = {
Packit d7e8d0
        { "bar",
Packit d7e8d0
	  "\xc3\xb6\xc3\xa4\xc3\xbc\xc3\x9f"
Packit d7e8d0
          " das waren Umlaute und jetzt ein prozent%-Zeichen" },
Packit d7e8d0
        { "foobar.1",
Packit d7e8d0
	  "this is a notation data with 2 lines" },
Packit d7e8d0
        { NULL,
Packit d7e8d0
	  "http://www.gu.org/policy/" }
Packit d7e8d0
      };
Packit d7e8d0
      int i;
Packit d7e8d0
      gpgme_sig_notation_t r;
Packit d7e8d0
Packit d7e8d0
      for (i=0; i < DIM(expected_notations); i++ )
Packit d7e8d0
        expected_notations[i].seen = 0;
Packit d7e8d0
Packit d7e8d0
      for (r = sig->notations; r; r = r->next)
Packit d7e8d0
        {
Packit d7e8d0
          int any = 0;
Packit d7e8d0
          for (i=0; i < DIM(expected_notations); i++)
Packit d7e8d0
            {
Packit d7e8d0
              if ( ((r->name && expected_notations[i].name
Packit d7e8d0
                     && !strcmp (r->name, expected_notations[i].name)
Packit d7e8d0
		     && r->name_len
Packit d7e8d0
		     == strlen (expected_notations[i].name))
Packit d7e8d0
                    || (!r->name && !expected_notations[i].name
Packit d7e8d0
			&& r->name_len == 0))
Packit d7e8d0
                   && r->value
Packit d7e8d0
                   && !strcmp (r->value, expected_notations[i].value)
Packit d7e8d0
		   && r->value_len == strlen (expected_notations[i].value))
Packit d7e8d0
                {
Packit d7e8d0
                  expected_notations[i].seen++;
Packit d7e8d0
                  any++;
Packit d7e8d0
                }
Packit d7e8d0
            }
Packit d7e8d0
          if (!any)
Packit d7e8d0
            {
Packit d7e8d0
              fprintf (stderr, "%s:%i: Unexpected notation data\n",
Packit d7e8d0
                       __FILE__, __LINE__);
Packit d7e8d0
              exit (1);
Packit d7e8d0
            }
Packit d7e8d0
        }
Packit d7e8d0
      for (i=0; i < DIM(expected_notations); i++ )
Packit d7e8d0
        {
Packit d7e8d0
          if (expected_notations[i].seen != 1)
Packit d7e8d0
            {
Packit d7e8d0
              fprintf (stderr, "%s:%i: Missing or duplicate notation data\n",
Packit d7e8d0
                       __FILE__, __LINE__);
Packit d7e8d0
              exit (1);
Packit d7e8d0
            }
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
  if (sig->wrong_key_usage)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpectedly wrong key usage\n",
Packit d7e8d0
	       __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (sig->validity != GPGME_VALIDITY_UNKNOWN)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected validity: %i\n",
Packit d7e8d0
	       __FILE__, __LINE__, sig->validity);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  if (gpgme_err_code (sig->validity_reason) != GPG_ERR_NO_ERROR)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Unexpected validity reason: %s\n",
Packit d7e8d0
	       __FILE__, __LINE__, gpgme_strerror (sig->validity_reason));
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
int
Packit d7e8d0
main (int argc, char *argv[])
Packit d7e8d0
{
Packit d7e8d0
  gpgme_ctx_t ctx;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  gpgme_data_t sig, text;
Packit d7e8d0
  gpgme_verify_result_t result;
Packit d7e8d0
  const char *s;
Packit d7e8d0
Packit d7e8d0
  (void)argc;
Packit d7e8d0
  (void)argv;
Packit d7e8d0
Packit d7e8d0
  init_gpgme (GPGME_PROTOCOL_OpenPGP);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_new (&ctx;;
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
Packit d7e8d0
  /* Checking a valid message.  */
Packit d7e8d0
  err = gpgme_data_new_from_mem (&text, test_text1, strlen (test_text1), 0);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_data_new_from_mem (&sig, test_sig1, strlen (test_sig1), 0);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_op_verify (ctx, sig, text, NULL);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  result = gpgme_op_verify_result (ctx);
Packit d7e8d0
  check_result (result, 0, "A0FF4590BB6122EDEF6E3C542D727CC768697734",
Packit d7e8d0
		GPG_ERR_NO_ERROR, 1);
Packit d7e8d0
Packit d7e8d0
  /* Checking a manipulated message.  */
Packit d7e8d0
  gpgme_data_release (text);
Packit d7e8d0
  err = gpgme_data_new_from_mem (&text, test_text1f, strlen (test_text1f), 0);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  gpgme_data_seek (sig, 0, SEEK_SET);
Packit d7e8d0
  err = gpgme_op_verify (ctx, sig, text, NULL);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  result = gpgme_op_verify_result (ctx);
Packit d7e8d0
  check_result (result, GPGME_SIGSUM_RED, "2D727CC768697734",
Packit d7e8d0
		GPG_ERR_BAD_SIGNATURE, 0);
Packit d7e8d0
Packit d7e8d0
  /* Checking a normal signature.  */
Packit d7e8d0
  gpgme_data_release (sig);
Packit d7e8d0
  gpgme_data_release (text);
Packit d7e8d0
  err = gpgme_data_new_from_mem (&sig, test_sig2, strlen (test_sig2), 0);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_data_new (&text);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_op_verify (ctx, sig, NULL, text);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  result = gpgme_op_verify_result (ctx);
Packit d7e8d0
  check_result (result, 0, "A0FF4590BB6122EDEF6E3C542D727CC768697734",
Packit d7e8d0
		GPG_ERR_NO_ERROR, 0);
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
  /* Checking an invalid message.  */
Packit d7e8d0
  gpgme_data_release (sig);
Packit d7e8d0
  gpgme_data_release (text);
Packit d7e8d0
  err = gpgme_data_new_from_mem (&sig, double_plaintext_sig,
Packit d7e8d0
                                 strlen (double_plaintext_sig), 0);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_data_new (&text);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  err = gpgme_op_verify (ctx, sig, NULL, text);
Packit d7e8d0
  if (gpgme_err_code (err) != GPG_ERR_BAD_DATA)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: Double plaintext message not detected\n",
Packit d7e8d0
	       __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  /* Checking that set/get_sernder works.  */
Packit d7e8d0
  err = gpgme_set_sender (ctx, "foo@example.org");
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  s = gpgme_get_sender (ctx);
Packit d7e8d0
  if (!s || strcmp (s, "foo@example.org"))
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: gpgme_{set,get}_sender mismatch\n",
Packit d7e8d0
               __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_set_sender (ctx, "<bar@example.org>");
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  s = gpgme_get_sender (ctx);
Packit d7e8d0
  if (!s || strcmp (s, "bar@example.org"))
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: gpgme_{set,get}_sender mismatch\n",
Packit d7e8d0
               __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_set_sender (ctx, "Foo bar (comment) <foo@example.org>");
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  s = gpgme_get_sender (ctx);
Packit d7e8d0
  if (!s || strcmp (s, "foo@example.org"))
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: gpgme_{set,get}_sender mismatch\n",
Packit d7e8d0
               __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_set_sender (ctx, "foo");
Packit d7e8d0
  if (gpgme_err_code (err) != GPG_ERR_INV_VALUE)
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: gpgme_set_sender didn't detect bogus address\n",
Packit d7e8d0
               __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
  /* (the former address should still be there.)  */
Packit d7e8d0
  s = gpgme_get_sender (ctx);
Packit d7e8d0
  if (!s || strcmp (s, "foo@example.org"))
Packit d7e8d0
    {
Packit d7e8d0
      fprintf (stderr, "%s:%i: gpgme_{set,get}_sender mismatch\n",
Packit d7e8d0
               __FILE__, __LINE__);
Packit d7e8d0
      exit (1);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
  gpgme_data_release (sig);
Packit d7e8d0
  gpgme_data_release (text);
Packit d7e8d0
  gpgme_release (ctx);
Packit d7e8d0
  return 0;
Packit d7e8d0
}