Blame gio/gdbusauthmechanismexternal.c

Packit ae235b
/* GDBus - GLib D-Bus Library
Packit ae235b
 *
Packit ae235b
 * Copyright (C) 2008-2010 Red Hat, Inc.
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: David Zeuthen <davidz@redhat.com>
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include "config.h"
Packit ae235b
Packit ae235b
#include <string.h>
Packit ae235b
Packit ae235b
#include "gdbusauthmechanismexternal.h"
Packit ae235b
#include "gcredentials.h"
Packit ae235b
#include "gdbuserror.h"
Packit ae235b
#include "gioenumtypes.h"
Packit ae235b
Packit ae235b
#include "glibintl.h"
Packit ae235b
Packit ae235b
struct _GDBusAuthMechanismExternalPrivate
Packit ae235b
{
Packit ae235b
  gboolean is_client;
Packit ae235b
  gboolean is_server;
Packit ae235b
  GDBusAuthMechanismState state;
Packit ae235b
};
Packit ae235b
Packit ae235b
static gint                     mechanism_get_priority              (void);
Packit ae235b
static const gchar             *mechanism_get_name                  (void);
Packit ae235b
Packit ae235b
static gboolean                 mechanism_is_supported              (GDBusAuthMechanism   *mechanism);
Packit ae235b
static gchar                   *mechanism_encode_data               (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     const gchar          *data,
Packit ae235b
                                                                     gsize                 data_len,
Packit ae235b
                                                                     gsize                *out_data_len);
Packit ae235b
static gchar                   *mechanism_decode_data               (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     const gchar          *data,
Packit ae235b
                                                                     gsize                 data_len,
Packit ae235b
                                                                     gsize                *out_data_len);
Packit ae235b
static GDBusAuthMechanismState  mechanism_server_get_state          (GDBusAuthMechanism   *mechanism);
Packit ae235b
static void                     mechanism_server_initiate           (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     const gchar          *initial_response,
Packit ae235b
                                                                     gsize                 initial_response_len);
Packit ae235b
static void                     mechanism_server_data_receive       (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     const gchar          *data,
Packit ae235b
                                                                     gsize                 data_len);
Packit ae235b
static gchar                   *mechanism_server_data_send          (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     gsize                *out_data_len);
Packit ae235b
static gchar                   *mechanism_server_get_reject_reason  (GDBusAuthMechanism   *mechanism);
Packit ae235b
static void                     mechanism_server_shutdown           (GDBusAuthMechanism   *mechanism);
Packit ae235b
static GDBusAuthMechanismState  mechanism_client_get_state          (GDBusAuthMechanism   *mechanism);
Packit ae235b
static gchar                   *mechanism_client_initiate           (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     gsize                *out_initial_response_len);
Packit ae235b
static void                     mechanism_client_data_receive       (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     const gchar          *data,
Packit ae235b
                                                                     gsize                 data_len);
Packit ae235b
static gchar                   *mechanism_client_data_send          (GDBusAuthMechanism   *mechanism,
Packit ae235b
                                                                     gsize                *out_data_len);
Packit ae235b
static void                     mechanism_client_shutdown           (GDBusAuthMechanism   *mechanism);
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
G_DEFINE_TYPE_WITH_PRIVATE (GDBusAuthMechanismExternal, _g_dbus_auth_mechanism_external, G_TYPE_DBUS_AUTH_MECHANISM)
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static void
Packit ae235b
_g_dbus_auth_mechanism_external_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  //GDBusAuthMechanismExternal *mechanism = G_DBUS_AUTH_MECHANISM_EXTERNAL (object);
Packit ae235b
Packit ae235b
  if (G_OBJECT_CLASS (_g_dbus_auth_mechanism_external_parent_class)->finalize != NULL)
Packit ae235b
    G_OBJECT_CLASS (_g_dbus_auth_mechanism_external_parent_class)->finalize (object);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
_g_dbus_auth_mechanism_external_class_init (GDBusAuthMechanismExternalClass *klass)
Packit ae235b
{
Packit ae235b
  GObjectClass *gobject_class;
Packit ae235b
  GDBusAuthMechanismClass *mechanism_class;
Packit ae235b
Packit ae235b
  gobject_class = G_OBJECT_CLASS (klass);
Packit ae235b
  gobject_class->finalize = _g_dbus_auth_mechanism_external_finalize;
Packit ae235b
Packit ae235b
  mechanism_class = G_DBUS_AUTH_MECHANISM_CLASS (klass);
Packit ae235b
  mechanism_class->get_name                  = mechanism_get_name;
Packit ae235b
  mechanism_class->get_priority              = mechanism_get_priority;
Packit ae235b
  mechanism_class->is_supported              = mechanism_is_supported;
Packit ae235b
  mechanism_class->encode_data               = mechanism_encode_data;
Packit ae235b
  mechanism_class->decode_data               = mechanism_decode_data;
Packit ae235b
  mechanism_class->server_get_state          = mechanism_server_get_state;
Packit ae235b
  mechanism_class->server_initiate           = mechanism_server_initiate;
Packit ae235b
  mechanism_class->server_data_receive       = mechanism_server_data_receive;
Packit ae235b
  mechanism_class->server_data_send          = mechanism_server_data_send;
Packit ae235b
  mechanism_class->server_get_reject_reason  = mechanism_server_get_reject_reason;
Packit ae235b
  mechanism_class->server_shutdown           = mechanism_server_shutdown;
Packit ae235b
  mechanism_class->client_get_state          = mechanism_client_get_state;
Packit ae235b
  mechanism_class->client_initiate           = mechanism_client_initiate;
Packit ae235b
  mechanism_class->client_data_receive       = mechanism_client_data_receive;
Packit ae235b
  mechanism_class->client_data_send          = mechanism_client_data_send;
Packit ae235b
  mechanism_class->client_shutdown           = mechanism_client_shutdown;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
_g_dbus_auth_mechanism_external_init (GDBusAuthMechanismExternal *mechanism)
Packit ae235b
{
Packit ae235b
  mechanism->priv = _g_dbus_auth_mechanism_external_get_instance_private (mechanism);
Packit ae235b
}
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
mechanism_is_supported (GDBusAuthMechanism *mechanism)
Packit ae235b
{
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), FALSE);
Packit ae235b
  /* This mechanism is only available if credentials has been exchanged */
Packit ae235b
  if (_g_dbus_auth_mechanism_get_credentials (mechanism) != NULL)
Packit ae235b
    return TRUE;
Packit ae235b
  else
Packit ae235b
    return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gint
Packit ae235b
mechanism_get_priority (void)
Packit ae235b
{
Packit ae235b
  /* We prefer EXTERNAL to most other mechanism (DBUS_COOKIE_SHA1 and ANONYMOUS) */
Packit ae235b
  return 100;
Packit ae235b
}
Packit ae235b
Packit ae235b
static const gchar *
Packit ae235b
mechanism_get_name (void)
Packit ae235b
{
Packit ae235b
  return "EXTERNAL";
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_encode_data (GDBusAuthMechanism   *mechanism,
Packit ae235b
                       const gchar          *data,
Packit ae235b
                       gsize                 data_len,
Packit ae235b
                       gsize                *out_data_len)
Packit ae235b
{
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_decode_data (GDBusAuthMechanism   *mechanism,
Packit ae235b
                       const gchar          *data,
Packit ae235b
                       gsize                 data_len,
Packit ae235b
                       gsize                *out_data_len)
Packit ae235b
{
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static GDBusAuthMechanismState
Packit ae235b
mechanism_server_get_state (GDBusAuthMechanism   *mechanism)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), G_DBUS_AUTH_MECHANISM_STATE_INVALID);
Packit ae235b
  g_return_val_if_fail (m->priv->is_server && !m->priv->is_client, G_DBUS_AUTH_MECHANISM_STATE_INVALID);
Packit ae235b
Packit ae235b
  return m->priv->state;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
data_matches_credentials (const gchar *data,
Packit ae235b
                          GCredentials *credentials)
Packit ae235b
{
Packit ae235b
  gboolean match;
Packit ae235b
Packit ae235b
  match = FALSE;
Packit ae235b
Packit ae235b
  if (credentials == NULL)
Packit ae235b
    goto out;
Packit ae235b
Packit ae235b
  if (data == NULL || strlen (data) == 0)
Packit ae235b
    goto out;
Packit ae235b
Packit ae235b
#if defined(G_OS_UNIX)
Packit ae235b
  {
Packit ae235b
    gint64 alleged_uid;
Packit ae235b
    gchar *endp;
Packit ae235b
Packit ae235b
    /* on UNIX, this is the uid as a string in base 10 */
Packit ae235b
    alleged_uid = g_ascii_strtoll (data, &endp, 10);
Packit ae235b
    if (*endp == '\0')
Packit ae235b
      {
Packit ae235b
        if (g_credentials_get_unix_user (credentials, NULL) == alleged_uid)
Packit ae235b
          {
Packit ae235b
            match = TRUE;
Packit ae235b
          }
Packit ae235b
      }
Packit ae235b
  }
Packit ae235b
#else
Packit ae235b
  /* TODO: Dont know how to compare credentials on this OS. Please implement. */
Packit ae235b
#endif
Packit ae235b
Packit ae235b
 out:
Packit ae235b
  return match;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
mechanism_server_initiate (GDBusAuthMechanism   *mechanism,
Packit ae235b
                           const gchar          *initial_response,
Packit ae235b
                           gsize                 initial_response_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism));
Packit ae235b
  g_return_if_fail (!m->priv->is_server && !m->priv->is_client);
Packit ae235b
Packit ae235b
  m->priv->is_server = TRUE;
Packit ae235b
Packit ae235b
  if (initial_response != NULL)
Packit ae235b
    {
Packit ae235b
      if (data_matches_credentials (initial_response, _g_dbus_auth_mechanism_get_credentials (mechanism)))
Packit ae235b
        {
Packit ae235b
          m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_ACCEPTED;
Packit ae235b
        }
Packit ae235b
      else
Packit ae235b
        {
Packit ae235b
          m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_REJECTED;
Packit ae235b
        }
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_WAITING_FOR_DATA;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
mechanism_server_data_receive (GDBusAuthMechanism   *mechanism,
Packit ae235b
                               const gchar          *data,
Packit ae235b
                               gsize                 data_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism));
Packit ae235b
  g_return_if_fail (m->priv->is_server && !m->priv->is_client);
Packit ae235b
  g_return_if_fail (m->priv->state == G_DBUS_AUTH_MECHANISM_STATE_WAITING_FOR_DATA);
Packit ae235b
Packit ae235b
  if (data_matches_credentials (data, _g_dbus_auth_mechanism_get_credentials (mechanism)))
Packit ae235b
    {
Packit ae235b
      m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_ACCEPTED;
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_REJECTED;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_server_data_send (GDBusAuthMechanism   *mechanism,
Packit ae235b
                            gsize                *out_data_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->is_server && !m->priv->is_client, NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->state == G_DBUS_AUTH_MECHANISM_STATE_HAVE_DATA_TO_SEND, NULL);
Packit ae235b
Packit ae235b
  /* can never end up here because we are never in the HAVE_DATA_TO_SEND state */
Packit ae235b
  g_assert_not_reached ();
Packit ae235b
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_server_get_reject_reason (GDBusAuthMechanism   *mechanism)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->is_server && !m->priv->is_client, NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->state == G_DBUS_AUTH_MECHANISM_STATE_REJECTED, NULL);
Packit ae235b
Packit ae235b
  /* can never end up here because we are never in the REJECTED state */
Packit ae235b
  g_assert_not_reached ();
Packit ae235b
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
mechanism_server_shutdown (GDBusAuthMechanism   *mechanism)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism));
Packit ae235b
  g_return_if_fail (m->priv->is_server && !m->priv->is_client);
Packit ae235b
Packit ae235b
  m->priv->is_server = FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static GDBusAuthMechanismState
Packit ae235b
mechanism_client_get_state (GDBusAuthMechanism   *mechanism)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), G_DBUS_AUTH_MECHANISM_STATE_INVALID);
Packit ae235b
  g_return_val_if_fail (m->priv->is_client && !m->priv->is_server, G_DBUS_AUTH_MECHANISM_STATE_INVALID);
Packit ae235b
Packit ae235b
  return m->priv->state;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_client_initiate (GDBusAuthMechanism   *mechanism,
Packit ae235b
                           gsize                *out_initial_response_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
  gchar *initial_response = NULL;
Packit ae235b
  GCredentials *credentials;
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), NULL);
Packit ae235b
  g_return_val_if_fail (!m->priv->is_server && !m->priv->is_client, NULL);
Packit ae235b
Packit ae235b
  m->priv->is_client = TRUE;
Packit ae235b
  m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_ACCEPTED;
Packit ae235b
Packit ae235b
  *out_initial_response_len = -1;
Packit ae235b
Packit ae235b
  credentials = _g_dbus_auth_mechanism_get_credentials (mechanism);
Packit ae235b
  g_assert (credentials != NULL);
Packit ae235b
Packit ae235b
  /* return the uid */
Packit ae235b
#if defined(G_OS_UNIX)
Packit ae235b
  initial_response = g_strdup_printf ("%" G_GINT64_FORMAT, (gint64) g_credentials_get_unix_user (credentials, NULL));
Packit ae235b
#elif defined(G_OS_WIN32)
Packit ae235b
#ifdef __GNUC__
Packit ae235b
#warning Dont know how to send credentials on this OS. The EXTERNAL D-Bus authentication mechanism will not work.
Packit ae235b
#endif
Packit ae235b
  m->priv->state = G_DBUS_AUTH_MECHANISM_STATE_REJECTED;
Packit ae235b
#endif
Packit ae235b
  return initial_response;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
mechanism_client_data_receive (GDBusAuthMechanism   *mechanism,
Packit ae235b
                               const gchar          *data,
Packit ae235b
                               gsize                 data_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism));
Packit ae235b
  g_return_if_fail (m->priv->is_client && !m->priv->is_server);
Packit ae235b
  g_return_if_fail (m->priv->state == G_DBUS_AUTH_MECHANISM_STATE_WAITING_FOR_DATA);
Packit ae235b
Packit ae235b
  /* can never end up here because we are never in the WAITING_FOR_DATA state */
Packit ae235b
  g_assert_not_reached ();
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
mechanism_client_data_send (GDBusAuthMechanism   *mechanism,
Packit ae235b
                            gsize                *out_data_len)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_val_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism), NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->is_client && !m->priv->is_server, NULL);
Packit ae235b
  g_return_val_if_fail (m->priv->state == G_DBUS_AUTH_MECHANISM_STATE_HAVE_DATA_TO_SEND, NULL);
Packit ae235b
Packit ae235b
  /* can never end up here because we are never in the HAVE_DATA_TO_SEND state */
Packit ae235b
  g_assert_not_reached ();
Packit ae235b
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
mechanism_client_shutdown (GDBusAuthMechanism   *mechanism)
Packit ae235b
{
Packit ae235b
  GDBusAuthMechanismExternal *m = G_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism);
Packit ae235b
Packit ae235b
  g_return_if_fail (G_IS_DBUS_AUTH_MECHANISM_EXTERNAL (mechanism));
Packit ae235b
  g_return_if_fail (m->priv->is_client && !m->priv->is_server);
Packit ae235b
Packit ae235b
  m->priv->is_client = FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */