Blame gio/tests/gtesttlsbackend.c

Packit ae235b
/* GIO - GLib Input, Output and Streaming Library
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
Packit ae235b
#include "gtesttlsbackend.h"
Packit ae235b
Packit ae235b
#include <glib.h>
Packit ae235b
Packit ae235b
static GType _g_test_tls_certificate_get_type (void);
Packit ae235b
static GType _g_test_tls_connection_get_type (void);
Packit ae235b
Packit ae235b
struct _GTestTlsBackend {
Packit ae235b
  GObject parent_instance;
Packit ae235b
};
Packit ae235b
Packit ae235b
static void g_test_tls_backend_iface_init (GTlsBackendInterface *iface);
Packit ae235b
Packit ae235b
#define g_test_tls_backend_get_type _g_test_tls_backend_get_type
Packit ae235b
G_DEFINE_TYPE_WITH_CODE (GTestTlsBackend, g_test_tls_backend, G_TYPE_OBJECT,
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_TLS_BACKEND,
Packit ae235b
						g_test_tls_backend_iface_init)
Packit ae235b
                         g_io_extension_point_set_required_type (
Packit ae235b
                           g_io_extension_point_register (G_TLS_BACKEND_EXTENSION_POINT_NAME),
Packit ae235b
                           G_TYPE_TLS_BACKEND);
Packit ae235b
			 g_io_extension_point_implement (G_TLS_BACKEND_EXTENSION_POINT_NAME,
Packit ae235b
							 g_define_type_id,
Packit ae235b
							 "test",
Packit ae235b
							 999);)
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_backend_init (GTestTlsBackend *backend)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_backend_class_init (GTestTlsBackendClass *backend_class)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_backend_iface_init (GTlsBackendInterface *iface)
Packit ae235b
{
Packit ae235b
  iface->get_certificate_type = _g_test_tls_certificate_get_type;
Packit ae235b
  iface->get_client_connection_type = _g_test_tls_connection_get_type;
Packit ae235b
  iface->get_server_connection_type = _g_test_tls_connection_get_type;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* Test certificate type */
Packit ae235b
Packit ae235b
typedef struct _GTestTlsCertificate      GTestTlsCertificate;
Packit ae235b
typedef struct _GTestTlsCertificateClass GTestTlsCertificateClass;
Packit ae235b
Packit ae235b
struct _GTestTlsCertificate {
Packit ae235b
  GTlsCertificate parent_instance;
Packit ae235b
  gchar *key_pem;
Packit ae235b
  gchar *cert_pem;
Packit ae235b
  GTlsCertificate *issuer;
Packit ae235b
};
Packit ae235b
Packit ae235b
struct _GTestTlsCertificateClass {
Packit ae235b
  GTlsCertificateClass parent_class;
Packit ae235b
};
Packit ae235b
Packit ae235b
enum
Packit ae235b
{
Packit ae235b
  PROP_CERTIFICATE_0,
Packit ae235b
Packit ae235b
  PROP_CERT_CERTIFICATE,
Packit ae235b
  PROP_CERT_CERTIFICATE_PEM,
Packit ae235b
  PROP_CERT_PRIVATE_KEY,
Packit ae235b
  PROP_CERT_PRIVATE_KEY_PEM,
Packit ae235b
  PROP_CERT_ISSUER
Packit ae235b
};
Packit ae235b
Packit ae235b
static void g_test_tls_certificate_initable_iface_init (GInitableIface *iface);
Packit ae235b
Packit ae235b
#define g_test_tls_certificate_get_type _g_test_tls_certificate_get_type
Packit ae235b
G_DEFINE_TYPE_WITH_CODE (GTestTlsCertificate, g_test_tls_certificate, G_TYPE_TLS_CERTIFICATE,
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE,
Packit ae235b
						g_test_tls_certificate_initable_iface_init))
Packit ae235b
Packit ae235b
static GTlsCertificateFlags
Packit ae235b
g_test_tls_certificate_verify (GTlsCertificate     *cert,
Packit ae235b
                               GSocketConnectable  *identity,
Packit ae235b
                               GTlsCertificate     *trusted_ca)
Packit ae235b
{
Packit ae235b
  /* For now, all of the tests expect the certificate to verify */
Packit ae235b
  return 0;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_get_property (GObject    *object,
Packit ae235b
				      guint       prop_id,
Packit ae235b
				      GValue     *value,
Packit ae235b
				      GParamSpec *pspec)
Packit ae235b
{
Packit ae235b
  GTestTlsCertificate *cert = (GTestTlsCertificate *) object;
Packit ae235b
Packit ae235b
  switch (prop_id)
Packit ae235b
    {
Packit ae235b
    case PROP_CERT_CERTIFICATE_PEM:
Packit ae235b
      g_value_set_string (value, cert->cert_pem);
Packit ae235b
      break;
Packit ae235b
    case PROP_CERT_PRIVATE_KEY_PEM:
Packit ae235b
      g_value_set_string (value, cert->key_pem);
Packit ae235b
      break;
Packit ae235b
    case PROP_CERT_ISSUER:
Packit ae235b
      g_value_set_object (value, cert->issuer);
Packit ae235b
      break;
Packit ae235b
    default:
Packit ae235b
      g_assert_not_reached ();
Packit ae235b
      break;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_set_property (GObject      *object,
Packit ae235b
				      guint         prop_id,
Packit ae235b
				      const GValue *value,
Packit ae235b
				      GParamSpec   *pspec)
Packit ae235b
{
Packit ae235b
  GTestTlsCertificate *cert = (GTestTlsCertificate *) object;
Packit ae235b
Packit ae235b
  switch (prop_id)
Packit ae235b
    {
Packit ae235b
    case PROP_CERT_CERTIFICATE_PEM:
Packit ae235b
      cert->cert_pem = g_value_dup_string (value);
Packit ae235b
      break;
Packit ae235b
    case PROP_CERT_PRIVATE_KEY_PEM:
Packit ae235b
      cert->key_pem = g_value_dup_string (value);
Packit ae235b
      break;
Packit ae235b
    case PROP_CERT_ISSUER:
Packit ae235b
      cert->issuer = g_value_dup_object (value);
Packit ae235b
      break;
Packit ae235b
    case PROP_CERT_CERTIFICATE:
Packit ae235b
    case PROP_CERT_PRIVATE_KEY:
Packit ae235b
      /* ignore */
Packit ae235b
      break;
Packit ae235b
    default:
Packit ae235b
      g_assert_not_reached ();
Packit ae235b
      break;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  GTestTlsCertificate *cert = (GTestTlsCertificate *) object;
Packit ae235b
Packit ae235b
  g_free (cert->cert_pem);
Packit ae235b
  g_free (cert->key_pem);
Packit ae235b
  g_clear_object (&cert->issuer);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_class_init (GTestTlsCertificateClass *test_class)
Packit ae235b
{
Packit ae235b
  GObjectClass *gobject_class = G_OBJECT_CLASS (test_class);
Packit ae235b
  GTlsCertificateClass *certificate_class = G_TLS_CERTIFICATE_CLASS (test_class);
Packit ae235b
Packit ae235b
  gobject_class->get_property = g_test_tls_certificate_get_property;
Packit ae235b
  gobject_class->set_property = g_test_tls_certificate_set_property;
Packit ae235b
  gobject_class->finalize = g_test_tls_certificate_finalize;
Packit ae235b
Packit ae235b
  certificate_class->verify = g_test_tls_certificate_verify;
Packit ae235b
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CERT_CERTIFICATE, "certificate");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CERT_CERTIFICATE_PEM, "certificate-pem");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CERT_PRIVATE_KEY, "private-key");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CERT_PRIVATE_KEY_PEM, "private-key-pem");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CERT_ISSUER, "issuer");
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_init (GTestTlsCertificate *certificate)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
g_test_tls_certificate_initable_init (GInitable       *initable,
Packit ae235b
				       GCancellable    *cancellable,
Packit ae235b
				       GError         **error)
Packit ae235b
{
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_certificate_initable_iface_init (GInitableIface  *iface)
Packit ae235b
{
Packit ae235b
  iface->init = g_test_tls_certificate_initable_init;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* Dummy connection type; since GTlsClientConnection and
Packit ae235b
 * GTlsServerConnection are just interfaces, we can implement them
Packit ae235b
 * both on a single object.
Packit ae235b
 */
Packit ae235b
Packit ae235b
typedef struct _GTestTlsConnection      GTestTlsConnection;
Packit ae235b
typedef struct _GTestTlsConnectionClass GTestTlsConnectionClass;
Packit ae235b
Packit ae235b
struct _GTestTlsConnection {
Packit ae235b
  GTlsConnection parent_instance;
Packit ae235b
};
Packit ae235b
Packit ae235b
struct _GTestTlsConnectionClass {
Packit ae235b
  GTlsConnectionClass parent_class;
Packit ae235b
};
Packit ae235b
Packit ae235b
enum
Packit ae235b
{
Packit ae235b
  PROP_CONNECTION_0,
Packit ae235b
Packit ae235b
  PROP_CONN_BASE_IO_STREAM,
Packit ae235b
  PROP_CONN_USE_SYSTEM_CERTDB,
Packit ae235b
  PROP_CONN_REQUIRE_CLOSE_NOTIFY,
Packit ae235b
  PROP_CONN_REHANDSHAKE_MODE,
Packit ae235b
  PROP_CONN_CERTIFICATE,
Packit ae235b
  PROP_CONN_PEER_CERTIFICATE,
Packit ae235b
  PROP_CONN_PEER_CERTIFICATE_ERRORS,
Packit ae235b
  PROP_CONN_VALIDATION_FLAGS,
Packit ae235b
  PROP_CONN_SERVER_IDENTITY,
Packit ae235b
  PROP_CONN_USE_SSL3,
Packit ae235b
  PROP_CONN_ACCEPTED_CAS,
Packit ae235b
  PROP_CONN_AUTHENTICATION_MODE
Packit ae235b
};
Packit ae235b
Packit ae235b
static void g_test_tls_connection_initable_iface_init (GInitableIface *iface);
Packit ae235b
Packit ae235b
#define g_test_tls_connection_get_type _g_test_tls_connection_get_type
Packit ae235b
G_DEFINE_TYPE_WITH_CODE (GTestTlsConnection, g_test_tls_connection, G_TYPE_TLS_CONNECTION,
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_TLS_CLIENT_CONNECTION, NULL)
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_TLS_SERVER_CONNECTION, NULL)
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE,
Packit ae235b
						g_test_tls_connection_initable_iface_init))
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_connection_get_property (GObject    *object,
Packit ae235b
				     guint       prop_id,
Packit ae235b
				     GValue     *value,
Packit ae235b
				     GParamSpec *pspec)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_connection_set_property (GObject      *object,
Packit ae235b
				     guint         prop_id,
Packit ae235b
				     const GValue *value,
Packit ae235b
				     GParamSpec   *pspec)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
g_test_tls_connection_close (GIOStream     *stream,
Packit ae235b
			      GCancellable  *cancellable,
Packit ae235b
			      GError       **error)
Packit ae235b
{
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_connection_class_init (GTestTlsConnectionClass *connection_class)
Packit ae235b
{
Packit ae235b
  GObjectClass *gobject_class = G_OBJECT_CLASS (connection_class);
Packit ae235b
  GIOStreamClass *io_stream_class = G_IO_STREAM_CLASS (connection_class);
Packit ae235b
Packit ae235b
  gobject_class->get_property = g_test_tls_connection_get_property;
Packit ae235b
  gobject_class->set_property = g_test_tls_connection_set_property;
Packit ae235b
Packit ae235b
  /* Need to override this because when initable_init fails it will
Packit ae235b
   * dispose the connection, which will close it, which would
Packit ae235b
   * otherwise try to close its input/output streams, which don't
Packit ae235b
   * exist.
Packit ae235b
   */
Packit ae235b
  io_stream_class->close_fn = g_test_tls_connection_close;
Packit ae235b
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_BASE_IO_STREAM, "base-io-stream");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_USE_SYSTEM_CERTDB, "use-system-certdb");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_REQUIRE_CLOSE_NOTIFY, "require-close-notify");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_REHANDSHAKE_MODE, "rehandshake-mode");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_CERTIFICATE, "certificate");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_PEER_CERTIFICATE, "peer-certificate");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_PEER_CERTIFICATE_ERRORS, "peer-certificate-errors");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_VALIDATION_FLAGS, "validation-flags");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_SERVER_IDENTITY, "server-identity");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_USE_SSL3, "use-ssl3");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_ACCEPTED_CAS, "accepted-cas");
Packit ae235b
  g_object_class_override_property (gobject_class, PROP_CONN_AUTHENTICATION_MODE, "authentication-mode");
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_connection_init (GTestTlsConnection *connection)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
g_test_tls_connection_initable_init (GInitable       *initable,
Packit ae235b
				      GCancellable    *cancellable,
Packit ae235b
				      GError         **error)
Packit ae235b
{
Packit ae235b
  g_set_error_literal (error, G_TLS_ERROR, G_TLS_ERROR_UNAVAILABLE,
Packit ae235b
		       "TLS Connection support is not available");
Packit ae235b
  return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_test_tls_connection_initable_iface_init (GInitableIface  *iface)
Packit ae235b
{
Packit ae235b
  iface->init = g_test_tls_connection_initable_init;
Packit ae235b
}
Packit ae235b
Packit ae235b
const gchar *
Packit ae235b
g_test_tls_connection_get_private_key_pem (GTlsCertificate *cert)
Packit ae235b
{
Packit ae235b
  return ((GTestTlsCertificate *)cert)->key_pem;
Packit ae235b
}