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