GNetworkService: fix iteration of multiple addresses
[platform/upstream/glib.git] / gio / gnetworkservice.c
1 /* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
2
3 /* GIO - GLib Input, Output and Streaming Library
4  *
5  * Copyright (C) 2008 Red Hat, Inc.
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General
18  * Public License along with this library; if not, write to the
19  * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
20  * Boston, MA 02111-1307, USA.
21  */
22
23 #include "config.h"
24 #include <glib.h>
25 #include "glibintl.h"
26
27 #include "gnetworkservice.h"
28
29 #include "gcancellable.h"
30 #include "ginetaddress.h"
31 #include "ginetsocketaddress.h"
32 #include "gioerror.h"
33 #include "gnetworkaddress.h"
34 #include "gnetworkingprivate.h"
35 #include "gresolver.h"
36 #include "gsimpleasyncresult.h"
37 #include "gsocketaddressenumerator.h"
38 #include "gsocketconnectable.h"
39 #include "gsrvtarget.h"
40
41 #include <stdlib.h>
42 #include <string.h>
43
44
45 /**
46  * SECTION:gnetworkservice
47  * @short_description: A GSocketConnectable for resolving SRV records
48  * @include: gio/gio.h
49  *
50  * Like #GNetworkAddress does with hostnames, #GNetworkService
51  * provides an easy way to resolve a SRV record, and then attempt to
52  * connect to one of the hosts that implements that service, handling
53  * service priority/weighting, multiple IP addresses, and multiple
54  * address families.
55  *
56  * See #GSrvTarget for more information about SRV records, and see
57  * #GSocketConnectable for and example of using the connectable
58  * interface.
59  */
60
61 /**
62  * GNetworkService:
63  *
64  * A #GSocketConnectable for resolving a SRV record and connecting to
65  * that service.
66  */
67
68 struct _GNetworkServicePrivate
69 {
70   gchar *service, *protocol, *domain, *scheme;
71   GList *targets;
72 };
73
74 enum {
75   PROP_0,
76   PROP_SERVICE,
77   PROP_PROTOCOL,
78   PROP_DOMAIN,
79   PROP_SCHEME
80 };
81
82 static void g_network_service_set_property (GObject      *object,
83                                             guint         prop_id,
84                                             const GValue *value,
85                                             GParamSpec   *pspec);
86 static void g_network_service_get_property (GObject      *object,
87                                             guint         prop_id,
88                                             GValue       *value,
89                                             GParamSpec   *pspec);
90
91 static void                      g_network_service_connectable_iface_init       (GSocketConnectableIface *iface);
92 static GSocketAddressEnumerator *g_network_service_connectable_enumerate        (GSocketConnectable      *connectable);
93 static GSocketAddressEnumerator *g_network_service_connectable_proxy_enumerate  (GSocketConnectable      *connectable);
94
95 G_DEFINE_TYPE_WITH_CODE (GNetworkService, g_network_service, G_TYPE_OBJECT,
96                          G_IMPLEMENT_INTERFACE (G_TYPE_SOCKET_CONNECTABLE,
97                                                 g_network_service_connectable_iface_init))
98
99 static void
100 g_network_service_finalize (GObject *object)
101 {
102   GNetworkService *srv = G_NETWORK_SERVICE (object);
103
104   g_free (srv->priv->service);
105   g_free (srv->priv->protocol);
106   g_free (srv->priv->domain);
107   g_free (srv->priv->scheme);
108
109   if (srv->priv->targets)
110     g_resolver_free_targets (srv->priv->targets);
111
112   G_OBJECT_CLASS (g_network_service_parent_class)->finalize (object);
113 }
114
115 static void
116 g_network_service_class_init (GNetworkServiceClass *klass)
117 {
118   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
119
120   g_type_class_add_private (klass, sizeof (GNetworkServicePrivate));
121
122   gobject_class->set_property = g_network_service_set_property;
123   gobject_class->get_property = g_network_service_get_property;
124   gobject_class->finalize = g_network_service_finalize;
125
126   g_object_class_install_property (gobject_class, PROP_SERVICE,
127                                    g_param_spec_string ("service",
128                                                         P_("Service"),
129                                                         P_("Service name, eg \"ldap\""),
130                                                         NULL,
131                                                         G_PARAM_READWRITE |
132                                                         G_PARAM_CONSTRUCT_ONLY |
133                                                         G_PARAM_STATIC_STRINGS));
134   g_object_class_install_property (gobject_class, PROP_PROTOCOL,
135                                    g_param_spec_string ("protocol",
136                                                         P_("Protocol"),
137                                                         P_("Network protocol, eg \"tcp\""),
138                                                         NULL,
139                                                         G_PARAM_READWRITE |
140                                                         G_PARAM_CONSTRUCT_ONLY |
141                                                         G_PARAM_STATIC_STRINGS));
142   g_object_class_install_property (gobject_class, PROP_DOMAIN,
143                                    g_param_spec_string ("domain",
144                                                         P_("Domain"),
145                                                         P_("Network domain, eg, \"example.com\""),
146                                                         NULL,
147                                                         G_PARAM_READWRITE |
148                                                         G_PARAM_CONSTRUCT_ONLY |
149                                                         G_PARAM_STATIC_STRINGS));
150   g_object_class_install_property (gobject_class, PROP_DOMAIN,
151                                    g_param_spec_string ("scheme",
152                                                         P_("Scheme"),
153                                                         P_("Network scheme (default is to use service)"),
154                                                         NULL,
155                                                         G_PARAM_READWRITE |
156                                                         G_PARAM_STATIC_STRINGS));
157
158 }
159
160 static void
161 g_network_service_connectable_iface_init (GSocketConnectableIface *connectable_iface)
162 {
163   connectable_iface->enumerate = g_network_service_connectable_enumerate;
164   connectable_iface->proxy_enumerate = g_network_service_connectable_proxy_enumerate;
165 }
166
167 static void
168 g_network_service_init (GNetworkService *srv)
169 {
170   srv->priv = G_TYPE_INSTANCE_GET_PRIVATE (srv, G_TYPE_NETWORK_SERVICE,
171                                            GNetworkServicePrivate);
172 }
173
174 static void
175 g_network_service_set_property (GObject      *object,
176                                 guint         prop_id,
177                                 const GValue *value,
178                                 GParamSpec   *pspec)
179 {
180   GNetworkService *srv = G_NETWORK_SERVICE (object);
181
182   switch (prop_id)
183     {
184     case PROP_SERVICE:
185       srv->priv->service = g_value_dup_string (value);
186       break;
187
188     case PROP_PROTOCOL:
189       srv->priv->protocol = g_value_dup_string (value);
190       break;
191
192     case PROP_DOMAIN:
193       srv->priv->domain = g_value_dup_string (value);
194       break;
195
196     case PROP_SCHEME:
197       g_network_service_set_scheme (srv, g_value_get_string (value));
198       break;
199
200     default:
201       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
202       break;
203     }
204 }
205
206 static void
207 g_network_service_get_property (GObject    *object,
208                                 guint       prop_id,
209                                 GValue     *value,
210                                 GParamSpec *pspec)
211 {
212   GNetworkService *srv = G_NETWORK_SERVICE (object);
213
214   switch (prop_id)
215     {
216     case PROP_SERVICE:
217       g_value_set_string (value, g_network_service_get_service (srv));
218       break;
219
220     case PROP_PROTOCOL:
221       g_value_set_string (value, g_network_service_get_protocol (srv));
222       break;
223
224     case PROP_DOMAIN:
225       g_value_set_string (value, g_network_service_get_domain (srv));
226       break;
227
228     case PROP_SCHEME:
229       g_value_set_string (value, g_network_service_get_scheme (srv));
230       break;
231
232     default:
233       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
234       break;
235     }
236 }
237
238 /**
239  * g_network_service_new:
240  * @service: the service type to look up (eg, "ldap")
241  * @protocol: the networking protocol to use for @service (eg, "tcp")
242  * @domain: the DNS domain to look up the service in
243  *
244  * Creates a new #GNetworkService representing the given @service,
245  * @protocol, and @domain. This will initially be unresolved; use the
246  * #GSocketConnectable interface to resolve it.
247  *
248  * Return value: (transfer full): a new #GNetworkService
249  *
250  * Since: 2.22
251  */
252 GSocketConnectable *
253 g_network_service_new (const gchar *service,
254                        const gchar *protocol,
255                        const gchar *domain)
256 {
257   return g_object_new (G_TYPE_NETWORK_SERVICE,
258                        "service", service,
259                        "protocol", protocol,
260                        "domain", domain,
261                        NULL);
262 }
263
264 /**
265  * g_network_service_get_service:
266  * @srv: a #GNetworkService
267  *
268  * Gets @srv's service name (eg, "ldap").
269  *
270  * Return value: @srv's service name
271  *
272  * Since: 2.22
273  */
274 const gchar *
275 g_network_service_get_service (GNetworkService *srv)
276 {
277   g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
278
279   return srv->priv->service;
280 }
281
282 /**
283  * g_network_service_get_protocol:
284  * @srv: a #GNetworkService
285  *
286  * Gets @srv's protocol name (eg, "tcp").
287  *
288  * Return value: @srv's protocol name
289  *
290  * Since: 2.22
291  */
292 const gchar *
293 g_network_service_get_protocol (GNetworkService *srv)
294 {
295   g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
296
297   return srv->priv->protocol;
298 }
299
300 /**
301  * g_network_service_get_domain:
302  * @srv: a #GNetworkService
303  *
304  * Gets the domain that @srv serves. This might be either UTF-8 or
305  * ASCII-encoded, depending on what @srv was created with.
306  *
307  * Return value: @srv's domain name
308  *
309  * Since: 2.22
310  */
311 const gchar *
312 g_network_service_get_domain (GNetworkService *srv)
313 {
314   g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
315
316   return srv->priv->domain;
317 }
318
319 /**
320  * g_network_service_get_scheme:
321  * @srv: a #GNetworkService
322  *
323  * Get's the URI scheme used to resolve proxies. By default, the service name
324  * is used as scheme.
325  *
326  * Return value: @srv's scheme name
327  *
328  * Since: 2.26
329  */
330 const gchar *
331 g_network_service_get_scheme (GNetworkService *srv)
332 {
333   g_return_val_if_fail (G_IS_NETWORK_SERVICE (srv), NULL);
334
335   if (srv->priv->scheme)
336     return srv->priv->scheme;
337   else
338     return srv->priv->service;
339 }
340
341 /**
342  * g_network_service_set_scheme:
343  * @srv: a #GNetworkService
344  * @scheme: a URI scheme
345  *
346  * Set's the URI scheme used to resolve proxies. By default, the service name
347  * is used as scheme.
348  *
349  * Since: 2.26
350  */
351 void
352 g_network_service_set_scheme (GNetworkService *srv,
353                               const gchar     *scheme)
354 {
355   g_return_if_fail (G_IS_NETWORK_SERVICE (srv));
356
357   if (srv->priv->scheme)
358     g_free (srv->priv->scheme);
359   srv->priv->scheme = g_strdup (scheme);
360
361   g_object_notify (G_OBJECT (srv), "scheme");
362 }
363
364
365 #define G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR (_g_network_service_address_enumerator_get_type ())
366 #define G_NETWORK_SERVICE_ADDRESS_ENUMERATOR(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR, GNetworkServiceAddressEnumerator))
367
368 typedef struct {
369   GSocketAddressEnumerator parent_instance;
370
371   GResolver *resolver;
372   GNetworkService *srv;
373   GSocketAddressEnumerator *addr_enum;
374   GList *t;
375   gboolean use_proxy;
376
377   GError *error;
378
379   /* For async operation */
380   GCancellable *cancellable;
381   GSimpleAsyncResult *result;
382 } GNetworkServiceAddressEnumerator;
383
384 typedef struct {
385   GSocketAddressEnumeratorClass parent_class;
386
387 } GNetworkServiceAddressEnumeratorClass;
388
389 G_DEFINE_TYPE (GNetworkServiceAddressEnumerator, _g_network_service_address_enumerator, G_TYPE_SOCKET_ADDRESS_ENUMERATOR)
390
391 static GSocketAddress *
392 g_network_service_address_enumerator_next (GSocketAddressEnumerator  *enumerator,
393                                            GCancellable              *cancellable,
394                                            GError                   **error)
395 {
396   GNetworkServiceAddressEnumerator *srv_enum =
397     G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (enumerator);
398   GSocketAddress *ret = NULL;
399
400   /* If we haven't yet resolved srv, do that */
401   if (!srv_enum->srv->priv->targets)
402     {
403       GList *targets;
404
405       targets = g_resolver_lookup_service (srv_enum->resolver,
406                                            srv_enum->srv->priv->service,
407                                            srv_enum->srv->priv->protocol,
408                                            srv_enum->srv->priv->domain,
409                                            cancellable, error);
410       if (!targets)
411         return NULL;
412
413       srv_enum->srv->priv->targets = targets;
414       srv_enum->t = srv_enum->srv->priv->targets;
415     }
416
417   /* Delegate to GNetworkAddress */
418   do
419     {
420       if (srv_enum->addr_enum == NULL && srv_enum->t)
421         {
422           GError *error = NULL;
423           gchar *uri;
424           gchar *hostname;
425           GSocketConnectable *addr;
426           GSrvTarget *target = srv_enum->t->data;
427
428           srv_enum->t = g_list_next (srv_enum->t);
429
430           hostname = g_hostname_to_ascii (g_srv_target_get_hostname (target));
431
432           if (hostname == NULL)
433             {
434               if (srv_enum->error == NULL)
435                 srv_enum->error =
436                   g_error_new (G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
437                                "Received invalid hostname '%s' from GSrvTarget",
438                                g_srv_target_get_hostname (target));
439               continue;
440             }
441
442           uri = _g_uri_from_authority (g_network_service_get_scheme (srv_enum->srv),
443                                        hostname,
444                                        g_srv_target_get_port (target),
445                                        NULL);
446           g_free (hostname);
447
448           addr = g_network_address_parse_uri (uri,
449                                               g_srv_target_get_port (target),
450                                               &error);
451           g_free (uri);
452
453           if (addr == NULL)
454             {
455               if (srv_enum->error == NULL)
456                 srv_enum->error = error;
457               continue;
458             }
459
460           if (srv_enum->use_proxy)
461             srv_enum->addr_enum = g_socket_connectable_proxy_enumerate (addr);
462           else
463             srv_enum->addr_enum = g_socket_connectable_enumerate (addr);
464           g_object_unref (addr);
465         }
466
467       if (srv_enum->addr_enum)
468         {
469           GError *error = NULL;
470
471           ret = g_socket_address_enumerator_next (srv_enum->addr_enum,
472                                                   cancellable,
473                                                   &error);
474
475           if (error)
476             {
477               if (srv_enum->error == NULL)
478                 srv_enum->error = error;
479               else
480                 g_error_free (error);
481             }
482
483           if (!ret)
484             {
485               g_object_unref (srv_enum->addr_enum);
486               srv_enum->addr_enum = NULL;
487             }
488         }
489     }
490   while (srv_enum->addr_enum == NULL && srv_enum->t);
491
492   if (ret == NULL && srv_enum->error)
493     {
494       g_propagate_error (error, srv_enum->error);
495       srv_enum->error = NULL;
496     }
497
498   return ret;
499 }
500
501 static void next_async_resolved_targets   (GObject                          *source_object,
502                                            GAsyncResult                     *result,
503                                            gpointer                          user_data);
504 static void next_async_have_targets       (GNetworkServiceAddressEnumerator *srv_enum);
505 static void next_async_have_address       (GObject                          *source_object,
506                                            GAsyncResult                     *result,
507                                            gpointer                          user_data);
508
509 static void
510 g_network_service_address_enumerator_next_async (GSocketAddressEnumerator  *enumerator,
511                                                  GCancellable              *cancellable,
512                                                  GAsyncReadyCallback        callback,
513                                                  gpointer                   user_data)
514 {
515   GNetworkServiceAddressEnumerator *srv_enum =
516     G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (enumerator);
517
518   g_return_if_fail (srv_enum->result == NULL);
519
520   srv_enum->result = g_simple_async_result_new (G_OBJECT (enumerator),
521                                                 callback, user_data,
522                                                 g_network_service_address_enumerator_next_async);
523
524   if (cancellable)
525     srv_enum->cancellable = g_object_ref (cancellable);
526
527   /* If we haven't yet resolved srv, do that */
528   if (!srv_enum->srv->priv->targets)
529     {
530       g_resolver_lookup_service_async (srv_enum->resolver,
531                                        srv_enum->srv->priv->service,
532                                        srv_enum->srv->priv->protocol,
533                                        srv_enum->srv->priv->domain,
534                                        cancellable,
535                                        next_async_resolved_targets,
536                                        srv_enum);
537     }
538   else
539     next_async_have_targets (srv_enum);
540 }
541
542 static void
543 next_async_resolved_targets (GObject      *source_object,
544                              GAsyncResult *result,
545                              gpointer      user_data)
546 {
547   GNetworkServiceAddressEnumerator *srv_enum = user_data;
548   GError *error = NULL;
549
550   srv_enum->srv->priv->targets =
551     g_resolver_lookup_service_finish (srv_enum->resolver, result, &error);
552
553   if (error)
554     {
555       GSimpleAsyncResult *simple = srv_enum->result;
556
557       srv_enum->result = NULL;
558       g_simple_async_result_set_from_error (simple, error);
559       g_error_free (error);
560       g_simple_async_result_complete (simple);
561       g_object_unref (simple);
562     }
563   else
564     {
565       srv_enum->t = srv_enum->srv->priv->targets;
566       next_async_have_targets (srv_enum);
567     }
568 }
569
570 static void
571 next_async_have_targets (GNetworkServiceAddressEnumerator *srv_enum)
572 {
573   /* Delegate to GNetworkAddress */
574   if (srv_enum->addr_enum == NULL && srv_enum->t)
575     {
576       GSocketConnectable *addr;
577       GSrvTarget *target = srv_enum->t->data;
578
579       srv_enum->t = g_list_next (srv_enum->t);
580       addr = g_network_address_new (g_srv_target_get_hostname (target),
581                                     (guint16) g_srv_target_get_port (target));
582
583       if (srv_enum->use_proxy)
584         srv_enum->addr_enum = g_socket_connectable_proxy_enumerate (addr);
585       else
586         srv_enum->addr_enum = g_socket_connectable_enumerate (addr);
587
588       g_object_unref (addr);
589     }
590
591   if (srv_enum->addr_enum)
592     {
593       g_socket_address_enumerator_next_async (srv_enum->addr_enum,
594                                               srv_enum->cancellable,
595                                               next_async_have_address,
596                                               srv_enum);
597     }
598   else
599     {
600       GSimpleAsyncResult *simple = srv_enum->result;
601       srv_enum->result = NULL;
602
603       if (srv_enum->error)
604         {
605           g_simple_async_result_set_from_error (simple, srv_enum->error);
606           g_error_free (srv_enum->error);
607           srv_enum->error = NULL;
608         }
609
610       g_simple_async_result_complete (simple);
611       g_object_unref (simple);
612     }
613 }
614
615 static void
616 next_async_have_address (GObject      *source_object,
617                          GAsyncResult *result,
618                          gpointer      user_data)
619 {
620   GNetworkServiceAddressEnumerator *srv_enum = user_data;
621   GSocketAddress *address;
622   GError *error = NULL;
623   
624   address = g_socket_address_enumerator_next_finish (srv_enum->addr_enum,
625                                                      result,
626                                                      &error);
627
628   if (error)
629     {
630       if (srv_enum->error == NULL)
631         srv_enum->error = error;
632       else
633         g_error_free (error);
634     }
635
636   if (!address)
637     {
638       g_object_unref (srv_enum->addr_enum);
639       srv_enum->addr_enum = NULL;
640
641       next_async_have_targets (srv_enum);
642     }
643   else
644     {
645       GSimpleAsyncResult *simple = srv_enum->result;
646
647       srv_enum->result = NULL;
648
649       if (address)
650         g_simple_async_result_set_op_res_gpointer (simple,
651                                                    address, g_object_unref);
652
653       g_simple_async_result_complete (simple);
654       g_object_unref (simple);
655     }
656 }
657
658 static GSocketAddress *
659 g_network_service_address_enumerator_next_finish (GSocketAddressEnumerator  *enumerator,
660                                                   GAsyncResult              *result,
661                                                   GError                   **error)
662 {
663   GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (result);
664   GSocketAddress *sockaddr;
665
666   if (g_simple_async_result_propagate_error (simple, error))
667     return NULL;
668
669   sockaddr = g_simple_async_result_get_op_res_gpointer (simple);
670   return sockaddr ? g_object_ref (sockaddr) : NULL;
671 }
672
673 static void
674 _g_network_service_address_enumerator_init (GNetworkServiceAddressEnumerator *enumerator)
675 {
676 }
677
678 static void
679 g_network_service_address_enumerator_finalize (GObject *object)
680 {
681   GNetworkServiceAddressEnumerator *srv_enum =
682     G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (object);
683
684   if (srv_enum->srv)
685     g_object_unref (srv_enum->srv);
686
687   if (srv_enum->addr_enum)
688     g_object_unref (srv_enum->addr_enum);
689
690   if (srv_enum->resolver)
691     g_object_unref (srv_enum->resolver);
692
693   if (srv_enum->result)
694     g_object_unref (srv_enum->result);
695
696   if (srv_enum->cancellable)
697     g_object_unref (srv_enum->cancellable);
698
699   if (srv_enum->error)
700     g_error_free (srv_enum->error);
701
702   G_OBJECT_CLASS (_g_network_service_address_enumerator_parent_class)->finalize (object);
703 }
704
705 static void
706 _g_network_service_address_enumerator_class_init (GNetworkServiceAddressEnumeratorClass *srvenum_class)
707 {
708   GObjectClass *object_class = G_OBJECT_CLASS (srvenum_class);
709   GSocketAddressEnumeratorClass *enumerator_class =
710     G_SOCKET_ADDRESS_ENUMERATOR_CLASS (srvenum_class);
711
712   enumerator_class->next        = g_network_service_address_enumerator_next;
713   enumerator_class->next_async  = g_network_service_address_enumerator_next_async;
714   enumerator_class->next_finish = g_network_service_address_enumerator_next_finish;
715
716   object_class->finalize = g_network_service_address_enumerator_finalize;
717 }
718
719 static GSocketAddressEnumerator *
720 g_network_service_connectable_enumerate (GSocketConnectable *connectable)
721 {
722   GNetworkServiceAddressEnumerator *srv_enum;
723
724   srv_enum = g_object_new (G_TYPE_NETWORK_SERVICE_ADDRESS_ENUMERATOR, NULL);
725   srv_enum->srv = g_object_ref (connectable);
726   srv_enum->resolver = g_resolver_get_default ();
727   srv_enum->use_proxy = FALSE;
728
729   return G_SOCKET_ADDRESS_ENUMERATOR (srv_enum);
730 }
731
732 static GSocketAddressEnumerator *
733 g_network_service_connectable_proxy_enumerate (GSocketConnectable *connectable)
734 {
735   GSocketAddressEnumerator *addr_enum;
736   GNetworkServiceAddressEnumerator *srv_enum;
737
738   addr_enum = g_network_service_connectable_enumerate (connectable);
739   srv_enum = G_NETWORK_SERVICE_ADDRESS_ENUMERATOR (addr_enum);
740   srv_enum->use_proxy = TRUE;
741
742   return addr_enum;
743 }