Blame gio/tests/tls-certificate.c

Packit ae235b
/* GLib testing framework examples and tests
Packit ae235b
 *
Packit ae235b
 * Copyright (C) 2011 Collabora Ltd.
Packit ae235b
 *
Packit ae235b
 * This library is free software; you can redistribute it and/or
Packit ae235b
 * modify it under the terms of the GNU Lesser General Public
Packit ae235b
 * License as published by the Free Software Foundation; either
Packit ae235b
 * version 2.1 of the License, or (at your option) any later version.
Packit ae235b
 *
Packit ae235b
 * This library is distributed in the hope that it will be useful,
Packit ae235b
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit ae235b
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit ae235b
 * Lesser General Public License for more details.
Packit ae235b
 *
Packit ae235b
 * You should have received a copy of the GNU Lesser General
Packit ae235b
 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit ae235b
 *
Packit ae235b
 * Author: Nicolas Dufresne <nicolas.dufresne@collabora.com>
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include "config.h"
Packit ae235b
Packit ae235b
#include <gio/gio.h>
Packit ae235b
Packit ae235b
#include "gtesttlsbackend.h"
Packit ae235b
Packit ae235b
typedef struct
Packit ae235b
{
Packit ae235b
  gchar *cert_pems[3];
Packit ae235b
  gchar *key_pem;
Packit ae235b
  gchar *key8_pem;
Packit ae235b
} Reference;
Packit ae235b
Packit ae235b
static void
Packit ae235b
pem_parser (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GTlsCertificate *cert;
Packit ae235b
  gchar *pem;
Packit ae235b
  gchar *parsed_cert_pem = NULL;
Packit ae235b
  const gchar *parsed_key_pem = NULL;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  /* Check PEM parsing in certificate, private key order. */
Packit ae235b
  g_file_get_contents (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert-key.pem", NULL), &pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (pem);
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_pem (pem, -1, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  g_object_unref (cert);
Packit ae235b
Packit ae235b
  /* Make sure length is respected and parser detect invalid (truncated) PEM. */
Packit ae235b
  cert = g_tls_certificate_new_from_pem (pem, 10, &error);
Packit ae235b
  g_assert_error (error, G_TLS_ERROR, G_TLS_ERROR_BAD_CERTIFICATE);
Packit ae235b
  g_clear_error (&error);
Packit ae235b
  g_free (pem);
Packit ae235b
Packit ae235b
  /* Check PEM parsing in private key, certificate order */
Packit ae235b
  g_file_get_contents (g_test_get_filename (G_TEST_DIST, "cert-tests", "key-cert.pem", NULL), &pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (pem);
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_pem (pem, -1, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  g_free (pem);
Packit ae235b
  g_object_unref (cert);
Packit ae235b
Packit ae235b
  /* Check certificate only PEM */
Packit ae235b
  g_file_get_contents (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert1.pem", NULL), &pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (pem);
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_pem (pem, -1, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert (parsed_key_pem == NULL);
Packit ae235b
Packit ae235b
  g_free (pem);
Packit ae235b
  g_object_unref (cert);
Packit ae235b
Packit ae235b
  /* Check error with private key only PEM */
Packit ae235b
  g_file_get_contents (g_test_get_filename (G_TEST_DIST, "cert-tests", "key.pem", NULL), &pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (pem);
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_pem (pem, -1, &error);
Packit ae235b
  g_assert_error (error, G_TLS_ERROR, G_TLS_ERROR_BAD_CERTIFICATE);
Packit ae235b
  g_clear_error (&error);
Packit ae235b
  g_assert (cert == NULL);
Packit ae235b
  g_free (pem);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
pem_parser_handles_chain (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GTlsCertificate *cert;
Packit ae235b
  GTlsCertificate *issuer;
Packit ae235b
  GTlsCertificate *original_cert;
Packit ae235b
  gchar *pem;
Packit ae235b
  gchar *parsed_cert_pem = NULL;
Packit ae235b
  const gchar *parsed_key_pem = NULL;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  /* Check that a chain with exactly three certificates is returned */
Packit ae235b
  g_file_get_contents (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert-list.pem", NULL), &pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (pem);
Packit ae235b
Packit ae235b
  cert = original_cert = g_tls_certificate_new_from_pem (pem, -1, &error);
Packit ae235b
  g_free (pem);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_clear_pointer (&parsed_cert_pem, g_free);
Packit ae235b
Packit ae235b
  /* Make sure the private key was parsed */
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  /* Now test the second cert */
Packit ae235b
  issuer = g_tls_certificate_get_issuer (cert);
Packit ae235b
  g_assert (issuer);
Packit ae235b
Packit ae235b
  cert = issuer;
Packit ae235b
  issuer = g_tls_certificate_get_issuer (cert);
Packit ae235b
  g_assert (issuer);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[1]);
Packit ae235b
  g_clear_pointer (&parsed_cert_pem, g_free);
Packit ae235b
Packit ae235b
  /* Only the first cert should have a private key */
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert (!parsed_key_pem);
Packit ae235b
Packit ae235b
  /* Now test the final cert */
Packit ae235b
  cert = issuer;
Packit ae235b
  issuer = g_tls_certificate_get_issuer (cert);
Packit ae235b
  g_assert (!issuer);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[2]);
Packit ae235b
  g_clear_pointer (&parsed_cert_pem, g_free);
Packit ae235b
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert (!parsed_key_pem);
Packit ae235b
Packit ae235b
  g_object_unref (original_cert);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
from_file (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GTlsCertificate *cert;
Packit ae235b
  gchar *parsed_cert_pem = NULL;
Packit ae235b
  const gchar *parsed_key_pem = NULL;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_file (g_test_get_filename (G_TEST_DIST, "cert-tests", "key-cert.pem", NULL),
Packit ae235b
                                          &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  g_object_unref (cert);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
from_files (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GTlsCertificate *cert;
Packit ae235b
  gchar *parsed_cert_pem = NULL;
Packit ae235b
  const gchar *parsed_key_pem = NULL;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_files (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert1.pem", NULL),
Packit ae235b
                                           g_test_get_filename (G_TEST_DIST, "cert-tests", "key.pem", NULL),
Packit ae235b
                                           &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  g_object_unref (cert);
Packit ae235b
Packit ae235b
  /* Missing private key */
Packit ae235b
  cert = g_tls_certificate_new_from_files (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert1.pem", NULL),
Packit ae235b
                                           g_test_get_filename (G_TEST_DIST, "cert-tests", "cert2.pem", NULL),
Packit ae235b
                                           &error);
Packit ae235b
  g_assert_error (error, G_TLS_ERROR, G_TLS_ERROR_BAD_CERTIFICATE);
Packit ae235b
  g_clear_error (&error);
Packit ae235b
  g_assert (cert == NULL);
Packit ae235b
Packit ae235b
  /* Missing certificate */
Packit ae235b
  cert = g_tls_certificate_new_from_files (g_test_get_filename (G_TEST_DIST, "cert-tests", "key.pem", NULL),
Packit ae235b
                                           g_test_get_filename (G_TEST_DIST, "cert-tests", "key.pem", NULL),
Packit ae235b
                                           &error);
Packit ae235b
  g_assert_error (error, G_TLS_ERROR, G_TLS_ERROR_BAD_CERTIFICATE);
Packit ae235b
  g_clear_error (&error);
Packit ae235b
  g_assert (cert == NULL);
Packit ae235b
Packit ae235b
  /* Using this method twice with a file containing both private key and
Packit ae235b
   * certificate as a way to inforce private key presence is a fair use
Packit ae235b
   */
Packit ae235b
  cert = g_tls_certificate_new_from_files (g_test_get_filename (G_TEST_DIST, "cert-tests", "key-cert.pem", NULL),
Packit ae235b
                                           g_test_get_filename (G_TEST_DIST, "cert-tests", "key-cert.pem", NULL),
Packit ae235b
                                           &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
  g_object_unref (cert);
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static void
Packit ae235b
from_files_pkcs8 (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GTlsCertificate *cert;
Packit ae235b
  gchar *parsed_cert_pem = NULL;
Packit ae235b
  const gchar *parsed_key_pem = NULL;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  cert = g_tls_certificate_new_from_files (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert1.pem", NULL),
Packit ae235b
                                           g_test_get_filename (G_TEST_DIST, "cert-tests", "key8.pem", NULL),
Packit ae235b
                                           &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (cert);
Packit ae235b
Packit ae235b
  g_object_get (cert,
Packit ae235b
      "certificate-pem", &parsed_cert_pem,
Packit ae235b
      NULL);
Packit ae235b
  parsed_key_pem = g_test_tls_connection_get_private_key_pem (cert);
Packit ae235b
  g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[0]);
Packit ae235b
  g_free (parsed_cert_pem);
Packit ae235b
  parsed_cert_pem = NULL;
Packit ae235b
  g_assert_cmpstr (parsed_key_pem, ==, ref->key8_pem);
Packit ae235b
  parsed_key_pem = NULL;
Packit ae235b
Packit ae235b
  g_object_unref (cert);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
list_from_file (const Reference *ref)
Packit ae235b
{
Packit ae235b
  GList *list, *l;
Packit ae235b
  GError *error = NULL;
Packit ae235b
  int i;
Packit ae235b
Packit ae235b
  list = g_tls_certificate_list_new_from_file (g_test_get_filename (G_TEST_DIST, "cert-tests", "cert-list.pem", NULL),
Packit ae235b
                                               &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert_cmpint (g_list_length (list), ==, 3);
Packit ae235b
Packit ae235b
  l = list;
Packit ae235b
  for (i = 0; i < 3; i++)
Packit ae235b
    {
Packit ae235b
      GTlsCertificate *cert = l->data;
Packit ae235b
      gchar *parsed_cert_pem = NULL;
Packit ae235b
      g_object_get (cert,
Packit ae235b
          "certificate-pem", &parsed_cert_pem,
Packit ae235b
          NULL);
Packit ae235b
      g_assert_cmpstr (parsed_cert_pem, ==, ref->cert_pems[i]);
Packit ae235b
      g_free (parsed_cert_pem);
Packit ae235b
      l = g_list_next (l);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  g_list_free_full (list, g_object_unref);
Packit ae235b
Packit ae235b
  /* Empty list is not an error */
Packit ae235b
  list = g_tls_certificate_list_new_from_file (g_test_get_filename (G_TEST_DIST, "cert-tests", "nothing.pem", NULL),
Packit ae235b
                                               &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert_cmpint (g_list_length (list), ==, 0);
Packit ae235b
}
Packit ae235b
Packit ae235b
int
Packit ae235b
main (int   argc,
Packit ae235b
      char *argv[])
Packit ae235b
{
Packit ae235b
  int rtv;
Packit ae235b
  Reference ref;
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gchar *path;
Packit ae235b
Packit ae235b
  g_test_init (&argc, &argv, NULL);
Packit ae235b
Packit ae235b
  _g_test_tls_backend_get_type ();
Packit ae235b
Packit ae235b
  /* Load reference PEM */
Packit ae235b
  path = g_test_build_filename (G_TEST_DIST, "cert-tests", "cert1.pem", NULL);
Packit ae235b
  g_file_get_contents (path, &ref.cert_pems[0], NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (ref.cert_pems[0]);
Packit ae235b
  g_free (path);
Packit ae235b
  path = g_test_build_filename (G_TEST_DIST, "cert-tests", "cert2.pem", NULL);
Packit ae235b
  g_file_get_contents (path, &ref.cert_pems[1], NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (ref.cert_pems[1]);
Packit ae235b
  g_free (path);
Packit ae235b
  path = g_test_build_filename (G_TEST_DIST, "cert-tests", "cert3.pem", NULL);
Packit ae235b
  g_file_get_contents (path, &ref.cert_pems[2], NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (ref.cert_pems[2]);
Packit ae235b
  g_free (path);
Packit ae235b
  path = g_test_build_filename (G_TEST_DIST, "cert-tests", "key.pem", NULL);
Packit ae235b
  g_file_get_contents (path, &ref.key_pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (ref.key_pem);
Packit ae235b
  g_free (path);
Packit ae235b
  path = g_test_build_filename (G_TEST_DIST, "cert-tests", "key8.pem", NULL);
Packit ae235b
  g_file_get_contents (path, &ref.key8_pem, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
  g_assert (ref.key8_pem);
Packit ae235b
  g_free (path);
Packit ae235b
Packit ae235b
  g_test_add_data_func ("/tls-certificate/pem-parser",
Packit ae235b
                        &ref, (GTestDataFunc)pem_parser);
Packit ae235b
  g_test_add_data_func ("/tls-certificate/pem-parser-handles-chain",
Packit ae235b
                        &ref, (GTestDataFunc)pem_parser_handles_chain);
Packit ae235b
  g_test_add_data_func ("/tls-certificate/from_file",
Packit ae235b
                        &ref, (GTestDataFunc)from_file);
Packit ae235b
  g_test_add_data_func ("/tls-certificate/from_files",
Packit ae235b
                        &ref, (GTestDataFunc)from_files);
Packit ae235b
  g_test_add_data_func ("/tls-certificate/from_files_pkcs8",
Packit ae235b
                        &ref, (GTestDataFunc)from_files_pkcs8);
Packit ae235b
  g_test_add_data_func ("/tls-certificate/list_from_file",
Packit ae235b
                        &ref, (GTestDataFunc)list_from_file);
Packit ae235b
Packit ae235b
  rtv = g_test_run();
Packit ae235b
Packit ae235b
  g_free (ref.cert_pems[0]);
Packit ae235b
  g_free (ref.cert_pems[1]);
Packit ae235b
  g_free (ref.cert_pems[2]);
Packit ae235b
  g_free (ref.key_pem);
Packit ae235b
  g_free (ref.key8_pem);
Packit ae235b
Packit ae235b
  return rtv;
Packit ae235b
}