2010-05-07 21:55:54 +02:00
|
|
|
|
/* GIO - GLib Input, Output and Streaming Library
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2010 Collabora, Ltd.
|
|
|
|
|
*
|
2022-05-18 10:12:45 +02:00
|
|
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
|
|
|
*
|
2010-05-07 21:55:54 +02:00
|
|
|
|
* 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
|
2017-05-27 18:21:30 +02:00
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* 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
|
2014-01-23 12:58:29 +01:00
|
|
|
|
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Author: Nicolas Dufresne <nicolas.dufresne@collabora.co.uk>
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
|
|
#include "gproxyresolver.h"
|
|
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
#include "glibintl.h"
|
|
|
|
|
|
|
|
|
|
#include "gasyncresult.h"
|
|
|
|
|
#include "gcancellable.h"
|
2017-01-06 13:33:25 +01:00
|
|
|
|
#include "gtask.h"
|
2010-05-07 21:55:54 +02:00
|
|
|
|
#include "giomodule.h"
|
2020-07-06 21:45:01 +02:00
|
|
|
|
#include "gioerror.h"
|
2010-05-07 21:55:54 +02:00
|
|
|
|
#include "giomodule-priv.h"
|
2017-01-06 13:33:25 +01:00
|
|
|
|
#include "gnetworkingprivate.h"
|
2010-05-07 21:55:54 +02:00
|
|
|
|
|
|
|
|
|
/**
|
2023-10-25 16:05:06 +02:00
|
|
|
|
* GProxyResolver:
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
2023-10-25 16:05:06 +02:00
|
|
|
|
* `GProxyResolver` provides synchronous and asynchronous network proxy
|
|
|
|
|
* resolution. `GProxyResolver` is used within [class@Gio.SocketClient] through
|
|
|
|
|
* the method [method@Gio.SocketConnectable.proxy_enumerate].
|
2017-05-02 13:23:32 +02:00
|
|
|
|
*
|
2023-10-25 16:05:06 +02:00
|
|
|
|
* Implementations of `GProxyResolver` based on
|
|
|
|
|
* [libproxy](https://github.com/libproxy/libproxy) and GNOME settings can be
|
|
|
|
|
* found in [glib-networking](https://gitlab.gnome.org/GNOME/glib-networking).
|
|
|
|
|
* GIO comes with an implementation for use inside Flatpak portals.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*/
|
|
|
|
|
|
2015-02-05 16:20:43 +01:00
|
|
|
|
/**
|
|
|
|
|
* GProxyResolverInterface:
|
|
|
|
|
* @g_iface: The parent interface.
|
|
|
|
|
* @is_supported: the virtual function pointer for g_proxy_resolver_is_supported()
|
|
|
|
|
* @lookup: the virtual function pointer for g_proxy_resolver_lookup()
|
|
|
|
|
* @lookup_async: the virtual function pointer for
|
|
|
|
|
* g_proxy_resolver_lookup_async()
|
|
|
|
|
* @lookup_finish: the virtual function pointer for
|
|
|
|
|
* g_proxy_resolver_lookup_finish()
|
|
|
|
|
*
|
|
|
|
|
* The virtual function table for #GProxyResolver.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-05-07 21:55:54 +02:00
|
|
|
|
G_DEFINE_INTERFACE (GProxyResolver, g_proxy_resolver, G_TYPE_OBJECT)
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
g_proxy_resolver_default_init (GProxyResolverInterface *iface)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-23 17:07:51 +01:00
|
|
|
|
static GProxyResolver *proxy_resolver_default_singleton = NULL; /* (owned) (atomic) */
|
|
|
|
|
|
2010-05-07 21:55:54 +02:00
|
|
|
|
/**
|
|
|
|
|
* g_proxy_resolver_get_default:
|
|
|
|
|
*
|
|
|
|
|
* Gets the default #GProxyResolver for the system.
|
|
|
|
|
*
|
2020-11-13 16:19:42 +01:00
|
|
|
|
* Returns: (not nullable) (transfer none): the default #GProxyResolver, which
|
|
|
|
|
* will be a dummy object if no proxy resolver is available
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Since: 2.26
|
|
|
|
|
*/
|
|
|
|
|
GProxyResolver *
|
|
|
|
|
g_proxy_resolver_get_default (void)
|
|
|
|
|
{
|
2023-09-12 08:59:52 +02:00
|
|
|
|
if (g_once_init_enter_pointer (&proxy_resolver_default_singleton))
|
2019-01-23 17:07:51 +01:00
|
|
|
|
{
|
|
|
|
|
GProxyResolver *singleton;
|
|
|
|
|
|
|
|
|
|
singleton = _g_io_module_get_default (G_PROXY_RESOLVER_EXTENSION_POINT_NAME,
|
|
|
|
|
"GIO_USE_PROXY_RESOLVER",
|
|
|
|
|
(GIOModuleVerifyFunc) g_proxy_resolver_is_supported);
|
|
|
|
|
|
2023-09-12 08:59:52 +02:00
|
|
|
|
g_once_init_leave_pointer (&proxy_resolver_default_singleton, singleton);
|
2019-01-23 17:07:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return proxy_resolver_default_singleton;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* g_proxy_resolver_is_supported:
|
|
|
|
|
* @resolver: a #GProxyResolver
|
|
|
|
|
*
|
|
|
|
|
* Checks if @resolver can be used on this system. (This is used
|
|
|
|
|
* internally; g_proxy_resolver_get_default() will only return a proxy
|
|
|
|
|
* resolver that returns %TRUE for this method.)
|
|
|
|
|
*
|
2014-02-20 01:35:23 +01:00
|
|
|
|
* Returns: %TRUE if @resolver is supported.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Since: 2.26
|
|
|
|
|
*/
|
|
|
|
|
gboolean
|
|
|
|
|
g_proxy_resolver_is_supported (GProxyResolver *resolver)
|
|
|
|
|
{
|
|
|
|
|
GProxyResolverInterface *iface;
|
|
|
|
|
|
|
|
|
|
g_return_val_if_fail (G_IS_PROXY_RESOLVER (resolver), FALSE);
|
|
|
|
|
|
|
|
|
|
iface = G_PROXY_RESOLVER_GET_IFACE (resolver);
|
|
|
|
|
|
|
|
|
|
return (* iface->is_supported) (resolver);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* g_proxy_resolver_lookup:
|
|
|
|
|
* @resolver: a #GProxyResolver
|
|
|
|
|
* @uri: a URI representing the destination to connect to
|
2016-10-29 03:29:02 +02:00
|
|
|
|
* @cancellable: (nullable): a #GCancellable, or %NULL
|
2010-05-07 21:55:54 +02:00
|
|
|
|
* @error: return location for a #GError, or %NULL
|
|
|
|
|
*
|
|
|
|
|
* Looks into the system proxy configuration to determine what proxy,
|
2014-02-06 14:04:52 +01:00
|
|
|
|
* if any, to use to connect to @uri. The returned proxy URIs are of
|
2022-12-22 22:34:44 +01:00
|
|
|
|
* the form `<protocol>://[user[:password]@]host[:port]` or
|
2014-02-09 08:07:26 +01:00
|
|
|
|
* `direct://`, where <protocol> could be http, rtsp, socks
|
2014-02-06 14:04:52 +01:00
|
|
|
|
* or other proxying protocol.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* If you don't know what network protocol is being used on the
|
2014-02-06 14:04:52 +01:00
|
|
|
|
* socket, you should use `none` as the URI protocol.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
* In this case, the resolver might still return a generic proxy type
|
|
|
|
|
* (such as SOCKS), but would not return protocol-specific proxy types
|
|
|
|
|
* (such as http).
|
|
|
|
|
*
|
2014-02-06 14:04:52 +01:00
|
|
|
|
* `direct://` is used when no proxy is needed.
|
2010-05-07 21:55:54 +02:00
|
|
|
|
* Direct connection should not be attempted unless it is part of the
|
|
|
|
|
* returned array of proxies.
|
|
|
|
|
*
|
2014-02-20 01:35:23 +01:00
|
|
|
|
* Returns: (transfer full) (array zero-terminated=1): A
|
2010-12-29 15:26:21 +01:00
|
|
|
|
* NULL-terminated array of proxy URIs. Must be freed
|
|
|
|
|
* with g_strfreev().
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Since: 2.26
|
|
|
|
|
*/
|
|
|
|
|
gchar **
|
|
|
|
|
g_proxy_resolver_lookup (GProxyResolver *resolver,
|
|
|
|
|
const gchar *uri,
|
|
|
|
|
GCancellable *cancellable,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
GProxyResolverInterface *iface;
|
2022-06-27 22:10:25 +02:00
|
|
|
|
gchar **proxy_uris;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
|
|
|
|
|
g_return_val_if_fail (G_IS_PROXY_RESOLVER (resolver), NULL);
|
|
|
|
|
g_return_val_if_fail (uri != NULL, NULL);
|
|
|
|
|
|
2020-08-06 16:14:26 +02:00
|
|
|
|
if (!g_uri_is_valid (uri, G_URI_FLAGS_NONE, NULL))
|
2020-07-06 21:45:01 +02:00
|
|
|
|
{
|
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
|
|
|
"Invalid URI ‘%s’", uri);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2017-01-06 13:33:25 +01:00
|
|
|
|
|
2010-05-07 21:55:54 +02:00
|
|
|
|
iface = G_PROXY_RESOLVER_GET_IFACE (resolver);
|
|
|
|
|
|
2022-06-27 22:10:25 +02:00
|
|
|
|
proxy_uris = (* iface->lookup) (resolver, uri, cancellable, error);
|
|
|
|
|
if (proxy_uris == NULL && error != NULL)
|
|
|
|
|
g_assert (*error != NULL);
|
|
|
|
|
return proxy_uris;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* g_proxy_resolver_lookup_async:
|
|
|
|
|
* @resolver: a #GProxyResolver
|
|
|
|
|
* @uri: a URI representing the destination to connect to
|
2016-10-29 03:29:02 +02:00
|
|
|
|
* @cancellable: (nullable): a #GCancellable, or %NULL
|
2010-12-29 15:26:21 +01:00
|
|
|
|
* @callback: (scope async): callback to call after resolution completes
|
2023-02-20 21:43:02 +01:00
|
|
|
|
* @user_data: data for @callback
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Asynchronous lookup of proxy. See g_proxy_resolver_lookup() for more
|
|
|
|
|
* details.
|
|
|
|
|
*
|
|
|
|
|
* Since: 2.26
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
g_proxy_resolver_lookup_async (GProxyResolver *resolver,
|
|
|
|
|
const gchar *uri,
|
|
|
|
|
GCancellable *cancellable,
|
|
|
|
|
GAsyncReadyCallback callback,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GProxyResolverInterface *iface;
|
2017-01-06 13:33:25 +01:00
|
|
|
|
GError *error = NULL;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
|
|
|
|
|
g_return_if_fail (G_IS_PROXY_RESOLVER (resolver));
|
|
|
|
|
g_return_if_fail (uri != NULL);
|
|
|
|
|
|
2020-08-06 16:14:26 +02:00
|
|
|
|
if (!g_uri_is_valid (uri, G_URI_FLAGS_NONE, NULL))
|
2017-01-06 13:33:25 +01:00
|
|
|
|
{
|
2020-07-06 21:45:01 +02:00
|
|
|
|
g_set_error (&error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
|
|
|
|
|
"Invalid URI ‘%s’", uri);
|
2017-01-06 13:33:25 +01:00
|
|
|
|
g_task_report_error (resolver, callback, user_data,
|
|
|
|
|
g_proxy_resolver_lookup_async,
|
|
|
|
|
g_steal_pointer (&error));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-07 21:55:54 +02:00
|
|
|
|
iface = G_PROXY_RESOLVER_GET_IFACE (resolver);
|
|
|
|
|
|
|
|
|
|
(* iface->lookup_async) (resolver, uri, cancellable, callback, user_data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* g_proxy_resolver_lookup_finish:
|
|
|
|
|
* @resolver: a #GProxyResolver
|
|
|
|
|
* @result: the result passed to your #GAsyncReadyCallback
|
|
|
|
|
* @error: return location for a #GError, or %NULL
|
|
|
|
|
*
|
|
|
|
|
* Call this function to obtain the array of proxy URIs when
|
|
|
|
|
* g_proxy_resolver_lookup_async() is complete. See
|
|
|
|
|
* g_proxy_resolver_lookup() for more details.
|
|
|
|
|
*
|
2014-02-20 01:35:23 +01:00
|
|
|
|
* Returns: (transfer full) (array zero-terminated=1): A
|
2010-12-29 15:26:21 +01:00
|
|
|
|
* NULL-terminated array of proxy URIs. Must be freed
|
|
|
|
|
* with g_strfreev().
|
2010-05-07 21:55:54 +02:00
|
|
|
|
*
|
|
|
|
|
* Since: 2.26
|
|
|
|
|
*/
|
|
|
|
|
gchar **
|
|
|
|
|
g_proxy_resolver_lookup_finish (GProxyResolver *resolver,
|
|
|
|
|
GAsyncResult *result,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
GProxyResolverInterface *iface;
|
2022-06-27 22:10:25 +02:00
|
|
|
|
gchar **proxy_uris;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
|
|
|
|
|
g_return_val_if_fail (G_IS_PROXY_RESOLVER (resolver), NULL);
|
|
|
|
|
|
2022-11-01 19:35:06 +01:00
|
|
|
|
if (g_async_result_is_tagged (result, g_proxy_resolver_lookup_async))
|
|
|
|
|
return g_task_propagate_pointer (G_TASK (result), error);
|
|
|
|
|
|
2010-05-07 21:55:54 +02:00
|
|
|
|
iface = G_PROXY_RESOLVER_GET_IFACE (resolver);
|
|
|
|
|
|
2022-06-27 22:10:25 +02:00
|
|
|
|
proxy_uris = (* iface->lookup_finish) (resolver, result, error);
|
|
|
|
|
if (proxy_uris == NULL && error != NULL)
|
|
|
|
|
g_assert (*error != NULL);
|
|
|
|
|
return proxy_uris;
|
2010-05-07 21:55:54 +02:00
|
|
|
|
}
|