Blame gio/gnetworkservice.c

Packit ae235b
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
Packit ae235b
Packit ae235b
/* GIO - GLib Input, Output and Streaming Library
Packit ae235b
 *
Packit ae235b
 * Copyright (C) 2008 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
Packit ae235b
#include "config.h"
Packit ae235b
#include <glib.h>
Packit ae235b
#include "glibintl.h"
Packit ae235b
Packit ae235b
#include "gnetworkservice.h"
Packit ae235b
Packit ae235b
#include "gcancellable.h"
Packit ae235b
#include "ginetaddress.h"
Packit ae235b
#include "ginetsocketaddress.h"
Packit ae235b
#include "gioerror.h"
Packit ae235b
#include "gnetworkaddress.h"
Packit ae235b
#include "gnetworkingprivate.h"
Packit ae235b
#include "gresolver.h"
Packit ae235b
#include "gtask.h"
Packit ae235b
#include "gsocketaddressenumerator.h"
Packit ae235b
#include "gsocketconnectable.h"
Packit ae235b
#include "gsrvtarget.h"
Packit ae235b
Packit ae235b
#include <stdlib.h>
Packit ae235b
#include <string.h>
Packit ae235b
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * SECTION:gnetworkservice
Packit ae235b
 * @short_description: A GSocketConnectable for resolving SRV records
Packit ae235b
 * @include: gio/gio.h
Packit ae235b
 *
Packit ae235b
 * Like #GNetworkAddress does with hostnames, #GNetworkService
Packit ae235b
 * provides an easy way to resolve a SRV record, and then attempt to
Packit ae235b
 * connect to one of the hosts that implements that service, handling
Packit ae235b
 * service priority/weighting, multiple IP addresses, and multiple
Packit ae235b
 * address families.
Packit ae235b
 *
Packit ae235b
 * See #GSrvTarget for more information about SRV records, and see
Packit ae235b
 * #GSocketConnectable for and example of using the connectable
Packit ae235b
 * interface.
Packit ae235b
 */
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * GNetworkService:
Packit ae235b
 *
Packit ae235b
 * A #GSocketConnectable for resolving a SRV record and connecting to
Packit ae235b
 * that service.
Packit ae235b
 */
Packit ae235b
Packit ae235b
struct _GNetworkServicePrivate
Packit ae235b
{
Packit ae235b
  gchar *service, *protocol, *domain, *scheme;
Packit ae235b
  GList *targets;
Packit ae235b
};
Packit ae235b
Packit ae235b
enum {
Packit ae235b
  PROP_0,
Packit ae235b
  PROP_SERVICE,
Packit ae235b
  PROP_PROTOCOL,
Packit ae235b
  PROP_DOMAIN,
Packit ae235b
  PROP_SCHEME
Packit ae235b
};
Packit ae235b
Packit ae235b
static void g_network_service_set_property (GObject      *object,
Packit ae235b
                                            guint         prop_id,
Packit ae235b
                                            const GValue *value,
Packit ae235b
                                            GParamSpec   *pspec);
Packit ae235b
static void g_network_service_get_property (GObject      *object,
Packit ae235b
                                            guint         prop_id,
Packit ae235b
                                            GValue       *value,
Packit ae235b
                                            GParamSpec   *pspec);
Packit ae235b
Packit ae235b
static void                      g_network_service_connectable_iface_init       (GSocketConnectableIface *iface);
Packit ae235b
static GSocketAddressEnumerator *g_network_service_connectable_enumerate        (GSocketConnectable      *connectable);
Packit ae235b
static GSocketAddressEnumerator *g_network_service_connectable_proxy_enumerate  (GSocketConnectable      *connectable);
Packit ae235b
static gchar                    *g_network_service_connectable_to_string        (GSocketConnectable      *connectable);
Packit ae235b
Packit ae235b
G_DEFINE_TYPE_WITH_CODE (GNetworkService, g_network_service, G_TYPE_OBJECT,
Packit ae235b
                         G_ADD_PRIVATE (GNetworkService)
Packit ae235b
                         G_IMPLEMENT_INTERFACE (G_TYPE_SOCKET_CONNECTABLE,
Packit ae235b
                                                g_network_service_connectable_iface_init))
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  GNetworkService *srv = G_NETWORK_SERVICE (object);
Packit ae235b
Packit ae235b
  g_free (srv->priv->service);
Packit ae235b
  g_free (srv->priv->protocol);
Packit ae235b
  g_free (srv->priv->domain);
Packit ae235b
  g_free (srv->priv->scheme);
Packit ae235b
Packit ae235b
  if (srv->priv->targets)
Packit ae235b
    g_resolver_free_targets (srv->priv->targets);
Packit ae235b
Packit ae235b
  G_OBJECT_CLASS (g_network_service_parent_class)->finalize (object);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_class_init (GNetworkServiceClass *klass)
Packit ae235b
{
Packit ae235b
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
Packit ae235b
Packit ae235b
  gobject_class->set_property = g_network_service_set_property;
Packit ae235b
  gobject_class->get_property = g_network_service_get_property;
Packit ae235b
  gobject_class->finalize = g_network_service_finalize;
Packit ae235b
Packit ae235b
  g_object_class_install_property (gobject_class, PROP_SERVICE,
Packit ae235b
                                   g_param_spec_string ("service",
Packit ae235b
                                                        P_("Service"),
Packit ae235b
                                                        P_("Service name, eg \"ldap\""),
Packit ae235b
                                                        NULL,
Packit ae235b
                                                        G_PARAM_READWRITE |
Packit ae235b
                                                        G_PARAM_CONSTRUCT_ONLY |
Packit ae235b
                                                        G_PARAM_STATIC_STRINGS));
Packit ae235b
  g_object_class_install_property (gobject_class, PROP_PROTOCOL,
Packit ae235b
                                   g_param_spec_string ("protocol",
Packit ae235b
                                                        P_("Protocol"),
Packit ae235b
                                                        P_("Network protocol, eg \"tcp\""),
Packit ae235b
                                                        NULL,
Packit ae235b
                                                        G_PARAM_READWRITE |
Packit ae235b
                                                        G_PARAM_CONSTRUCT_ONLY |
Packit ae235b
                                                        G_PARAM_STATIC_STRINGS));
Packit ae235b
  g_object_class_install_property (gobject_class, PROP_DOMAIN,
Packit ae235b
                                   g_param_spec_string ("domain",
Packit ae235b
                                                        P_("Domain"),
Packit ae235b
                                                        P_("Network domain, eg, \"example.com\""),
Packit ae235b
                                                        NULL,
Packit ae235b
                                                        G_PARAM_READWRITE |
Packit ae235b
                                                        G_PARAM_CONSTRUCT_ONLY |
Packit ae235b
                                                        G_PARAM_STATIC_STRINGS));
Packit ae235b
  g_object_class_install_property (gobject_class, PROP_DOMAIN,
Packit ae235b
                                   g_param_spec_string ("scheme",
Packit ae235b
                                                        P_("Scheme"),
Packit ae235b
                                                        P_("Network scheme (default is to use service)"),
Packit ae235b
                                                        NULL,
Packit ae235b
                                                        G_PARAM_READWRITE |
Packit ae235b
                                                        G_PARAM_STATIC_STRINGS));
Packit ae235b
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_connectable_iface_init (GSocketConnectableIface *connectable_iface)
Packit ae235b
{
Packit ae235b
  connectable_iface->enumerate = g_network_service_connectable_enumerate;
Packit ae235b
  connectable_iface->proxy_enumerate = g_network_service_connectable_proxy_enumerate;
Packit ae235b
  connectable_iface->to_string = g_network_service_connectable_to_string;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_init (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  srv->priv = g_network_service_get_instance_private (srv);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_set_property (GObject      *object,
Packit ae235b
                                guint         prop_id,
Packit ae235b
                                const GValue *value,
Packit ae235b
                                GParamSpec   *pspec)
Packit ae235b
{
Packit ae235b
  GNetworkService *srv = G_NETWORK_SERVICE (object);
Packit ae235b
Packit ae235b
  switch (prop_id)
Packit ae235b
    {
Packit ae235b
    case PROP_SERVICE:
Packit ae235b
      srv->priv->service = g_value_dup_string (value);
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_PROTOCOL:
Packit ae235b
      srv->priv->protocol = g_value_dup_string (value);
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_DOMAIN:
Packit ae235b
      srv->priv->domain = g_value_dup_string (value);
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_SCHEME:
Packit ae235b
      g_network_service_set_scheme (srv, g_value_get_string (value));
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    default:
Packit ae235b
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit ae235b
      break;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_get_property (GObject    *object,
Packit ae235b
                                guint       prop_id,
Packit ae235b
                                GValue     *value,
Packit ae235b
                                GParamSpec *pspec)
Packit ae235b
{
Packit ae235b
  GNetworkService *srv = G_NETWORK_SERVICE (object);
Packit ae235b
Packit ae235b
  switch (prop_id)
Packit ae235b
    {
Packit ae235b
    case PROP_SERVICE:
Packit ae235b
      g_value_set_string (value, g_network_service_get_service (srv));
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_PROTOCOL:
Packit ae235b
      g_value_set_string (value, g_network_service_get_protocol (srv));
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_DOMAIN:
Packit ae235b
      g_value_set_string (value, g_network_service_get_domain (srv));
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    case PROP_SCHEME:
Packit ae235b
      g_value_set_string (value, g_network_service_get_scheme (srv));
Packit ae235b
      break;
Packit ae235b
Packit ae235b
    default:
Packit ae235b
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit ae235b
      break;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_new:
Packit ae235b
 * @service: the service type to look up (eg, "ldap")
Packit ae235b
 * @protocol: the networking protocol to use for @service (eg, "tcp")
Packit ae235b
 * @domain: the DNS domain to look up the service in
Packit ae235b
 *
Packit ae235b
 * Creates a new #GNetworkService representing the given @service,
Packit ae235b
 * @protocol, and @domain. This will initially be unresolved; use the
Packit ae235b
 * #GSocketConnectable interface to resolve it.
Packit ae235b
 *
Packit ae235b
 * Returns: (transfer full) (type GNetworkService): a new #GNetworkService
Packit ae235b
 *
Packit ae235b
 * Since: 2.22
Packit ae235b
 */
Packit ae235b
GSocketConnectable *
Packit ae235b
g_network_service_new (const gchar *service,
Packit ae235b
                       const gchar *protocol,
Packit ae235b
                       const gchar *domain)
Packit ae235b
{
Packit ae235b
  return g_object_new (G_TYPE_NETWORK_SERVICE,
Packit ae235b
                       "service", service,
Packit ae235b
                       "protocol", protocol,
Packit ae235b
                       "domain", domain,
Packit ae235b
                       NULL);
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_get_service:
Packit ae235b
 * @srv: a #GNetworkService
Packit ae235b
 *
Packit ae235b
 * Gets @srv's service name (eg, "ldap").
Packit ae235b
 *
Packit ae235b
 * Returns: @srv's service name
Packit ae235b
 *
Packit ae235b
 * Since: 2.22
Packit ae235b
 */
Packit ae235b
const gchar *
Packit ae235b
g_network_service_get_service (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
Packit ae235b
Packit ae235b
  return srv->priv->service;
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_get_protocol:
Packit ae235b
 * @srv: a #GNetworkService
Packit ae235b
 *
Packit ae235b
 * Gets @srv's protocol name (eg, "tcp").
Packit ae235b
 *
Packit ae235b
 * Returns: @srv's protocol name
Packit ae235b
 *
Packit ae235b
 * Since: 2.22
Packit ae235b
 */
Packit ae235b
const gchar *
Packit ae235b
g_network_service_get_protocol (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
Packit ae235b
Packit ae235b
  return srv->priv->protocol;
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_get_domain:
Packit ae235b
 * @srv: a #GNetworkService
Packit ae235b
 *
Packit ae235b
 * Gets the domain that @srv serves. This might be either UTF-8 or
Packit ae235b
 * ASCII-encoded, depending on what @srv was created with.
Packit ae235b
 *
Packit ae235b
 * Returns: @srv's domain name
Packit ae235b
 *
Packit ae235b
 * Since: 2.22
Packit ae235b
 */
Packit ae235b
const gchar *
Packit ae235b
g_network_service_get_domain (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
Packit ae235b
Packit ae235b
  return srv->priv->domain;
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_get_scheme:
Packit ae235b
 * @srv: a #GNetworkService
Packit ae235b
 *
Packit ae235b
 * Get's the URI scheme used to resolve proxies. By default, the service name
Packit ae235b
 * is used as scheme.
Packit ae235b
 *
Packit ae235b
 * Returns: @srv's scheme name
Packit ae235b
 *
Packit ae235b
 * Since: 2.26
Packit ae235b
 */
Packit ae235b
const gchar *
Packit ae235b
g_network_service_get_scheme (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
Packit ae235b
Packit ae235b
  if (srv->priv->scheme)
Packit ae235b
    return srv->priv->scheme;
Packit ae235b
  else
Packit ae235b
    return srv->priv->service;
Packit ae235b
}
Packit ae235b
Packit ae235b
/**
Packit ae235b
 * g_network_service_set_scheme:
Packit ae235b
 * @srv: a #GNetworkService
Packit ae235b
 * @scheme: a URI scheme
Packit ae235b
 *
Packit ae235b
 * Set's the URI scheme used to resolve proxies. By default, the service name
Packit ae235b
 * is used as scheme.
Packit ae235b
 *
Packit ae235b
 * Since: 2.26
Packit ae235b
 */
Packit ae235b
void
Packit ae235b
g_network_service_set_scheme (GNetworkService *srv,
Packit ae235b
                              const gchar     *scheme)
Packit ae235b
{
Packit ae235b
  g_return_if_fail (G_IS_NETWORK_SERVICE (srv));
Packit ae235b
Packit ae235b
  g_free (srv->priv->scheme);
Packit ae235b
  srv->priv->scheme = g_strdup (scheme);
Packit ae235b
Packit ae235b
  g_object_notify (G_OBJECT (srv), "scheme");
Packit ae235b
}
Packit ae235b
Packit ae235b
static GList *
Packit ae235b
g_network_service_fallback_targets (GNetworkService *srv)
Packit ae235b
{
Packit ae235b
  GSrvTarget *target;
Packit ae235b
  struct servent *entry;
Packit ae235b
  guint16 port;
Packit ae235b
Packit ae235b
  entry = getservbyname (srv->priv->service, "tcp");
Packit ae235b
  port = entry ? g_ntohs (entry->s_port) : 0;
Packit ae235b
#ifdef HAVE_ENDSERVENT
Packit ae235b
  endservent ();
Packit ae235b
#endif
Packit ae235b
Packit ae235b
  if (entry == NULL)
Packit ae235b
      return NULL;
Packit ae235b
Packit ae235b
  target = g_srv_target_new (srv->priv->domain, port, 0, 0);
Packit ae235b
  return g_list_append (NULL, target);
Packit ae235b
}
Packit ae235b
Packit ae235b
#define G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR (_g_network_service_address_enumerator_get_type ())
Packit ae235b
#define G_NETWORK_SERVICE_ADDRESS_ENUMERATOR(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR, GNetworkServiceAddressEnumerator))
Packit ae235b
Packit ae235b
typedef struct {
Packit ae235b
  GSocketAddressEnumerator parent_instance;
Packit ae235b
Packit ae235b
  GResolver *resolver;
Packit ae235b
  GNetworkService *srv;
Packit ae235b
  GSocketAddressEnumerator *addr_enum;
Packit ae235b
  GList *t;
Packit ae235b
  gboolean use_proxy;
Packit ae235b
Packit ae235b
  GError *error;
Packit ae235b
Packit ae235b
} GNetworkServiceAddressEnumerator;
Packit ae235b
Packit ae235b
typedef struct {
Packit ae235b
  GSocketAddressEnumeratorClass parent_class;
Packit ae235b
Packit ae235b
} GNetworkServiceAddressEnumeratorClass;
Packit ae235b
Packit ae235b
static GType _g_network_service_address_enumerator_get_type (void);
Packit ae235b
G_DEFINE_TYPE (GNetworkServiceAddressEnumerator, _g_network_service_address_enumerator, G_TYPE_SOCKET_ADDRESS_ENUMERATOR)
Packit ae235b
Packit ae235b
static GSocketAddress *
Packit ae235b
g_network_service_address_enumerator_next (GSocketAddressEnumerator  *enumerator,
Packit ae235b
                                           GCancellable              *cancellable,
Packit ae235b
                                           GError                   **error)
Packit ae235b
{
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum =
Packit ae235b
    G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (enumerator);
Packit ae235b
  GSocketAddress *ret = NULL;
Packit ae235b
Packit ae235b
  /* If we haven't yet resolved srv, do that */
Packit ae235b
  if (!srv_enum->srv->priv->targets)
Packit ae235b
    {
Packit ae235b
      GList *targets;
Packit ae235b
      GError *my_error = NULL;
Packit ae235b
Packit ae235b
      targets = g_resolver_lookup_service (srv_enum->resolver,
Packit ae235b
                                           srv_enum->srv->priv->service,
Packit ae235b
                                           srv_enum->srv->priv->protocol,
Packit ae235b
                                           srv_enum->srv->priv->domain,
Packit ae235b
                                           cancellable, &my_error);
Packit ae235b
      if (!targets && g_error_matches (my_error, G_RESOLVER_ERROR,
Packit ae235b
                                       G_RESOLVER_ERROR_NOT_FOUND))
Packit ae235b
        {
Packit ae235b
          targets = g_network_service_fallback_targets (srv_enum->srv);
Packit ae235b
          if (targets)
Packit ae235b
            g_clear_error (&my_error);
Packit ae235b
        }
Packit ae235b
Packit ae235b
      if (my_error)
Packit ae235b
        {
Packit ae235b
          g_propagate_error (error, my_error);
Packit ae235b
          return NULL;
Packit ae235b
        }
Packit ae235b
Packit ae235b
      srv_enum->srv->priv->targets = targets;
Packit ae235b
      srv_enum->t = srv_enum->srv->priv->targets;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  /* Delegate to GNetworkAddress */
Packit ae235b
  do
Packit ae235b
    {
Packit ae235b
      if (srv_enum->addr_enum == NULL && srv_enum->t)
Packit ae235b
        {
Packit ae235b
          GError *error = NULL;
Packit ae235b
          gchar *uri;
Packit ae235b
          gchar *hostname;
Packit ae235b
          GSocketConnectable *addr;
Packit ae235b
          GSrvTarget *target = srv_enum->t->data;
Packit ae235b
Packit ae235b
          srv_enum->t = g_list_next (srv_enum->t);
Packit ae235b
Packit ae235b
          hostname = g_hostname_to_ascii (g_srv_target_get_hostname (target));
Packit ae235b
Packit ae235b
          if (hostname == NULL)
Packit ae235b
            {
Packit ae235b
              if (srv_enum->error == NULL)
Packit ae235b
                srv_enum->error =
Packit ae235b
                  g_error_new (G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
Packit ae235b
                               "Received invalid hostname '%s' from GSrvTarget",
Packit ae235b
                               g_srv_target_get_hostname (target));
Packit ae235b
              continue;
Packit ae235b
            }
Packit ae235b
Packit ae235b
          uri = _g_uri_from_authority (g_network_service_get_scheme (srv_enum->srv),
Packit ae235b
                                       hostname,
Packit ae235b
                                       g_srv_target_get_port (target),
Packit ae235b
                                       NULL);
Packit ae235b
          g_free (hostname);
Packit ae235b
Packit ae235b
          addr = g_network_address_parse_uri (uri,
Packit ae235b
                                              g_srv_target_get_port (target),
Packit ae235b
                                              &error);
Packit ae235b
          g_free (uri);
Packit ae235b
Packit ae235b
          if (addr == NULL)
Packit ae235b
            {
Packit ae235b
              if (srv_enum->error == NULL)
Packit ae235b
                srv_enum->error = error;
Packit ae235b
              else
Packit ae235b
                g_error_free (error);
Packit ae235b
              continue;
Packit ae235b
            }
Packit ae235b
Packit ae235b
          if (srv_enum->use_proxy)
Packit ae235b
            srv_enum->addr_enum = g_socket_connectable_proxy_enumerate (addr);
Packit ae235b
          else
Packit ae235b
            srv_enum->addr_enum = g_socket_connectable_enumerate (addr);
Packit ae235b
          g_object_unref (addr);
Packit ae235b
        }
Packit ae235b
Packit ae235b
      if (srv_enum->addr_enum)
Packit ae235b
        {
Packit ae235b
          GError *error = NULL;
Packit ae235b
Packit ae235b
          ret = g_socket_address_enumerator_next (srv_enum->addr_enum,
Packit ae235b
                                                  cancellable,
Packit ae235b
                                                  &error);
Packit ae235b
Packit ae235b
          if (error)
Packit ae235b
            {
Packit ae235b
              if (srv_enum->error == NULL)
Packit ae235b
                srv_enum->error = error;
Packit ae235b
              else
Packit ae235b
                g_error_free (error);
Packit ae235b
            }
Packit ae235b
Packit ae235b
          if (!ret)
Packit ae235b
            {
Packit ae235b
              g_object_unref (srv_enum->addr_enum);
Packit ae235b
              srv_enum->addr_enum = NULL;
Packit ae235b
            }
Packit ae235b
        }
Packit ae235b
    }
Packit ae235b
  while (srv_enum->addr_enum == NULL && srv_enum->t);
Packit ae235b
Packit ae235b
  if (ret == NULL && srv_enum->error)
Packit ae235b
    {
Packit ae235b
      g_propagate_error (error, srv_enum->error);
Packit ae235b
      srv_enum->error = NULL;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return ret;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void next_async_resolved_targets   (GObject      *source_object,
Packit ae235b
                                           GAsyncResult *result,
Packit ae235b
                                           gpointer      user_data);
Packit ae235b
static void next_async_have_targets       (GTask        *srv_enum);
Packit ae235b
static void next_async_have_address       (GObject      *source_object,
Packit ae235b
                                           GAsyncResult *result,
Packit ae235b
                                           gpointer      user_data);
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_address_enumerator_next_async (GSocketAddressEnumerator  *enumerator,
Packit ae235b
                                                 GCancellable              *cancellable,
Packit ae235b
                                                 GAsyncReadyCallback        callback,
Packit ae235b
                                                 gpointer                   user_data)
Packit ae235b
{
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum =
Packit ae235b
    G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (enumerator);
Packit ae235b
  GTask *task;
Packit ae235b
Packit ae235b
  task = g_task_new (enumerator, cancellable, callback, user_data);
Packit ae235b
  g_task_set_source_tag (task, g_network_service_address_enumerator_next_async);
Packit ae235b
Packit ae235b
  /* If we haven't yet resolved srv, do that */
Packit ae235b
  if (!srv_enum->srv->priv->targets)
Packit ae235b
    {
Packit ae235b
      g_resolver_lookup_service_async (srv_enum->resolver,
Packit ae235b
                                       srv_enum->srv->priv->service,
Packit ae235b
                                       srv_enum->srv->priv->protocol,
Packit ae235b
                                       srv_enum->srv->priv->domain,
Packit ae235b
                                       cancellable,
Packit ae235b
                                       next_async_resolved_targets,
Packit ae235b
                                       task);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    next_async_have_targets (task);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
next_async_resolved_targets (GObject      *source_object,
Packit ae235b
                             GAsyncResult *result,
Packit ae235b
                             gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum = g_task_get_source_object (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  GList *targets;
Packit ae235b
Packit ae235b
  targets = g_resolver_lookup_service_finish (srv_enum->resolver,
Packit ae235b
                                              result, &error);
Packit ae235b
Packit ae235b
  if (!targets && g_error_matches (error, G_RESOLVER_ERROR,
Packit ae235b
                                   G_RESOLVER_ERROR_NOT_FOUND))
Packit ae235b
    {
Packit ae235b
      targets = g_network_service_fallback_targets (srv_enum->srv);
Packit ae235b
      if (targets)
Packit ae235b
        g_clear_error (&error);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  if (error)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      srv_enum->t = srv_enum->srv->priv->targets = targets;
Packit ae235b
      next_async_have_targets (task);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
next_async_have_targets (GTask *task)
Packit ae235b
{
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum = g_task_get_source_object (task);
Packit ae235b
Packit ae235b
  /* Delegate to GNetworkAddress */
Packit ae235b
  if (srv_enum->addr_enum == NULL && srv_enum->t)
Packit ae235b
    {
Packit ae235b
      GSocketConnectable *addr;
Packit ae235b
      GSrvTarget *target = srv_enum->t->data;
Packit ae235b
Packit ae235b
      srv_enum->t = g_list_next (srv_enum->t);
Packit ae235b
      addr = g_network_address_new (g_srv_target_get_hostname (target),
Packit ae235b
                                    (guint16) g_srv_target_get_port (target));
Packit ae235b
Packit ae235b
      if (srv_enum->use_proxy)
Packit ae235b
        srv_enum->addr_enum = g_socket_connectable_proxy_enumerate (addr);
Packit ae235b
      else
Packit ae235b
        srv_enum->addr_enum = g_socket_connectable_enumerate (addr);
Packit ae235b
Packit ae235b
      g_object_unref (addr);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  if (srv_enum->addr_enum)
Packit ae235b
    {
Packit ae235b
      g_socket_address_enumerator_next_async (srv_enum->addr_enum,
Packit ae235b
                                              g_task_get_cancellable (task),
Packit ae235b
                                              next_async_have_address,
Packit ae235b
                                              task);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      if (srv_enum->error)
Packit ae235b
        {
Packit ae235b
          g_task_return_error (task, srv_enum->error);
Packit ae235b
          srv_enum->error = NULL;
Packit ae235b
        }
Packit ae235b
      else
Packit ae235b
        g_task_return_pointer (task, NULL, NULL);
Packit ae235b
Packit ae235b
      g_object_unref (task);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
next_async_have_address (GObject      *source_object,
Packit ae235b
                         GAsyncResult *result,
Packit ae235b
                         gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum = g_task_get_source_object (task);
Packit ae235b
  GSocketAddress *address;
Packit ae235b
  GError *error = NULL;
Packit ae235b
  
Packit ae235b
  address = g_socket_address_enumerator_next_finish (srv_enum->addr_enum,
Packit ae235b
                                                     result,
Packit ae235b
                                                     &error);
Packit ae235b
Packit ae235b
  if (error)
Packit ae235b
    {
Packit ae235b
      if (srv_enum->error == NULL)
Packit ae235b
        srv_enum->error = error;
Packit ae235b
      else
Packit ae235b
        g_error_free (error);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  if (!address)
Packit ae235b
    {
Packit ae235b
      g_object_unref (srv_enum->addr_enum);
Packit ae235b
      srv_enum->addr_enum = NULL;
Packit ae235b
Packit ae235b
      next_async_have_targets (task);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      g_task_return_pointer (task, address, g_object_unref);
Packit ae235b
      g_object_unref (task);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static GSocketAddress *
Packit ae235b
g_network_service_address_enumerator_next_finish (GSocketAddressEnumerator  *enumerator,
Packit ae235b
                                                  GAsyncResult              *result,
Packit ae235b
                                                  GError                   **error)
Packit ae235b
{
Packit ae235b
  return g_task_propagate_pointer (G_TASK (result), error);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
_g_network_service_address_enumerator_init (GNetworkServiceAddressEnumerator *enumerator)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_network_service_address_enumerator_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum =
Packit ae235b
    G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (object);
Packit ae235b
Packit ae235b
  if (srv_enum->srv)
Packit ae235b
    g_object_unref (srv_enum->srv);
Packit ae235b
Packit ae235b
  if (srv_enum->addr_enum)
Packit ae235b
    g_object_unref (srv_enum->addr_enum);
Packit ae235b
Packit ae235b
  if (srv_enum->resolver)
Packit ae235b
    g_object_unref (srv_enum->resolver);
Packit ae235b
Packit ae235b
  if (srv_enum->error)
Packit ae235b
    g_error_free (srv_enum->error);
Packit ae235b
Packit ae235b
  G_OBJECT_CLASS (_g_network_service_address_enumerator_parent_class)->finalize (object);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
_g_network_service_address_enumerator_class_init (GNetworkServiceAddressEnumeratorClass *srvenum_class)
Packit ae235b
{
Packit ae235b
  GObjectClass *object_class = G_OBJECT_CLASS (srvenum_class);
Packit ae235b
  GSocketAddressEnumeratorClass *enumerator_class =
Packit ae235b
    G_SOCKET_ADDRESS_ENUMERATOR_CLASS (srvenum_class);
Packit ae235b
Packit ae235b
  enumerator_class->next        = g_network_service_address_enumerator_next;
Packit ae235b
  enumerator_class->next_async  = g_network_service_address_enumerator_next_async;
Packit ae235b
  enumerator_class->next_finish = g_network_service_address_enumerator_next_finish;
Packit ae235b
Packit ae235b
  object_class->finalize = g_network_service_address_enumerator_finalize;
Packit ae235b
}
Packit ae235b
Packit ae235b
static GSocketAddressEnumerator *
Packit ae235b
g_network_service_connectable_enumerate (GSocketConnectable *connectable)
Packit ae235b
{
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum;
Packit ae235b
Packit ae235b
  srv_enum = g_object_new (G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR, NULL);
Packit ae235b
  srv_enum->srv = g_object_ref (G_NETWORK_SERVICE (connectable));
Packit ae235b
  srv_enum->resolver = g_resolver_get_default ();
Packit ae235b
  srv_enum->use_proxy = FALSE;
Packit ae235b
Packit ae235b
  return G_SOCKET_ADDRESS_ENUMERATOR (srv_enum);
Packit ae235b
}
Packit ae235b
Packit ae235b
static GSocketAddressEnumerator *
Packit ae235b
g_network_service_connectable_proxy_enumerate (GSocketConnectable *connectable)
Packit ae235b
{
Packit ae235b
  GSocketAddressEnumerator *addr_enum;
Packit ae235b
  GNetworkServiceAddressEnumerator *srv_enum;
Packit ae235b
Packit ae235b
  addr_enum = g_network_service_connectable_enumerate (connectable);
Packit ae235b
  srv_enum = G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (addr_enum);
Packit ae235b
  srv_enum->use_proxy = TRUE;
Packit ae235b
Packit ae235b
  return addr_enum;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
g_network_service_connectable_to_string (GSocketConnectable *connectable)
Packit ae235b
{
Packit ae235b
  GNetworkService *service;
Packit ae235b
Packit ae235b
  service = G_NETWORK_SERVICE (connectable);
Packit ae235b
Packit ae235b
  return g_strdup_printf ("(%s, %s, %s, %s)", service->priv->service,
Packit ae235b
                          service->priv->protocol, service->priv->domain,
Packit ae235b
                          service->priv->scheme);
Packit ae235b
}