2 * This library is free software you can redistribute it and/or modify it
3 * under the terms of the GNU Lesser General Public License as published by
4 * the Free Software Foundation.
6 * This library is distributed in the hope that it will be useful, but
7 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
8 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
11 * You should have received a copy of the GNU Lesser General Public License
12 * along with this library; if not, see <http://www.gnu.org/licenses/>.
16 * Jeffrey Stedfast <fejj@ximian.com>
17 * Veerapuram Varadhan <vvaradhan@novell.com>
19 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
32 #ifndef IN6_ARE_ADDR_EQUAL
33 #define IN6_ARE_ADDR_EQUAL(a, b) \
34 (memcmp ((gpointer)(a), (gpointer)(b), sizeof (struct in6_addr)) == 0)
37 #include <netinet/in.h>
38 #include <sys/socket.h>
41 #include <libsoup/soup-address.h>
42 #include <libsoup/soup-uri.h>
45 #define E_PROXY_GET_PRIVATE(obj) \
46 (G_TYPE_INSTANCE_GET_PRIVATE \
47 ((obj), E_TYPE_PROXY, EProxyPrivate))
49 G_DEFINE_TYPE (EProxy, e_proxy, G_TYPE_OBJECT)
55 PROXY_TYPE_SYSTEM = 0,
58 PROXY_TYPE_AUTO_URL /* no auto-proxy at the moment */
63 E_PROXY_KEY_USE_HTTP_PROXY,
64 E_PROXY_KEY_HTTP_HOST,
65 E_PROXY_KEY_HTTP_PORT,
66 E_PROXY_KEY_HTTP_USE_AUTH,
67 E_PROXY_KEY_HTTP_AUTH_USER,
68 E_PROXY_KEY_HTTP_AUTH_PWD,
69 E_PROXY_KEY_HTTP_IGNORE_HOSTS,
70 E_PROXY_KEY_HTTPS_HOST,
71 E_PROXY_KEY_HTTPS_PORT,
72 E_PROXY_KEY_SOCKS_HOST,
73 E_PROXY_KEY_SOCKS_PORT,
74 E_PROXY_KEY_AUTOCONFIG_URL
77 struct _EProxyPrivate {
78 SoupURI *uri_http, *uri_https, *uri_socks;
79 GSList * ign_hosts; /* List of hostnames. (Strings) */
80 GSList * ign_addrs; /* List of hostaddrs. (ProxyHostAddrs) */
81 gboolean use_proxy; /* Is our-proxy enabled? */
83 GSettings *evolution_proxy_settings;
84 GSettings *proxy_settings;
85 GSettings *proxy_http_settings;
86 GSettings *proxy_https_settings;
87 GSettings *proxy_socks_settings;
90 /* Enum definition is copied from gnome-vfs/modules/http-proxy.c */
97 ProxyAddrType type; /* Specifies whether IPV4 or IPV6 */
98 gpointer addr; /* Either in_addr * or in6_addr * */
99 gpointer mask; /* Either in_addr * or in6_addr * */
108 static guint signals[LAST_SIGNAL] = { 0 };
110 /* Forward declarations. */
112 static void ipv6_network_addr (const struct in6_addr *addr,
113 const struct in6_addr *mask,
114 struct in6_addr *res);
117 ep_free_proxy_host_addr (ProxyHostAddr *host)
133 ep_read_key_boolean (EProxy *proxy,
136 gboolean res = FALSE;
138 g_return_val_if_fail (E_IS_PROXY (proxy), FALSE);
141 case E_PROXY_KEY_USE_HTTP_PROXY:
142 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
143 /* it's not used in the UI, thus behave like always set to TRUE */
144 res = TRUE; /* g_settings_get_boolean (proxy->priv->proxy_http_settings, "enabled"); */
146 res = g_settings_get_boolean (proxy->priv->evolution_proxy_settings, "use-http-proxy");
148 case E_PROXY_KEY_HTTP_USE_AUTH:
149 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
150 res = g_settings_get_boolean (proxy->priv->proxy_http_settings, "use-authentication");
152 res = g_settings_get_boolean (proxy->priv->evolution_proxy_settings, "use-authentication");
155 g_warn_if_reached ();
163 ep_read_key_int (EProxy *proxy,
168 g_return_val_if_fail (E_IS_PROXY (proxy), 0);
171 case E_PROXY_KEY_HTTP_PORT:
172 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
173 res = g_settings_get_int (proxy->priv->proxy_http_settings, "port");
175 res = g_settings_get_int (proxy->priv->evolution_proxy_settings, "http-port");
177 case E_PROXY_KEY_HTTPS_PORT:
178 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
179 res = g_settings_get_int (proxy->priv->proxy_https_settings, "port");
181 res = g_settings_get_int (proxy->priv->evolution_proxy_settings, "secure-port");
183 case E_PROXY_KEY_SOCKS_PORT:
184 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
185 res = g_settings_get_int (proxy->priv->proxy_socks_settings, "port");
187 res = g_settings_get_int (proxy->priv->evolution_proxy_settings, "socks-port");
190 g_warn_if_reached ();
197 /* free returned pointer with g_free() */
199 ep_read_key_string (EProxy *proxy,
204 g_return_val_if_fail (E_IS_PROXY (proxy), NULL);
207 case E_PROXY_KEY_MODE:
208 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
209 res = g_settings_get_string (proxy->priv->proxy_settings, "mode");
211 g_warn_if_reached ();
213 case E_PROXY_KEY_HTTP_HOST:
214 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
215 res = g_settings_get_string (proxy->priv->proxy_http_settings, "host");
217 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "http-host");
219 case E_PROXY_KEY_HTTPS_HOST:
220 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
221 res = g_settings_get_string (proxy->priv->proxy_https_settings, "host");
223 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "secure-host");
225 case E_PROXY_KEY_SOCKS_HOST:
226 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
227 res = g_settings_get_string (proxy->priv->proxy_socks_settings, "host");
229 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "socks-host");
231 case E_PROXY_KEY_HTTP_AUTH_USER:
232 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
233 res = g_settings_get_string (proxy->priv->proxy_http_settings, "authentication-user");
235 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "authentication-user");
237 case E_PROXY_KEY_HTTP_AUTH_PWD:
238 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
239 res = g_settings_get_string (proxy->priv->proxy_http_settings, "authentication-password");
241 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "authentication-password");
243 case E_PROXY_KEY_AUTOCONFIG_URL:
244 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
245 res = g_settings_get_string (proxy->priv->proxy_settings, "autoconfig-url");
247 res = g_settings_get_string (proxy->priv->evolution_proxy_settings, "autoconfig-url");
250 g_warn_if_reached ();
257 /* list of newly allocated strings, use g_free() for each member and free list itself too */
259 ep_read_key_list (EProxy *proxy,
265 g_return_val_if_fail (E_IS_PROXY (proxy), NULL);
268 case E_PROXY_KEY_HTTP_IGNORE_HOSTS:
269 if (proxy->priv->type == PROXY_TYPE_SYSTEM)
270 strv = g_settings_get_strv (proxy->priv->proxy_settings, "ignore-hosts");
272 strv = g_settings_get_strv (proxy->priv->evolution_proxy_settings, "ignore-hosts");
275 g_warn_if_reached ();
282 for (ii = 0; strv && strv[ii]; ii++) {
283 res = g_slist_prepend (res, g_strdup (strv[ii]));
288 res = g_slist_reverse (res);
295 ep_is_in_ignored (EProxy *proxy,
302 g_return_val_if_fail (proxy != NULL, FALSE);
303 g_return_val_if_fail (host != NULL, FALSE);
306 if (!priv->ign_hosts)
309 hn = g_ascii_strdown (host, -1);
311 for (l = priv->ign_hosts; l; l = l->next) {
312 if (*((gchar *) l->data) == '*') {
313 if (g_str_has_suffix (hn, ((gchar *) l->data) + 1)) {
317 } else if (strcmp (hn, l->data) == 0) {
328 ep_need_proxy_http (EProxy *proxy,
331 SoupAddress *addr = NULL;
332 EProxyPrivate *priv = proxy->priv;
333 ProxyHostAddr *p_addr = NULL;
337 /* check for ignored first */
338 if (ep_is_in_ignored (proxy, host))
341 addr = soup_address_new (host, 0);
342 status = soup_address_resolve_sync (addr, NULL);
343 if (status == SOUP_STATUS_OK) {
345 struct sockaddr * so_addr = NULL;
347 so_addr = soup_address_get_sockaddr (addr, &addr_len);
349 /* This will never happen, since we have already called
350 * soup_address_resolve_sync ().
353 g_object_unref (addr);
357 if (so_addr->sa_family == AF_INET) {
358 struct in_addr in, *mask, *addr_in;
360 in = ((struct sockaddr_in *) so_addr)->sin_addr;
361 for (l = priv->ign_addrs; l; l = l->next) {
362 p_addr = (ProxyHostAddr *) l->data;
363 if (p_addr->type == PROXY_IPV4) {
364 addr_in = ((struct in_addr *) p_addr->addr);
365 mask = ((struct in_addr *) p_addr->mask);
366 if ((in.s_addr & mask->s_addr) == addr_in->s_addr) {
367 d (g_print ("Host [%s] doesn't require proxy\n", host));
368 g_object_unref (addr);
374 struct in6_addr in6, net6;
375 struct in_addr *addr_in, *mask;
377 in6 = ((struct sockaddr_in6 *) so_addr)->sin6_addr;
378 for (l = priv->ign_addrs; l; l = l->next) {
379 p_addr = (ProxyHostAddr *) l->data;
380 ipv6_network_addr (&in6, (struct in6_addr *) p_addr->mask, &net6);
381 if (p_addr->type == PROXY_IPV6) {
382 if (IN6_ARE_ADDR_EQUAL (&net6, (struct in6_addr *) p_addr->addr)) {
383 d (g_print ("Host [%s] doesn't require proxy\n", host));
384 g_object_unref (addr);
387 } else if (p_addr->type == PROXY_IPV6 &&
388 IN6_IS_ADDR_V4MAPPED (&net6)) {
391 addr_in = ((struct in_addr *) p_addr->addr);
392 mask = ((struct in_addr *) p_addr->mask);
394 v4addr = net6.s6_addr[12] << 24
395 | net6.s6_addr[13] << 16
396 | net6.s6_addr[14] << 8
398 if ((v4addr & mask->s_addr) != addr_in->s_addr) {
399 d (g_print ("Host [%s] doesn't require proxy\n", host));
400 g_object_unref (addr);
408 d (g_print ("%s needs a proxy to connect to internet\n", host));
409 g_object_unref (addr);
415 ep_need_proxy_https (EProxy *proxy,
418 /* Can we share ignore list from HTTP at all? */
419 return !ep_is_in_ignored (proxy, host);
423 ep_need_proxy_socks (EProxy *proxy,
426 /* Can we share ignore list from HTTP at all? */
427 return !ep_is_in_ignored (proxy, host);
431 ep_manipulate_ipv4 (ProxyHostAddr *host_addr,
432 struct in_addr *addr_in,
435 gboolean has_error = FALSE;
436 struct in_addr *addr, *mask;
441 host_addr->type = PROXY_IPV4;
442 addr = g_new0 (struct in_addr, 1);
443 memcpy (addr, addr_in, sizeof (struct in_addr));
444 mask = g_new0 (struct in_addr, 1);
448 gint width = strtol (netmask, &endptr, 10);
450 if (*endptr != '\0' || width < 0 || width > 32) {
453 mask->s_addr = htonl (~0 << width);
454 addr->s_addr &= mask->s_addr;
456 mask->s_addr = 0xFFFFFFFF;
459 host_addr->addr = addr;
460 host_addr->mask = mask;
466 ipv6_network_addr (const struct in6_addr *addr,
467 const struct in6_addr *mask,
468 struct in6_addr *res)
472 for (i = 0; i < 16; ++i) {
473 res->s6_addr[i] = addr->s6_addr[i] & mask->s6_addr[i];
478 ep_manipulate_ipv6 (ProxyHostAddr *host_addr,
479 struct in6_addr *addr_in6,
482 gboolean has_error = FALSE;
483 struct in6_addr *addr, *mask;
489 host_addr->type = PROXY_IPV6;
491 addr = g_new0 (struct in6_addr, 1);
492 mask = g_new0 (struct in6_addr, 1);
494 for (i = 0; i < 16; ++i) {
495 addr->s6_addr[i] = addr_in6->s6_addr[i];
499 gint width = strtol (netmask, &endptr, 10);
501 if (*endptr != '\0' || width < 0 || width > 128) {
504 for (i = 0; i < 16; ++i) {
505 mask->s6_addr[i] = 0;
507 for (i = 0; i < width / 8; i++) {
508 mask->s6_addr[i] = 0xff;
510 mask->s6_addr[i] = (0xff << (8 - width % 8)) & 0xff;
511 ipv6_network_addr (addr, mask, addr);
513 for (i = 0; i < 16; ++i) {
514 mask->s6_addr[i] = 0xff;
518 host_addr->addr = addr;
519 host_addr->mask = mask;
525 ep_parse_ignore_host (gpointer data,
528 EProxy * proxy = (EProxy *) user_data;
529 EProxyPrivate * priv = NULL;
532 gchar *input, *netmask, *hostname;
533 ProxyHostAddr *host_addr;
534 gboolean has_error = FALSE;
536 if (!proxy || !proxy->priv)
540 input = (gchar *) data;
542 if ((netmask = strrchr (input, '/')) != NULL) {
543 hostname = g_strndup (input, netmask - input);
546 hostname = g_ascii_strdown (input, -1);
549 addr = soup_address_new (hostname, 0);
550 status = soup_address_resolve_sync (addr, NULL);
551 if (status == SOUP_STATUS_OK) {
553 struct sockaddr * so_addr = NULL;
555 host_addr = g_new0 (ProxyHostAddr, 1);
557 so_addr = soup_address_get_sockaddr (addr, &addr_len);
559 /* This will never happen, since we have already called
560 * soup_address_resolve_sync ().
565 if (so_addr->sa_family == AF_INET)
566 has_error = ep_manipulate_ipv4 (
568 &((struct sockaddr_in *) so_addr)->sin_addr,
571 has_error = ep_manipulate_ipv6 (
573 &((struct sockaddr_in6 *) so_addr)->sin6_addr,
577 priv->ign_addrs = g_slist_append (
578 priv->ign_addrs, host_addr);
579 priv->ign_hosts = g_slist_append (
580 priv->ign_hosts, hostname);
585 d (g_print ("Unable to resolve %s\n", hostname));
586 priv->ign_hosts = g_slist_append (priv->ign_hosts, hostname);
589 g_object_unref (addr);
593 ep_change_uri (SoupURI **soup_uri,
596 gboolean changed = FALSE;
598 g_return_val_if_fail (soup_uri != NULL, FALSE);
602 soup_uri_free (*soup_uri);
606 } else if (*soup_uri) {
607 gchar *old = soup_uri_to_string (*soup_uri, FALSE);
610 gint len = strlen (old);
612 /* remove ending slash, if there */
613 if (old[len - 1] == '/')
617 changed = old && uri && g_ascii_strcasecmp (old, uri) != 0;
619 soup_uri_free (*soup_uri);
620 *soup_uri = soup_uri_new (uri);
625 *soup_uri = soup_uri_new (uri);
633 update_proxy_uri (const gchar *uri,
634 const gchar *proxy_user,
635 const gchar *proxy_pw)
637 gchar *res, *user = NULL, *pw = NULL;
640 g_return_val_if_fail (uri != NULL, NULL);
642 if (proxy_user && *proxy_user) {
643 user = soup_uri_encode (proxy_user, ":/;#@?\\");
645 pw = soup_uri_encode (proxy_pw, ":/;#@?\\");
649 return g_strdup (uri);
651 /* here can be only http or https and nothing else */
652 is_https = g_str_has_prefix (uri, "https://");
654 res = g_strdup_printf (
656 is_https ? "https" : "http",
660 uri + strlen ("http://") + (is_https ? 1 : 0));
669 ep_set_proxy (EProxy *proxy,
670 gboolean regen_ign_host_list)
672 gchar *proxy_server, *uri_http = NULL, *uri_https = NULL, *uri_socks = NULL;
673 gint proxy_port, old_type;
674 EProxyPrivate * priv = proxy->priv;
676 gboolean changed = FALSE, sys_manual = TRUE;
678 old_type = priv->type;
679 priv->type = g_settings_get_int (priv->evolution_proxy_settings, "proxy-type");
680 if (priv->type > PROXY_TYPE_AUTO_URL)
681 priv->type = PROXY_TYPE_SYSTEM;
682 changed = priv->type != old_type;
684 if (priv->type == PROXY_TYPE_SYSTEM) {
685 gchar *mode = ep_read_key_string (proxy, E_PROXY_KEY_MODE);
687 /* supporting only manual system proxy setting */
688 sys_manual = mode && g_str_equal (mode, "manual");
693 priv->use_proxy = ep_read_key_boolean (proxy, E_PROXY_KEY_USE_HTTP_PROXY);
694 if (!priv->use_proxy || priv->type == PROXY_TYPE_NO_PROXY || !sys_manual) {
695 changed = ep_change_uri (&priv->uri_http, NULL) || changed;
696 changed = ep_change_uri (&priv->uri_https, NULL) || changed;
697 changed = ep_change_uri (&priv->uri_socks, NULL) || changed;
701 proxy_server = ep_read_key_string (proxy, E_PROXY_KEY_HTTP_HOST);
702 proxy_port = ep_read_key_int (proxy, E_PROXY_KEY_HTTP_PORT);
703 if (proxy_server != NULL && *proxy_server && !g_ascii_isspace (*proxy_server)) {
705 uri_http = g_strdup_printf ("http://%s:%d", proxy_server, proxy_port);
707 uri_http = g_strdup_printf ("http://%s", proxy_server);
710 g_free (proxy_server);
711 d (g_print ("ep_set_proxy: uri_http: %s\n", uri_http));
713 proxy_server = ep_read_key_string (proxy, E_PROXY_KEY_HTTPS_HOST);
714 proxy_port = ep_read_key_int (proxy, E_PROXY_KEY_HTTPS_PORT);
715 if (proxy_server != NULL && *proxy_server && !g_ascii_isspace (*proxy_server)) {
717 uri_https = g_strdup_printf ("https://%s:%d", proxy_server, proxy_port);
719 uri_https = g_strdup_printf ("https://%s", proxy_server);
722 g_free (proxy_server);
723 d (g_print ("ep_set_proxy: uri_https: %s\n", uri_https));
725 proxy_server = ep_read_key_string (proxy, E_PROXY_KEY_SOCKS_HOST);
726 proxy_port = ep_read_key_int (proxy, E_PROXY_KEY_SOCKS_PORT);
727 if (proxy_server != NULL && *proxy_server && !g_ascii_isspace (*proxy_server)) {
729 uri_socks = g_strdup_printf ("socks://%s:%d", proxy_server, proxy_port);
731 uri_socks = g_strdup_printf ("socks://%s", proxy_server);
734 g_free (proxy_server);
735 d (g_print ("ep_set_proxy: uri_socks: %s\n", uri_socks));
737 if (regen_ign_host_list) {
738 if (priv->ign_hosts) {
739 g_slist_foreach (priv->ign_hosts, (GFunc) g_free, NULL);
740 g_slist_free (priv->ign_hosts);
741 priv->ign_hosts = NULL;
744 if (priv->ign_addrs) {
745 g_slist_foreach (priv->ign_addrs, (GFunc) ep_free_proxy_host_addr, NULL);
746 g_slist_free (priv->ign_addrs);
747 priv->ign_addrs = NULL;
750 ignore = ep_read_key_list (proxy, E_PROXY_KEY_HTTP_IGNORE_HOSTS);
752 g_slist_foreach (ignore, (GFunc) ep_parse_ignore_host, proxy);
753 g_slist_foreach (ignore, (GFunc) g_free, NULL);
754 g_slist_free (ignore);
758 if (ep_read_key_boolean (proxy, E_PROXY_KEY_HTTP_USE_AUTH)) {
759 gchar *proxy_user, *proxy_pw, *tmp = NULL, *tmps = NULL;
761 proxy_user = ep_read_key_string (proxy, E_PROXY_KEY_HTTP_AUTH_USER);
762 proxy_pw = ep_read_key_string (proxy, E_PROXY_KEY_HTTP_AUTH_PWD);
764 if (uri_http && proxy_user && *proxy_user) {
766 uri_http = update_proxy_uri (uri_http, proxy_user, proxy_pw);
769 if (uri_https && proxy_user && *proxy_user) {
771 uri_https = update_proxy_uri (uri_https, proxy_user, proxy_pw);
780 changed = ep_change_uri (&priv->uri_http, uri_http) || changed;
781 changed = ep_change_uri (&priv->uri_https, uri_https) || changed;
782 changed = ep_change_uri (&priv->uri_socks, uri_socks) || changed;
790 G_STRFUNC, changed ? 1 : 0,
791 uri_http ? uri_http : "[null]",
792 uri_https ? uri_https : "[null]",
793 uri_socks ? uri_socks : "[null]"));
795 g_signal_emit (proxy, signals[CHANGED], 0);
803 ep_evo_proxy_changed_cb (GSettings *settings,
809 g_return_if_fail (E_IS_PROXY (proxy));
813 d (g_print ("%s: proxy settings changed, key '%s'\n", G_STRFUNC, key ? key : "NULL"));
814 if (g_strcmp0 (key, "proxy-type") == 0) {
815 ep_set_proxy (proxy, TRUE);
816 } else if (priv->type == PROXY_TYPE_SYSTEM) {
820 ep_set_proxy (proxy, g_strcmp0 (key, "ignore-hosts") == 0);
824 ep_sys_proxy_changed_cb (GSettings *settings,
828 g_return_if_fail (proxy != NULL);
830 if (proxy->priv->type != PROXY_TYPE_SYSTEM)
833 ep_set_proxy (proxy, g_strcmp0 (key, "ignore-hosts") == 0);
837 ep_sys_proxy_http_changed_cb (GSettings *settings,
841 g_return_if_fail (proxy != NULL);
843 if (proxy->priv->type != PROXY_TYPE_SYSTEM)
846 ep_set_proxy (proxy, FALSE);
850 ep_sys_proxy_https_changed_cb (GSettings *settings,
854 g_return_if_fail (proxy != NULL);
856 if (proxy->priv->type != PROXY_TYPE_SYSTEM)
859 ep_set_proxy (proxy, FALSE);
863 ep_sys_proxy_socks_changed_cb (GSettings *settings,
867 g_return_if_fail (proxy != NULL);
869 if (proxy->priv->type != PROXY_TYPE_SYSTEM)
872 ep_set_proxy (proxy, FALSE);
876 e_proxy_dispose (GObject *object)
881 proxy = E_PROXY (object);
884 if (priv->evolution_proxy_settings) {
885 g_signal_handlers_disconnect_by_func (priv->evolution_proxy_settings, ep_evo_proxy_changed_cb, proxy);
886 g_object_unref (priv->evolution_proxy_settings);
887 priv->evolution_proxy_settings = NULL;
890 if (priv->proxy_settings) {
891 g_signal_handlers_disconnect_by_func (priv->proxy_settings, ep_sys_proxy_changed_cb, proxy);
892 g_object_unref (priv->proxy_settings);
893 priv->proxy_settings = NULL;
896 if (priv->proxy_http_settings) {
897 g_signal_handlers_disconnect_by_func (priv->proxy_http_settings, ep_sys_proxy_http_changed_cb, proxy);
898 g_object_unref (priv->proxy_http_settings);
899 priv->proxy_http_settings = NULL;
902 if (priv->proxy_https_settings) {
903 g_signal_handlers_disconnect_by_func (priv->proxy_https_settings, ep_sys_proxy_https_changed_cb, proxy);
904 g_object_unref (priv->proxy_https_settings);
905 priv->proxy_https_settings = NULL;
908 if (priv->proxy_socks_settings) {
909 g_signal_handlers_disconnect_by_func (priv->proxy_socks_settings, ep_sys_proxy_socks_changed_cb, proxy);
910 g_object_unref (priv->proxy_socks_settings);
911 priv->proxy_socks_settings = NULL;
915 soup_uri_free (priv->uri_http);
918 soup_uri_free (priv->uri_https);
921 soup_uri_free (priv->uri_socks);
923 g_slist_foreach (priv->ign_hosts, (GFunc) g_free, NULL);
924 g_slist_free (priv->ign_hosts);
926 g_slist_foreach (priv->ign_addrs, (GFunc) ep_free_proxy_host_addr, NULL);
927 g_slist_free (priv->ign_addrs);
929 /* Chain up to parent's dispose() method. */
930 G_OBJECT_CLASS (e_proxy_parent_class)->dispose (object);
934 e_proxy_class_init (EProxyClass *class)
936 GObjectClass *object_class;
938 g_type_class_add_private (class, sizeof (EProxyPrivate));
940 object_class = G_OBJECT_CLASS (class);
941 object_class->dispose = e_proxy_dispose;
945 * @proxy: the #EProxy which emitted the signal
947 * Emitted when proxy settings changes.
949 signals[CHANGED] = g_signal_new (
951 G_OBJECT_CLASS_TYPE (object_class),
953 G_STRUCT_OFFSET (EProxyClass, changed),
960 e_proxy_init (EProxy *proxy)
962 proxy->priv = E_PROXY_GET_PRIVATE (proxy);
964 proxy->priv->type = PROXY_TYPE_SYSTEM;
966 proxy->priv->evolution_proxy_settings = g_settings_new ("org.gnome.evolution.shell.network-config");
967 proxy->priv->proxy_settings = g_settings_new ("org.gnome.system.proxy");
968 proxy->priv->proxy_http_settings = g_settings_get_child (proxy->priv->proxy_settings, "http");
969 proxy->priv->proxy_https_settings = g_settings_get_child (proxy->priv->proxy_settings, "https");
970 proxy->priv->proxy_socks_settings = g_settings_get_child (proxy->priv->proxy_settings, "socks");
972 g_signal_connect (proxy->priv->evolution_proxy_settings, "changed", G_CALLBACK (ep_evo_proxy_changed_cb), proxy);
973 g_signal_connect (proxy->priv->proxy_settings, "changed", G_CALLBACK (ep_sys_proxy_changed_cb), proxy);
974 g_signal_connect (proxy->priv->proxy_http_settings, "changed", G_CALLBACK (ep_sys_proxy_http_changed_cb), proxy);
975 g_signal_connect (proxy->priv->proxy_https_settings, "changed", G_CALLBACK (ep_sys_proxy_https_changed_cb), proxy);
976 g_signal_connect (proxy->priv->proxy_socks_settings, "changed", G_CALLBACK (ep_sys_proxy_socks_changed_cb), proxy);
987 return g_object_new (E_TYPE_PROXY, NULL);
991 * e_proxy_setup_proxy:
996 e_proxy_setup_proxy (EProxy *proxy)
998 g_return_if_fail (E_IS_PROXY (proxy));
1000 /* We get the evolution-shell proxy keys here
1001 * set soup up to use the proxy,
1002 * and listen to any changes */
1004 /* XXX Why can't we do this automatically in constructed() ? */
1006 ep_set_proxy (proxy, TRUE);
1010 * e_proxy_peek_uri_for:
1015 e_proxy_peek_uri_for (EProxy *proxy,
1018 SoupURI *res = NULL;
1021 g_return_val_if_fail (E_IS_PROXY (proxy), NULL);
1022 g_return_val_if_fail (uri != NULL, NULL);
1024 soup_uri = soup_uri_new (uri);
1025 if (soup_uri == NULL)
1028 if (soup_uri->scheme == SOUP_URI_SCHEME_HTTP)
1029 res = proxy->priv->uri_http;
1030 else if (soup_uri->scheme == SOUP_URI_SCHEME_HTTPS)
1031 res = proxy->priv->uri_https;
1032 else if (soup_uri->scheme && g_ascii_strcasecmp (soup_uri->scheme, "socks") == 0)
1033 res = proxy->priv->uri_socks;
1035 soup_uri_free (soup_uri);
1041 * e_proxy_require_proxy_for_uri:
1046 e_proxy_require_proxy_for_uri (EProxy *proxy,
1049 SoupURI *soup_uri = NULL;
1050 gboolean need_proxy = FALSE;
1052 g_return_val_if_fail (E_IS_PROXY (proxy), FALSE);
1053 g_return_val_if_fail (uri != NULL, FALSE);
1055 if (!proxy->priv->use_proxy || proxy->priv->type == PROXY_TYPE_NO_PROXY) {
1056 d (g_print ("[%s] don't need a proxy to connect to internet\n", uri));
1060 soup_uri = soup_uri_new (uri);
1061 if (soup_uri == NULL)
1064 if (soup_uri->scheme == SOUP_URI_SCHEME_HTTP)
1065 need_proxy = ep_need_proxy_http (proxy, soup_uri->host);
1066 else if (soup_uri->scheme == SOUP_URI_SCHEME_HTTPS)
1067 need_proxy = ep_need_proxy_https (proxy, soup_uri->host);
1068 else if (soup_uri->scheme && g_ascii_strcasecmp (soup_uri->scheme, "socks") == 0)
1069 need_proxy = ep_need_proxy_socks (proxy, soup_uri->host);
1071 soup_uri_free (soup_uri);