mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 08:22:16 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1571 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1571 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GLib testing framework examples and tests
 | |
|  *
 | |
|  * Copyright 2012 Red Hat, Inc.
 | |
|  *
 | |
|  * SPDX-License-Identifier: LGPL-2.1-or-later
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Lesser General Public
 | |
|  * License as published by the Free Software Foundation; either
 | |
|  * version 2.1 of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This library is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|  * Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General
 | |
|  * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #include <string.h>
 | |
| 
 | |
| #include <gio/gio.h>
 | |
| 
 | |
| /* Overview:
 | |
|  *
 | |
|  * We have an echo server, two proxy servers, two GProxy
 | |
|  * implementations, and two GProxyResolver implementations.
 | |
|  *
 | |
|  * The echo server runs at @server.server_addr (on
 | |
|  * @server.server_port).
 | |
|  *
 | |
|  * The two proxy servers, A and B, run on @proxy_a.port and
 | |
|  * @proxy_b.port, with @proxy_a.uri and @proxy_b.uri pointing to them.
 | |
|  * The "negotiation" with the two proxies is just sending the single
 | |
|  * letter "a" or "b" and receiving it back in uppercase; the proxy
 | |
|  * then connects to @server_addr.
 | |
|  *
 | |
|  * Proxy A supports "alpha://" URIs, and does not support hostname
 | |
|  * resolution, and Proxy B supports "beta://" URIs, and does support
 | |
|  * hostname resolution (but it just ignores the hostname and always
 | |
|  * connects to @server_addr anyway).
 | |
|  *
 | |
|  * The default GProxyResolver (GTestProxyResolver) looks at its URI
 | |
|  * and returns [ "direct://" ] for "simple://" URIs, and
 | |
|  * [ proxy_a.uri, proxy_b.uri ] for most other URIs. It can also return
 | |
|  * invalid results for other URIs (empty://, invalid://,
 | |
|  * invalid-then-simple://, and simple-then-invalid://) to test error
 | |
|  * handling.
 | |
|  *
 | |
|  * The other GProxyResolver (GTestAltProxyResolver) always returns
 | |
|  * [ proxy_a.uri ].
 | |
|  */
 | |
| 
 | |
| typedef struct {
 | |
|   gchar *proxy_command;
 | |
|   gchar *supported_protocol;
 | |
| 
 | |
|   GSocket *server;
 | |
|   GThread *thread;
 | |
|   GCancellable *cancellable;
 | |
|   gchar *uri;
 | |
|   gushort port;
 | |
| 
 | |
|   GSocket *client_sock, *server_sock;
 | |
|   GMainLoop *loop;
 | |
| 
 | |
|   GError *last_error;
 | |
| } ProxyData;
 | |
| 
 | |
| static ProxyData proxy_a, proxy_b;
 | |
| 
 | |
| typedef struct {
 | |
|   GSocket *server;
 | |
|   GThread *server_thread;
 | |
|   GCancellable *cancellable;
 | |
|   GSocketAddress *server_addr;
 | |
|   gushort server_port;
 | |
| } ServerData;
 | |
| 
 | |
| static ServerData server;
 | |
| 
 | |
| static gchar **last_proxies;
 | |
| 
 | |
| static GSocketClient *client;
 | |
| 
 | |
| 
 | |
| /**************************************/
 | |
| /* Test GProxyResolver implementation */
 | |
| /**************************************/
 | |
| 
 | |
| typedef struct {
 | |
|   GObject parent_instance;
 | |
| } GTestProxyResolver;
 | |
| 
 | |
| typedef struct {
 | |
|   GObjectClass parent_class;
 | |
| } GTestProxyResolverClass;
 | |
| 
 | |
| static void g_test_proxy_resolver_iface_init (GProxyResolverInterface *iface);
 | |
| 
 | |
| static GType _g_test_proxy_resolver_get_type (void);
 | |
| #define g_test_proxy_resolver_get_type _g_test_proxy_resolver_get_type
 | |
| G_DEFINE_TYPE_WITH_CODE (GTestProxyResolver, g_test_proxy_resolver, G_TYPE_OBJECT,
 | |
| 			 G_IMPLEMENT_INTERFACE (G_TYPE_PROXY_RESOLVER,
 | |
| 						g_test_proxy_resolver_iface_init)
 | |
| 			 g_io_extension_point_implement (G_PROXY_RESOLVER_EXTENSION_POINT_NAME,
 | |
| 							 g_define_type_id,
 | |
| 							 "test",
 | |
| 							 0))
 | |
| 
 | |
| static void
 | |
| g_test_proxy_resolver_init (GTestProxyResolver *resolver)
 | |
| {
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| g_test_proxy_resolver_is_supported (GProxyResolver *resolver)
 | |
| {
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static gchar **
 | |
| g_test_proxy_resolver_lookup (GProxyResolver  *resolver,
 | |
| 			      const gchar     *uri,
 | |
| 			      GCancellable    *cancellable,
 | |
| 			      GError         **error)
 | |
| {
 | |
|   gchar **proxies;
 | |
| 
 | |
|   g_assert (last_proxies == NULL);
 | |
| 
 | |
|   if (g_cancellable_set_error_if_cancelled (cancellable, error))
 | |
|     return NULL;
 | |
| 
 | |
|   proxies = g_new (gchar *, 3);
 | |
| 
 | |
|   if (g_str_has_prefix (uri, "simple://"))
 | |
|     {
 | |
|       proxies[0] = g_strdup ("direct://");
 | |
|       proxies[1] = NULL;
 | |
|     }
 | |
|   else if (g_str_has_prefix (uri, "empty://"))
 | |
|     {
 | |
|       proxies[0] = g_strdup ("");
 | |
|       proxies[1] = NULL;
 | |
|     }
 | |
|   else if (g_str_has_prefix (uri, "invalid://"))
 | |
|     {
 | |
|       proxies[0] = g_strdup ("😼");
 | |
|       proxies[1] = NULL;
 | |
|     }
 | |
|   else if (g_str_has_prefix (uri, "invalid-then-simple://"))
 | |
|     {
 | |
|       proxies[0] = g_strdup ("😼");
 | |
|       proxies[1] = g_strdup ("direct://");
 | |
|       proxies[2] = NULL;
 | |
|     }
 | |
|   else if (g_str_has_prefix (uri, "simple-then-invalid://"))
 | |
|     {
 | |
|       proxies[0] = g_strdup ("direct://");
 | |
|       proxies[1] = g_strdup ("😼");
 | |
|       proxies[2] = NULL;
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       /* Proxy A can only deal with "alpha://" URIs, not
 | |
|        * "beta://", but we always return both URIs
 | |
|        * anyway so we can test error handling when the first
 | |
|        * fails.
 | |
|        */
 | |
|       proxies[0] = g_strdup (proxy_a.uri);
 | |
|       proxies[1] = g_strdup (proxy_b.uri);
 | |
|       proxies[2] = NULL;
 | |
|     }
 | |
| 
 | |
|   last_proxies = g_strdupv (proxies);
 | |
| 
 | |
|   return proxies;
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_test_proxy_resolver_lookup_async (GProxyResolver      *resolver,
 | |
| 				    const gchar         *uri,
 | |
| 				    GCancellable        *cancellable,
 | |
| 				    GAsyncReadyCallback  callback,
 | |
| 				    gpointer             user_data)
 | |
| {
 | |
|   GError *error = NULL;
 | |
|   GTask *task;
 | |
|   gchar **proxies;
 | |
| 
 | |
|   proxies = g_proxy_resolver_lookup (resolver, uri, cancellable, &error);
 | |
| 
 | |
|   task = g_task_new (resolver, NULL, callback, user_data);
 | |
|   g_task_set_source_tag (task, g_test_proxy_resolver_lookup_async);
 | |
|   if (proxies == NULL)
 | |
|     g_task_return_error (task, error);
 | |
|   else
 | |
|     g_task_return_pointer (task, proxies, (GDestroyNotify) g_strfreev);
 | |
| 
 | |
|   g_object_unref (task);
 | |
| }
 | |
| 
 | |
| static gchar **
 | |
| g_test_proxy_resolver_lookup_finish (GProxyResolver     *resolver,
 | |
| 				     GAsyncResult       *result,
 | |
| 				     GError            **error)
 | |
| {
 | |
|   g_assert_true (g_task_is_valid (result, resolver));
 | |
|   g_assert_true (g_task_get_source_tag (G_TASK (result)) == g_test_proxy_resolver_lookup_async);
 | |
| 
 | |
|   return g_task_propagate_pointer (G_TASK (result), error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_test_proxy_resolver_class_init (GTestProxyResolverClass *resolver_class)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_test_proxy_resolver_iface_init (GProxyResolverInterface *iface)
 | |
| {
 | |
|   iface->is_supported = g_test_proxy_resolver_is_supported;
 | |
|   iface->lookup = g_test_proxy_resolver_lookup;
 | |
|   iface->lookup_async = g_test_proxy_resolver_lookup_async;
 | |
|   iface->lookup_finish = g_test_proxy_resolver_lookup_finish;
 | |
| }
 | |
| 
 | |
| /****************************/
 | |
| /* Alternate GProxyResolver */
 | |
| /****************************/
 | |
| 
 | |
| typedef GTestProxyResolver GTestAltProxyResolver;
 | |
| typedef GTestProxyResolverClass GTestAltProxyResolverClass;
 | |
| 
 | |
| static void g_test_alt_proxy_resolver_iface_init (GProxyResolverInterface *iface);
 | |
| 
 | |
| static GType _g_test_alt_proxy_resolver_get_type (void);
 | |
| #define g_test_alt_proxy_resolver_get_type _g_test_alt_proxy_resolver_get_type
 | |
| G_DEFINE_TYPE_WITH_CODE (GTestAltProxyResolver, g_test_alt_proxy_resolver, g_test_proxy_resolver_get_type (),
 | |
| 			 G_IMPLEMENT_INTERFACE (G_TYPE_PROXY_RESOLVER,
 | |
| 						g_test_alt_proxy_resolver_iface_init);
 | |
|                          )
 | |
| 
 | |
| static void
 | |
| g_test_alt_proxy_resolver_init (GTestProxyResolver *resolver)
 | |
| {
 | |
| }
 | |
| 
 | |
| static gchar **
 | |
| g_test_alt_proxy_resolver_lookup (GProxyResolver  *resolver,
 | |
|                                   const gchar     *uri,
 | |
|                                   GCancellable    *cancellable,
 | |
|                                   GError         **error)
 | |
| {
 | |
|   gchar **proxies;
 | |
| 
 | |
|   proxies = g_new (gchar *, 2);
 | |
| 
 | |
|   proxies[0] = g_strdup (proxy_a.uri);
 | |
|   proxies[1] = NULL;
 | |
| 
 | |
|   last_proxies = g_strdupv (proxies);
 | |
| 
 | |
|   return proxies;
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_test_alt_proxy_resolver_class_init (GTestProxyResolverClass *resolver_class)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_test_alt_proxy_resolver_iface_init (GProxyResolverInterface *iface)
 | |
| {
 | |
|   iface->lookup = g_test_alt_proxy_resolver_lookup;
 | |
| }
 | |
| 
 | |
| 
 | |
| /****************************************/
 | |
| /* Test proxy implementation base class */
 | |
| /****************************************/
 | |
| 
 | |
| typedef struct {
 | |
|   GObject parent;
 | |
| 
 | |
|   ProxyData *proxy_data;
 | |
| } GProxyBase;
 | |
| 
 | |
| typedef struct {
 | |
|   GObjectClass parent_class;
 | |
| } GProxyBaseClass;
 | |
| 
 | |
| static GType _g_proxy_base_get_type (void);
 | |
| #define g_proxy_base_get_type _g_proxy_base_get_type
 | |
| G_DEFINE_ABSTRACT_TYPE (GProxyBase, g_proxy_base, G_TYPE_OBJECT)
 | |
| 
 | |
| static void
 | |
| g_proxy_base_init (GProxyBase *proxy)
 | |
| {
 | |
| }
 | |
| 
 | |
| static GIOStream *
 | |
| g_proxy_base_connect (GProxy            *proxy,
 | |
| 		      GIOStream         *io_stream,
 | |
| 		      GProxyAddress     *proxy_address,
 | |
| 		      GCancellable      *cancellable,
 | |
| 		      GError           **error)
 | |
| {
 | |
|   ProxyData *data = ((GProxyBase *) proxy)->proxy_data;
 | |
|   const gchar *protocol;
 | |
|   GOutputStream *ostream;
 | |
|   GInputStream *istream;
 | |
|   gchar response;
 | |
| 
 | |
|   g_assert_no_error (data->last_error);
 | |
| 
 | |
|   protocol = g_proxy_address_get_destination_protocol (proxy_address);
 | |
|   if (strcmp (protocol, data->supported_protocol) != 0)
 | |
|     {
 | |
|       g_set_error_literal (&data->last_error,
 | |
| 			   G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
 | |
| 			   "Unsupported protocol");
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   ostream = g_io_stream_get_output_stream (io_stream);
 | |
|   if (g_output_stream_write (ostream, data->proxy_command, 1, cancellable,
 | |
| 			     &data->last_error) != 1)
 | |
|     goto fail;
 | |
| 
 | |
|   istream = g_io_stream_get_input_stream (io_stream);
 | |
|   if (g_input_stream_read (istream, &response, 1, cancellable,
 | |
| 			   &data->last_error) != 1)
 | |
|     goto fail;
 | |
| 
 | |
|   if (response != g_ascii_toupper (*data->proxy_command))
 | |
|     {
 | |
|       g_set_error_literal (&data->last_error,
 | |
| 			   G_IO_ERROR, G_IO_ERROR_FAILED,
 | |
| 			   "Failed");
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   return g_object_ref (io_stream);
 | |
| 
 | |
|  fail:
 | |
|   g_propagate_error (error, g_error_copy (data->last_error));
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_base_connect_async (GProxy               *proxy,
 | |
| 			    GIOStream            *io_stream,
 | |
| 			    GProxyAddress        *proxy_address,
 | |
| 			    GCancellable         *cancellable,
 | |
| 			    GAsyncReadyCallback   callback,
 | |
| 			    gpointer              user_data)
 | |
| {
 | |
|   GError *error = NULL;
 | |
|   GTask *task;
 | |
|   GIOStream *proxy_io_stream;
 | |
| 
 | |
|   task = g_task_new (proxy, NULL, callback, user_data);
 | |
| 
 | |
|   proxy_io_stream = g_proxy_connect (proxy, io_stream, proxy_address,
 | |
| 				     cancellable, &error);
 | |
|   if (proxy_io_stream)
 | |
|     g_task_return_pointer (task, proxy_io_stream, g_object_unref);
 | |
|   else
 | |
|     g_task_return_error (task, error);
 | |
|   g_object_unref (task);
 | |
| }
 | |
| 
 | |
| static GIOStream *
 | |
| g_proxy_base_connect_finish (GProxy        *proxy,
 | |
| 			     GAsyncResult  *result,
 | |
| 			     GError       **error)
 | |
| {
 | |
|   return g_task_propagate_pointer (G_TASK (result), error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_base_class_init (GProxyBaseClass *class)
 | |
| {
 | |
| }
 | |
| 
 | |
| 
 | |
| /********************************************/
 | |
| /* Test proxy implementation #1 ("Proxy A") */
 | |
| /********************************************/
 | |
| 
 | |
| typedef GProxyBase GProxyA;
 | |
| typedef GProxyBaseClass GProxyAClass;
 | |
| 
 | |
| static void g_proxy_a_iface_init (GProxyInterface *proxy_iface);
 | |
| 
 | |
| static GType _g_proxy_a_get_type (void);
 | |
| #define g_proxy_a_get_type _g_proxy_a_get_type
 | |
| G_DEFINE_TYPE_WITH_CODE (GProxyA, g_proxy_a, g_proxy_base_get_type (),
 | |
| 			 G_IMPLEMENT_INTERFACE (G_TYPE_PROXY,
 | |
| 						g_proxy_a_iface_init)
 | |
| 			 g_io_extension_point_implement (G_PROXY_EXTENSION_POINT_NAME,
 | |
| 							 g_define_type_id,
 | |
| 							 "proxy-a",
 | |
| 							 0))
 | |
| 
 | |
| static void
 | |
| g_proxy_a_init (GProxyA *proxy)
 | |
| {
 | |
|   ((GProxyBase *) proxy)->proxy_data = &proxy_a;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| g_proxy_a_supports_hostname (GProxy *proxy)
 | |
| {
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_a_class_init (GProxyAClass *class)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_a_iface_init (GProxyInterface *proxy_iface)
 | |
| {
 | |
|   proxy_iface->connect = g_proxy_base_connect;
 | |
|   proxy_iface->connect_async = g_proxy_base_connect_async;
 | |
|   proxy_iface->connect_finish = g_proxy_base_connect_finish;
 | |
|   proxy_iface->supports_hostname = g_proxy_a_supports_hostname;
 | |
| }
 | |
| 
 | |
| /********************************************/
 | |
| /* Test proxy implementation #2 ("Proxy B") */
 | |
| /********************************************/
 | |
| 
 | |
| typedef GProxyBase GProxyB;
 | |
| typedef GProxyBaseClass GProxyBClass;
 | |
| 
 | |
| static void g_proxy_b_iface_init (GProxyInterface *proxy_iface);
 | |
| 
 | |
| static GType _g_proxy_b_get_type (void);
 | |
| #define g_proxy_b_get_type _g_proxy_b_get_type
 | |
| G_DEFINE_TYPE_WITH_CODE (GProxyB, g_proxy_b, g_proxy_base_get_type (),
 | |
| 			 G_IMPLEMENT_INTERFACE (G_TYPE_PROXY,
 | |
| 						g_proxy_b_iface_init)
 | |
| 			 g_io_extension_point_implement (G_PROXY_EXTENSION_POINT_NAME,
 | |
| 							 g_define_type_id,
 | |
| 							 "proxy-b",
 | |
| 							 0))
 | |
| 
 | |
| static void
 | |
| g_proxy_b_init (GProxyB *proxy)
 | |
| {
 | |
|   ((GProxyBase *) proxy)->proxy_data = &proxy_b;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| g_proxy_b_supports_hostname (GProxy *proxy)
 | |
| {
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_b_class_init (GProxyBClass *class)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_proxy_b_iface_init (GProxyInterface *proxy_iface)
 | |
| {
 | |
|   proxy_iface->connect = g_proxy_base_connect;
 | |
|   proxy_iface->connect_async = g_proxy_base_connect_async;
 | |
|   proxy_iface->connect_finish = g_proxy_base_connect_finish;
 | |
|   proxy_iface->supports_hostname = g_proxy_b_supports_hostname;
 | |
| }
 | |
| 
 | |
| 
 | |
| /***********************************/
 | |
| /* The proxy server implementation */
 | |
| /***********************************/
 | |
| 
 | |
| static gboolean
 | |
| proxy_bytes (GSocket      *socket,
 | |
| 	     GIOCondition  condition,
 | |
| 	     gpointer      user_data)
 | |
| {
 | |
|   ProxyData *proxy = user_data;
 | |
|   gssize nread, nwrote, total;
 | |
|   gchar buffer[8];
 | |
|   GSocket *out_socket;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   nread = g_socket_receive_with_blocking (socket, buffer, sizeof (buffer),
 | |
| 					  TRUE, NULL, &error);
 | |
|   if (nread == -1)
 | |
|     {
 | |
|       g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CLOSED);
 | |
|       return FALSE;
 | |
|     }
 | |
|   else
 | |
|     g_assert_no_error (error);
 | |
| 
 | |
|   if (nread == 0)
 | |
|     {
 | |
|       g_main_loop_quit (proxy->loop);
 | |
|       return FALSE;
 | |
|     }
 | |
| 
 | |
|   if (socket == proxy->client_sock)
 | |
|     out_socket = proxy->server_sock;
 | |
|   else
 | |
|     out_socket = proxy->client_sock;
 | |
| 
 | |
|   for (total = 0; total < nread; total += nwrote)
 | |
|     {
 | |
|       nwrote = g_socket_send_with_blocking (out_socket,
 | |
| 					    buffer + total, nread - total,
 | |
| 					    TRUE, NULL, &error);
 | |
|       g_assert_no_error (error);
 | |
|     }
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static gpointer
 | |
| proxy_thread (gpointer user_data)
 | |
| {
 | |
|   ProxyData *proxy = user_data;
 | |
|   GError *error = NULL;
 | |
|   gssize nread, nwrote;
 | |
|   gchar command[2] = { 0, 0 };
 | |
|   GMainContext *context;
 | |
|   GSource *read_source, *write_source;
 | |
| 
 | |
|   context = g_main_context_new ();
 | |
|   proxy->loop = g_main_loop_new (context, FALSE);
 | |
| 
 | |
|   while (TRUE)
 | |
|     {
 | |
|       proxy->client_sock = g_socket_accept (proxy->server, proxy->cancellable, &error);
 | |
|       if (!proxy->client_sock)
 | |
| 	{
 | |
| 	  g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
 | |
|           g_error_free (error);
 | |
| 	  break;
 | |
| 	}
 | |
|       else
 | |
| 	g_assert_no_error (error);
 | |
| 
 | |
|       nread = g_socket_receive (proxy->client_sock, command, 1, NULL, &error);
 | |
|       g_assert_no_error (error);
 | |
| 
 | |
|       if (nread == 0)
 | |
| 	{
 | |
| 	  g_clear_object (&proxy->client_sock);
 | |
| 	  continue;
 | |
| 	}
 | |
| 
 | |
|       g_assert_cmpint (nread, ==, 1);
 | |
|       g_assert_cmpstr (command, ==, proxy->proxy_command);
 | |
| 
 | |
|       *command = g_ascii_toupper (*command);
 | |
|       nwrote = g_socket_send (proxy->client_sock, command, 1, NULL, &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_assert_cmpint (nwrote, ==, 1);
 | |
| 
 | |
|       proxy->server_sock = g_socket_new (G_SOCKET_FAMILY_IPV4,
 | |
| 					 G_SOCKET_TYPE_STREAM,
 | |
| 					 G_SOCKET_PROTOCOL_DEFAULT,
 | |
| 					 &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_socket_connect (proxy->server_sock, server.server_addr, NULL, &error);
 | |
|       g_assert_no_error (error);
 | |
| 
 | |
|       read_source = g_socket_create_source (proxy->client_sock, G_IO_IN, NULL);
 | |
|       g_source_set_callback (read_source, (GSourceFunc)proxy_bytes, proxy, NULL);
 | |
|       g_source_attach (read_source, context);
 | |
| 
 | |
|       write_source = g_socket_create_source (proxy->server_sock, G_IO_IN, NULL);
 | |
|       g_source_set_callback (write_source, (GSourceFunc)proxy_bytes, proxy, NULL);
 | |
|       g_source_attach (write_source, context);
 | |
| 
 | |
|       g_main_loop_run (proxy->loop);
 | |
| 
 | |
|       g_socket_close (proxy->client_sock, &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_clear_object (&proxy->client_sock);
 | |
| 
 | |
|       g_socket_close (proxy->server_sock, &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_clear_object (&proxy->server_sock);
 | |
| 
 | |
|       g_source_destroy (read_source);
 | |
|       g_source_unref (read_source);
 | |
|       g_source_destroy (write_source);
 | |
|       g_source_unref (write_source);
 | |
|     }
 | |
| 
 | |
|   g_main_loop_unref (proxy->loop);
 | |
|   g_main_context_unref (context);
 | |
| 
 | |
|   g_object_unref (proxy->server);
 | |
|   g_object_unref (proxy->cancellable);
 | |
| 
 | |
|   g_free (proxy->proxy_command);
 | |
|   g_free (proxy->supported_protocol);
 | |
|   g_free (proxy->uri);
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| create_proxy (ProxyData    *proxy,
 | |
| 	      gchar         proxy_protocol,
 | |
| 	      const gchar  *destination_protocol,
 | |
| 	      GCancellable *cancellable)
 | |
| {
 | |
|   GError *error = NULL;
 | |
|   GSocketAddress *addr;
 | |
|   GInetAddress *iaddr;
 | |
| 
 | |
|   proxy->proxy_command = g_strdup_printf ("%c", proxy_protocol);
 | |
|   proxy->supported_protocol = g_strdup (destination_protocol);
 | |
|   proxy->cancellable = g_object_ref (cancellable);
 | |
| 
 | |
|   proxy->server = g_socket_new (G_SOCKET_FAMILY_IPV4,
 | |
| 				G_SOCKET_TYPE_STREAM,
 | |
| 				G_SOCKET_PROTOCOL_DEFAULT,
 | |
| 				&error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   iaddr = g_inet_address_new_loopback (G_SOCKET_FAMILY_IPV4);
 | |
|   addr = g_inet_socket_address_new (iaddr, 0);
 | |
|   g_object_unref (iaddr);
 | |
| 
 | |
|   g_socket_bind (proxy->server, addr, TRUE, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_object_unref (addr);
 | |
| 
 | |
|   addr = g_socket_get_local_address (proxy->server, &error);
 | |
|   proxy->port = g_inet_socket_address_get_port (G_INET_SOCKET_ADDRESS (addr));
 | |
|   proxy->uri = g_strdup_printf ("proxy-%c://127.0.0.1:%u",
 | |
| 				g_ascii_tolower (proxy_protocol),
 | |
| 				proxy->port);
 | |
|   g_object_unref (addr);
 | |
| 
 | |
|   g_socket_listen (proxy->server, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   proxy->thread = g_thread_new ("proxy", proxy_thread, proxy);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /**************************/
 | |
| /* The actual echo server */
 | |
| /**************************/
 | |
| 
 | |
| static gpointer
 | |
| echo_server_thread (gpointer user_data)
 | |
| {
 | |
|   ServerData *data = user_data;
 | |
|   GSocket *sock;
 | |
|   GError *error = NULL;
 | |
|   gssize nread, nwrote;
 | |
|   gchar buf[128];
 | |
| 
 | |
|   while (TRUE)
 | |
|     {
 | |
|       sock = g_socket_accept (data->server, data->cancellable, &error);
 | |
|       if (!sock)
 | |
| 	{
 | |
| 	  g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
 | |
|           g_error_free (error);
 | |
| 	  break;
 | |
| 	}
 | |
|       else
 | |
| 	g_assert_no_error (error);
 | |
| 
 | |
|       while (TRUE)
 | |
| 	{
 | |
| 	  nread = g_socket_receive (sock, buf, sizeof (buf), NULL, &error);
 | |
| 	  g_assert_no_error (error);
 | |
| 	  g_assert_cmpint (nread, >=, 0);
 | |
| 
 | |
| 	  if (nread == 0)
 | |
| 	    break;
 | |
| 
 | |
| 	  nwrote = g_socket_send (sock, buf, nread, NULL, &error);
 | |
| 	  g_assert_no_error (error);
 | |
| 	  g_assert_cmpint (nwrote, ==, nread);
 | |
| 	}
 | |
| 
 | |
|       g_socket_close (sock, &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_object_unref (sock);
 | |
|     }
 | |
| 
 | |
|   g_object_unref (data->server);
 | |
|   g_object_unref (data->server_addr);
 | |
|   g_object_unref (data->cancellable);
 | |
| 
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| static void
 | |
| create_server (ServerData *data, GCancellable *cancellable)
 | |
| {
 | |
|   GError *error = NULL;
 | |
|   GSocketAddress *addr;
 | |
|   GInetAddress *iaddr;
 | |
| 
 | |
|   data->cancellable = g_object_ref (cancellable);
 | |
| 
 | |
|   data->server = g_socket_new (G_SOCKET_FAMILY_IPV4,
 | |
| 			       G_SOCKET_TYPE_STREAM,
 | |
| 			       G_SOCKET_PROTOCOL_DEFAULT,
 | |
| 			       &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   g_socket_set_blocking (data->server, TRUE);
 | |
|   iaddr = g_inet_address_new_loopback (G_SOCKET_FAMILY_IPV4);
 | |
|   addr = g_inet_socket_address_new (iaddr, 0);
 | |
|   g_object_unref (iaddr);
 | |
| 
 | |
|   g_socket_bind (data->server, addr, TRUE, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_object_unref (addr);
 | |
| 
 | |
|   data->server_addr = g_socket_get_local_address (data->server, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   data->server_port = g_inet_socket_address_get_port (G_INET_SOCKET_ADDRESS (data->server_addr));
 | |
| 
 | |
|   g_socket_listen (data->server, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   data->server_thread = g_thread_new ("server", echo_server_thread, data);
 | |
| }
 | |
| 
 | |
| 
 | |
| /******************************************************************/
 | |
| /* Now a GResolver implementation, so the can't-resolve test will */
 | |
| /* pass even if you have an evil DNS-faking ISP.                  */
 | |
| /******************************************************************/
 | |
| 
 | |
| typedef GResolver GFakeResolver;
 | |
| typedef GResolverClass GFakeResolverClass;
 | |
| 
 | |
| static GType g_fake_resolver_get_type (void);
 | |
| G_DEFINE_TYPE (GFakeResolver, g_fake_resolver, G_TYPE_RESOLVER)
 | |
| 
 | |
| static void
 | |
| g_fake_resolver_init (GFakeResolver *gtr)
 | |
| {
 | |
| }
 | |
| 
 | |
| static GList *
 | |
| g_fake_resolver_lookup_by_name (GResolver     *resolver,
 | |
| 				const gchar   *hostname,
 | |
| 				GCancellable  *cancellable,
 | |
| 				GError       **error)
 | |
| {
 | |
|   if (!strcmp (hostname, "example.com"))
 | |
|     return g_list_prepend (NULL, g_inet_address_new_from_string ("127.0.0.1"));
 | |
|   else
 | |
|     {
 | |
|       /* Anything else is expected to fail. */
 | |
|       g_set_error (error,
 | |
|                    G_RESOLVER_ERROR,
 | |
|                    G_RESOLVER_ERROR_NOT_FOUND,
 | |
|                    "Not found");
 | |
|       return NULL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_fake_resolver_lookup_by_name_async (GResolver           *resolver,
 | |
| 				      const gchar         *hostname,
 | |
| 				      GCancellable        *cancellable,
 | |
| 				      GAsyncReadyCallback  callback,
 | |
| 				      gpointer             user_data)
 | |
| {
 | |
|   GTask *task;
 | |
| 
 | |
|   task = g_task_new (resolver, cancellable, callback, user_data);
 | |
| 
 | |
|   if (!strcmp (hostname, "example.com"))
 | |
|     {
 | |
|       GList *result;
 | |
| 
 | |
|       result = g_list_prepend (NULL, g_inet_address_new_from_string ("127.0.0.1"));
 | |
|       g_task_return_pointer (task, result, (GDestroyNotify) g_resolver_free_addresses);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       g_task_return_new_error_literal (task,
 | |
|                                        G_RESOLVER_ERROR, G_RESOLVER_ERROR_NOT_FOUND,
 | |
|                                        "Not found");
 | |
|     }
 | |
|   g_object_unref (task);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_fake_resolver_lookup_by_name_with_flags_async (GResolver               *resolver,
 | |
|                                                  const gchar             *hostname,
 | |
|                                                  GResolverNameLookupFlags flags,
 | |
|                                                  GCancellable            *cancellable,
 | |
|                                                  GAsyncReadyCallback      callback,
 | |
|                                                  gpointer                 user_data)
 | |
| {
 | |
|   /* Note this isn't a real implementation as it ignores the flags */
 | |
|   g_fake_resolver_lookup_by_name_async (resolver,
 | |
|                                         hostname,
 | |
|                                         cancellable,
 | |
|                                         callback,
 | |
|                                         user_data);
 | |
| }
 | |
| 
 | |
| static GList *
 | |
| g_fake_resolver_lookup_by_name_finish (GResolver            *resolver,
 | |
| 				       GAsyncResult         *result,
 | |
| 				       GError              **error)
 | |
| {
 | |
|   return g_task_propagate_pointer (G_TASK (result), error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| g_fake_resolver_class_init (GFakeResolverClass *fake_class)
 | |
| {
 | |
|   GResolverClass *resolver_class = G_RESOLVER_CLASS (fake_class);
 | |
| 
 | |
|   resolver_class->lookup_by_name                   = g_fake_resolver_lookup_by_name;
 | |
|   resolver_class->lookup_by_name_async             = g_fake_resolver_lookup_by_name_async;
 | |
|   resolver_class->lookup_by_name_finish            = g_fake_resolver_lookup_by_name_finish;
 | |
|   resolver_class->lookup_by_name_with_flags_async  = g_fake_resolver_lookup_by_name_with_flags_async;
 | |
|   resolver_class->lookup_by_name_with_flags_finish = g_fake_resolver_lookup_by_name_finish;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************/
 | |
| /* We made it! Now for the actual test! */
 | |
| /****************************************/
 | |
| 
 | |
| static void
 | |
| setup_test (gpointer fixture,
 | |
| 	    gconstpointer user_data)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void
 | |
| teardown_test (gpointer fixture,
 | |
| 	       gconstpointer user_data)
 | |
| {
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   g_clear_error (&proxy_a.last_error);
 | |
|   g_clear_error (&proxy_b.last_error);
 | |
| }
 | |
| 
 | |
| 
 | |
| static const gchar *testbuf = "0123456789abcdef";
 | |
| 
 | |
| static void
 | |
| do_echo_test (GSocketConnection *conn)
 | |
| {
 | |
|   GIOStream *iostream = G_IO_STREAM (conn);
 | |
|   GInputStream *istream = g_io_stream_get_input_stream (iostream);
 | |
|   GOutputStream *ostream = g_io_stream_get_output_stream (iostream);
 | |
|   gssize nread;
 | |
|   gsize nwrote, total;
 | |
|   gchar buf[128];
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_output_stream_write_all (ostream, testbuf, strlen (testbuf),
 | |
| 			     &nwrote, NULL, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert_cmpint (nwrote, ==, strlen (testbuf));
 | |
| 
 | |
|   for (total = 0; total < nwrote; total += nread)
 | |
|     {
 | |
|       nread = g_input_stream_read (istream,
 | |
| 				   buf + total, sizeof (buf) - total,
 | |
| 				   NULL, &error);
 | |
|       g_assert_no_error (error);
 | |
|       g_assert_cmpint (nread, >, 0);
 | |
|     }
 | |
| 
 | |
|   buf[total] = '\0';
 | |
|   g_assert_cmpstr (buf, ==, testbuf);
 | |
| }
 | |
| 
 | |
| static void
 | |
| async_got_conn (GObject      *source,
 | |
| 		GAsyncResult *result,
 | |
| 		gpointer      user_data)
 | |
| {
 | |
|   GSocketConnection **conn = user_data;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   *conn = g_socket_client_connect_finish (G_SOCKET_CLIENT (source),
 | |
| 					  result, &error);
 | |
|   g_assert_no_error (error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| async_got_error (GObject      *source,
 | |
| 		 GAsyncResult *result,
 | |
| 		 gpointer      user_data)
 | |
| {
 | |
|   GError **error = user_data;
 | |
| 
 | |
|   g_assert (error != NULL && *error == NULL);
 | |
|   g_socket_client_connect_finish (G_SOCKET_CLIENT (source),
 | |
| 				  result, error);
 | |
|   g_assert (*error != NULL);
 | |
| }
 | |
| 
 | |
| static void
 | |
| async_resolver_got_error (GObject      *source,
 | |
|                           GAsyncResult *result,
 | |
|                           gpointer      user_data)
 | |
| {
 | |
|   GError **error = user_data;
 | |
| 
 | |
|   g_assert (error != NULL && *error == NULL);
 | |
|   g_proxy_resolver_lookup_finish (G_PROXY_RESOLVER (source),
 | |
| 				  result, error);
 | |
|   g_assert (*error != NULL);
 | |
| }
 | |
| 
 | |
| static void
 | |
| assert_direct (GSocketConnection *conn)
 | |
| {
 | |
|   GSocketAddress *addr;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_assert_cmpint (g_strv_length (last_proxies), ==, 1);
 | |
|   g_assert_cmpstr (last_proxies[0], ==, "direct://");
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
| 
 | |
|   addr = g_socket_connection_get_remote_address (conn, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert (addr != NULL && !G_IS_PROXY_ADDRESS (addr));
 | |
|   g_object_unref (addr);
 | |
| 
 | |
|   addr = g_socket_connection_get_local_address (conn, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_object_unref (addr);
 | |
| 
 | |
|   g_assert (g_socket_connection_is_connected (conn));
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_direct_sync (gpointer fixture,
 | |
| 		  gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   gchar *uri;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   /* The simple:// URI should not require any proxy. */
 | |
| 
 | |
|   uri = g_strdup_printf ("simple://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   assert_direct (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_direct_async (gpointer fixture,
 | |
| 		   gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The simple:// URI should not require any proxy. */
 | |
|   uri = g_strdup_printf ("simple://127.0.0.1:%u", server.server_port);
 | |
|   conn = NULL;
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   g_free (uri);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
| 
 | |
|   assert_direct (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| assert_single (GSocketConnection *conn)
 | |
| {
 | |
|   GSocketAddress *addr;
 | |
|   const gchar *proxy_uri;
 | |
|   gushort proxy_port;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_assert_cmpint (g_strv_length (last_proxies), ==, 2);
 | |
|   g_assert_cmpstr (last_proxies[0], ==, proxy_a.uri);
 | |
|   g_assert_cmpstr (last_proxies[1], ==, proxy_b.uri);
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
| 
 | |
|   addr = g_socket_connection_get_remote_address (conn, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert (G_IS_PROXY_ADDRESS (addr));
 | |
|   proxy_uri = g_proxy_address_get_uri (G_PROXY_ADDRESS (addr));
 | |
|   g_assert_cmpstr (proxy_uri, ==, proxy_a.uri);
 | |
|   proxy_port = g_inet_socket_address_get_port (G_INET_SOCKET_ADDRESS (addr));
 | |
|   g_assert_cmpint (proxy_port, ==, proxy_a.port);
 | |
| 
 | |
|   g_object_unref (addr);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_single_sync (gpointer fixture,
 | |
| 		  gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   GError *error = NULL;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The alpha:// URI should be proxied via Proxy A */
 | |
|   uri = g_strdup_printf ("alpha://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   assert_single (conn);
 | |
| 
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_single_async (gpointer fixture,
 | |
| 		   gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The alpha:// URI should be proxied via Proxy A */
 | |
|   uri = g_strdup_printf ("alpha://127.0.0.1:%u", server.server_port);
 | |
|   conn = NULL;
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   g_free (uri);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
| 
 | |
|   assert_single (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| assert_multiple (GSocketConnection *conn)
 | |
| {
 | |
|   GSocketAddress *addr;
 | |
|   const gchar *proxy_uri;
 | |
|   gushort proxy_port;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_assert_cmpint (g_strv_length (last_proxies), ==, 2);
 | |
|   g_assert_cmpstr (last_proxies[0], ==, proxy_a.uri);
 | |
|   g_assert_cmpstr (last_proxies[1], ==, proxy_b.uri);
 | |
|   g_assert_error (proxy_a.last_error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
| 
 | |
|   addr = g_socket_connection_get_remote_address (conn, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert (G_IS_PROXY_ADDRESS (addr));
 | |
|   proxy_uri = g_proxy_address_get_uri (G_PROXY_ADDRESS (addr));
 | |
|   g_assert_cmpstr (proxy_uri, ==, proxy_b.uri);
 | |
|   proxy_port = g_inet_socket_address_get_port (G_INET_SOCKET_ADDRESS (addr));
 | |
|   g_assert_cmpint (proxy_port, ==, proxy_b.port);
 | |
| 
 | |
|   g_object_unref (addr);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_multiple_sync (gpointer fixture,
 | |
| 		    gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   GError *error = NULL;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The beta:// URI should be proxied via Proxy B, after failing
 | |
|    * via Proxy A.
 | |
|    */
 | |
|   uri = g_strdup_printf ("beta://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   assert_multiple (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_multiple_async (gpointer fixture,
 | |
| 		     gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The beta:// URI should be proxied via Proxy B, after failing
 | |
|    * via Proxy A.
 | |
|    */
 | |
|   uri = g_strdup_printf ("beta://127.0.0.1:%u", server.server_port);
 | |
|   conn = NULL;
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   g_free (uri);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
| 
 | |
|   assert_multiple (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_invalid_uris_sync (gpointer fixture,
 | |
| 		        gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   gchar *uri;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   g_test_bug ("https://gitlab.gnome.org/GNOME/glib/-/issues/2597");
 | |
| 
 | |
|   /* The empty:// URI causes the proxy resolver to return an empty string. */
 | |
|   uri = g_strdup_printf ("empty://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
 | |
|   g_assert_null (conn);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* The invalid:// URI causes the proxy resolver to return a cat emoji. */
 | |
|   uri = g_strdup_printf ("invalid://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
 | |
|   g_assert_null (conn);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* If the proxy resolver returns an invalid URI before a valid URI,
 | |
|    * we should succeed.
 | |
|    */
 | |
|   uri = g_strdup_printf ("invalid-then-simple://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_no_error (error);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* If the proxy resolver returns a valid URI before an invalid URI,
 | |
|    * we should succeed.
 | |
|    */
 | |
|   uri = g_strdup_printf ("simple-then-invalid://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_free (uri);
 | |
|   g_assert_no_error (error);
 | |
|   do_echo_test (conn);
 | |
|   g_object_unref (conn);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* Trying to use something that is not a URI at all should fail. */
 | |
|   conn = g_socket_client_connect_to_uri (client, "asdf", 0, NULL, &error);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* Should still fail if using GProxyResolver directly. */
 | |
|   g_proxy_resolver_lookup (g_proxy_resolver_get_default (), "asdf", NULL, &error);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
 | |
|   g_clear_error (&error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_invalid_uris_async (gpointer fixture,
 | |
| 		         gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn = NULL;
 | |
|   GError *error = NULL;
 | |
|   gchar *uri;
 | |
| 
 | |
|   g_test_bug ("https://gitlab.gnome.org/GNOME/glib/-/issues/2597");
 | |
| 
 | |
|   /* The empty:// URI causes the proxy resolver to return an empty string. */
 | |
|   uri = g_strdup_printf ("empty://127.0.0.1:%u", server.server_port);
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   g_free (uri);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* The invalid:// URI causes the proxy resolver to return a cat emoji. */
 | |
|   uri = g_strdup_printf ("invalid://127.0.0.1:%u", server.server_port);
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   g_free (uri);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* If the proxy resolver returns an invalid URI before a valid URI,
 | |
|    * we should succeed.
 | |
|    */
 | |
|   uri = g_strdup_printf ("invalid-then-simple://127.0.0.1:%u", server.server_port);
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   g_free (uri);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* If the proxy resolver returns a valid URI before an invalid URI,
 | |
|    * we should succeed.
 | |
|    */
 | |
|   uri = g_strdup_printf ("simple-then-invalid://127.0.0.1:%u", server.server_port);
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   g_free (uri);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* Trying to use something that is not a URI at all should fail. */
 | |
|   g_socket_client_connect_to_uri_async (client, "asdf", 0, NULL,
 | |
|                                         async_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
 | |
|   g_clear_error (&error);
 | |
|   g_clear_pointer (&last_proxies, g_strfreev);
 | |
| 
 | |
|   /* Should still fail if using GProxyResolver directly. */
 | |
|   g_proxy_resolver_lookup_async (g_proxy_resolver_get_default (), "asdf", NULL,
 | |
|                                  async_resolver_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
 | |
|   g_clear_error (&error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_dns (gpointer fixture,
 | |
| 	  gconstpointer user_data)
 | |
| {
 | |
|   GSocketConnection *conn;
 | |
|   GError *error = NULL;
 | |
|   gchar *uri;
 | |
| 
 | |
|   /* The simple:// and alpha:// URIs should fail with a DNS error,
 | |
|    * but the beta:// URI should succeed, because we pass it to
 | |
|    * Proxy B without trying to resolve it first
 | |
|    */
 | |
| 
 | |
|   /* simple */
 | |
|   uri = g_strdup_printf ("simple://no-such-host.xx:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_assert_error (error, G_RESOLVER_ERROR, G_RESOLVER_ERROR_NOT_FOUND);
 | |
|   g_clear_error (&error);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_RESOLVER_ERROR, G_RESOLVER_ERROR_NOT_FOUND);
 | |
|   g_clear_error (&error);
 | |
|   g_free (uri);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   /* alpha */
 | |
|   uri = g_strdup_printf ("alpha://no-such-host.xx:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   /* Since Proxy A fails, @client will try Proxy B too, which won't
 | |
|    * load an alpha:// URI.
 | |
|    */
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_error (proxy_b.last_error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
|   g_free (uri);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_error (proxy_b.last_error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   /* beta */
 | |
|   uri = g_strdup_printf ("beta://no-such-host.xx:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
| 
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_free (uri);
 | |
| 
 | |
|   g_assert_no_error (proxy_a.last_error);
 | |
|   g_assert_no_error (proxy_b.last_error);
 | |
| 
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   teardown_test (NULL, NULL);
 | |
| }
 | |
| 
 | |
| static void
 | |
| assert_override (GSocketConnection *conn)
 | |
| {
 | |
|   g_assert_cmpint (g_strv_length (last_proxies), ==, 1);
 | |
|   g_assert_cmpstr (last_proxies[0], ==, proxy_a.uri);
 | |
| 
 | |
|   if (conn)
 | |
|     g_assert_no_error (proxy_a.last_error);
 | |
|   else
 | |
|     g_assert_error (proxy_a.last_error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_override (gpointer fixture,
 | |
|                gconstpointer user_data)
 | |
| {
 | |
|   GProxyResolver *alt_resolver;
 | |
|   GSocketConnection *conn;
 | |
|   GError *error = NULL;
 | |
|   gchar *uri;
 | |
| 
 | |
|   g_assert (g_socket_client_get_proxy_resolver (client) == g_proxy_resolver_get_default ());
 | |
|   alt_resolver = g_object_new (g_test_alt_proxy_resolver_get_type (), NULL);
 | |
|   g_socket_client_set_proxy_resolver (client, alt_resolver);
 | |
|   g_assert (g_socket_client_get_proxy_resolver (client) == alt_resolver);
 | |
| 
 | |
|   /* Alt proxy resolver always returns Proxy A, so alpha:// should
 | |
|    * succeed, and simple:// and beta:// should fail.
 | |
|    */
 | |
| 
 | |
|   /* simple */
 | |
|   uri = g_strdup_printf ("simple://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
|   assert_override (conn);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
|   assert_override (conn);
 | |
|   g_free (uri);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   /* alpha */
 | |
|   uri = g_strdup_printf ("alpha://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_assert_no_error (error);
 | |
|   assert_override (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   conn = NULL;
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_conn, &conn);
 | |
|   while (conn == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   assert_override (conn);
 | |
|   do_echo_test (conn);
 | |
|   g_clear_object (&conn);
 | |
|   g_free (uri);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   /* beta */
 | |
|   uri = g_strdup_printf ("beta://127.0.0.1:%u", server.server_port);
 | |
|   conn = g_socket_client_connect_to_uri (client, uri, 0, NULL, &error);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
|   assert_override (conn);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_socket_client_connect_to_uri_async (client, uri, 0, NULL,
 | |
| 					async_got_error, &error);
 | |
|   while (error == NULL)
 | |
|     g_main_context_iteration (NULL, TRUE);
 | |
|   g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
 | |
|   g_clear_error (&error);
 | |
|   assert_override (conn);
 | |
|   g_free (uri);
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   g_assert (g_socket_client_get_proxy_resolver (client) == alt_resolver);
 | |
|   g_socket_client_set_proxy_resolver (client, NULL);
 | |
|   g_assert (g_socket_client_get_proxy_resolver (client) == g_proxy_resolver_get_default ());
 | |
|   g_object_unref (alt_resolver);
 | |
| }
 | |
| 
 | |
| static void
 | |
| assert_destination_port (GSocketAddressEnumerator *etor,
 | |
|                          guint16                   port)
 | |
| {
 | |
|   GSocketAddress *addr;
 | |
|   GProxyAddress *paddr;
 | |
|   GError *error = NULL;
 | |
| 
 | |
|   while ((addr = g_socket_address_enumerator_next (etor, NULL, &error)))
 | |
|     {
 | |
|       g_assert_no_error (error);
 | |
| 
 | |
|       g_assert (G_IS_PROXY_ADDRESS (addr));
 | |
|       paddr = G_PROXY_ADDRESS (addr);
 | |
|       g_assert_cmpint (g_proxy_address_get_destination_port (paddr), ==, port);
 | |
|       g_object_unref (addr);
 | |
|     }
 | |
|   g_assert_no_error (error);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_proxy_enumerator_ports (void)
 | |
| {
 | |
|   GSocketAddressEnumerator *etor;
 | |
| 
 | |
|   etor = g_object_new (G_TYPE_PROXY_ADDRESS_ENUMERATOR,
 | |
|                        "uri", "http://example.com/",
 | |
|                        NULL);
 | |
|   assert_destination_port (etor, 0);
 | |
|   g_object_unref (etor);
 | |
| 
 | |
|   /* Have to call this to clear last_proxies so the next call to
 | |
|    * g_test_proxy_resolver_lookup() won't assert.
 | |
|    */
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   etor = g_object_new (G_TYPE_PROXY_ADDRESS_ENUMERATOR,
 | |
|                        "uri", "http://example.com:8080/",
 | |
|                        NULL);
 | |
|   assert_destination_port (etor, 8080);
 | |
|   g_object_unref (etor);
 | |
| 
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   etor = g_object_new (G_TYPE_PROXY_ADDRESS_ENUMERATOR,
 | |
|                        "uri", "http://example.com/",
 | |
|                        "default-port", 80,
 | |
|                        NULL);
 | |
|   assert_destination_port (etor, 80);
 | |
|   g_object_unref (etor);
 | |
| 
 | |
|   teardown_test (NULL, NULL);
 | |
| 
 | |
|   etor = g_object_new (G_TYPE_PROXY_ADDRESS_ENUMERATOR,
 | |
|                        "uri", "http://example.com:8080/",
 | |
|                        "default-port", 80,
 | |
|                        NULL);
 | |
|   assert_destination_port (etor, 8080);
 | |
|   g_object_unref (etor);
 | |
| 
 | |
|   teardown_test (NULL, NULL);
 | |
| }
 | |
| 
 | |
| int
 | |
| main (int   argc,
 | |
|       char *argv[])
 | |
| {
 | |
|   GResolver *fake_resolver;
 | |
|   GCancellable *cancellable;
 | |
|   gint result;
 | |
| 
 | |
|   g_test_init (&argc, &argv, NULL);
 | |
| 
 | |
|   /* Register stuff. The dummy g_proxy_get_default_for_protocol() call
 | |
|    * is to force _g_io_modules_ensure_extension_points_registered() to
 | |
|    * get called, so we can then register a proxy resolver extension
 | |
|    * point.
 | |
|    */
 | |
|   g_proxy_get_default_for_protocol ("foo");
 | |
|   g_test_proxy_resolver_get_type ();
 | |
|   g_proxy_a_get_type ();
 | |
|   g_proxy_b_get_type ();
 | |
|   g_setenv ("GIO_USE_PROXY_RESOLVER", "test", TRUE);
 | |
| 
 | |
|   fake_resolver = g_object_new (g_fake_resolver_get_type (), NULL);
 | |
|   g_resolver_set_default (fake_resolver);
 | |
| 
 | |
|   cancellable = g_cancellable_new ();
 | |
|   create_server (&server, cancellable);
 | |
|   create_proxy (&proxy_a, 'a', "alpha", cancellable);
 | |
|   create_proxy (&proxy_b, 'b', "beta", cancellable);
 | |
| 
 | |
|   client = g_socket_client_new ();
 | |
|   g_assert_cmpint (g_socket_client_get_enable_proxy (client), ==, TRUE);
 | |
| 
 | |
|   g_test_add_vtable ("/proxy/direct_sync", 0, NULL, setup_test, test_direct_sync, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/direct_async", 0, NULL, setup_test, test_direct_async, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/single_sync", 0, NULL, setup_test, test_single_sync, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/single_async", 0, NULL, setup_test, test_single_async, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/multiple_sync", 0, NULL, setup_test, test_multiple_sync, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/multiple_async", 0, NULL, setup_test, test_multiple_async, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/invalid-uris-sync", 0, NULL, setup_test, test_invalid_uris_sync, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/invalid-uris-async", 0, NULL, setup_test, test_invalid_uris_async, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/dns", 0, NULL, setup_test, test_dns, teardown_test);
 | |
|   g_test_add_vtable ("/proxy/override", 0, NULL, setup_test, test_override, teardown_test);
 | |
|   g_test_add_func ("/proxy/enumerator-ports", test_proxy_enumerator_ports);
 | |
| 
 | |
|   result = g_test_run();
 | |
| 
 | |
|   g_object_unref (client);
 | |
| 
 | |
|   g_cancellable_cancel (cancellable);
 | |
|   g_thread_join (proxy_a.thread);
 | |
|   g_thread_join (proxy_b.thread);
 | |
|   g_thread_join (server.server_thread);
 | |
| 
 | |
|   g_object_unref (cancellable);
 | |
| 
 | |
|   return result;
 | |
| }
 |