Blame gio/gsocks5proxy.c

Packit ae235b
 /* GIO - GLib Input, Output and Streaming Library
Packit ae235b
 *
Packit ae235b
 * Copyright (C) 2008, 2010 Collabora, Ltd.
Packit ae235b
 * Copyright (C) 2008 Nokia Corporation. All rights reserved.
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
 * Author:  Youness Alaoui 
Packit ae235b
 *
Packit ae235b
 * Contributors:
Packit ae235b
 *	    Nicolas Dufresne <nicolas.dufresne@collabora.co.uk>
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include "config.h"
Packit ae235b
Packit ae235b
#include "gsocks5proxy.h"
Packit ae235b
Packit ae235b
#include <string.h>
Packit ae235b
Packit ae235b
#include "giomodule.h"
Packit ae235b
#include "giomodule-priv.h"
Packit ae235b
#include "giostream.h"
Packit ae235b
#include "ginetaddress.h"
Packit ae235b
#include "ginputstream.h"
Packit ae235b
#include "glibintl.h"
Packit ae235b
#include "goutputstream.h"
Packit ae235b
#include "gproxy.h"
Packit ae235b
#include "gproxyaddress.h"
Packit ae235b
#include "gtask.h"
Packit ae235b
Packit ae235b
#define SOCKS5_VERSION		  0x05
Packit ae235b
Packit ae235b
#define SOCKS5_CMD_CONNECT	  0x01
Packit ae235b
#define SOCKS5_CMD_BIND		  0x02
Packit ae235b
#define SOCKS5_CMD_UDP_ASSOCIATE  0x03
Packit ae235b
Packit ae235b
#define SOCKS5_ATYP_IPV4	  0x01
Packit ae235b
#define SOCKS5_ATYP_DOMAINNAME	  0x03
Packit ae235b
#define SOCKS5_ATYP_IPV6	  0x04
Packit ae235b
Packit ae235b
#define SOCKS5_AUTH_VERSION	  0x01
Packit ae235b
Packit ae235b
#define SOCKS5_AUTH_NONE	  0x00
Packit ae235b
#define SOCKS5_AUTH_GSSAPI	  0x01
Packit ae235b
#define SOCKS5_AUTH_USR_PASS	  0x02
Packit ae235b
#define SOCKS5_AUTH_NO_ACCEPT	  0xff
Packit ae235b
Packit ae235b
#define SOCKS5_MAX_LEN		  255
Packit ae235b
#define SOCKS5_RESERVED		  0x00
Packit ae235b
Packit ae235b
#define SOCKS5_REP_SUCCEEDED	  0x00
Packit ae235b
#define SOCKS5_REP_SRV_FAILURE    0x01
Packit ae235b
#define SOCKS5_REP_NOT_ALLOWED    0x02
Packit ae235b
#define SOCKS5_REP_NET_UNREACH    0x03
Packit ae235b
#define SOCKS5_REP_HOST_UNREACH   0x04
Packit ae235b
#define SOCKS5_REP_REFUSED        0x05
Packit ae235b
#define SOCKS5_REP_TTL_EXPIRED    0x06
Packit ae235b
#define SOCKS5_REP_CMD_NOT_SUP    0x07
Packit ae235b
#define SOCKS5_REP_ATYPE_NOT_SUP  0x08
Packit ae235b
Packit ae235b
Packit ae235b
struct _GSocks5Proxy
Packit ae235b
{
Packit ae235b
  GObject parent;
Packit ae235b
};
Packit ae235b
Packit ae235b
struct _GSocks5ProxyClass
Packit ae235b
{
Packit ae235b
  GObjectClass parent_class;
Packit ae235b
};
Packit ae235b
Packit ae235b
static void g_socks5_proxy_iface_init (GProxyInterface *proxy_iface);
Packit ae235b
Packit ae235b
#define g_socks5_proxy_get_type _g_socks5_proxy_get_type
Packit ae235b
G_DEFINE_TYPE_WITH_CODE (GSocks5Proxy, g_socks5_proxy, G_TYPE_OBJECT,
Packit ae235b
			 G_IMPLEMENT_INTERFACE (G_TYPE_PROXY,
Packit ae235b
						g_socks5_proxy_iface_init)
Packit ae235b
			 _g_io_modules_ensure_extension_points_registered ();
Packit ae235b
			 g_io_extension_point_implement (G_PROXY_EXTENSION_POINT_NAME,
Packit ae235b
							 g_define_type_id,
Packit ae235b
							 "socks5",
Packit ae235b
							 0))
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_socks5_proxy_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  /* must chain up */
Packit ae235b
  G_OBJECT_CLASS (g_socks5_proxy_parent_class)->finalize (object);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_socks5_proxy_init (GSocks5Proxy *proxy)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * +----+----------+----------+
Packit ae235b
 * |VER | NMETHODS | METHODS  |
Packit ae235b
 * +----+----------+----------+
Packit ae235b
 * | 1  |    1     | 1 to 255 |
Packit ae235b
 * +----+----------+----------+
Packit ae235b
 */
Packit ae235b
#define SOCKS5_NEGO_MSG_LEN	  4
Packit ae235b
static gint
Packit ae235b
set_nego_msg (guint8 *msg, gboolean has_auth)
Packit ae235b
{
Packit ae235b
  gint len = 3;
Packit ae235b
Packit ae235b
  msg[0] = SOCKS5_VERSION;
Packit ae235b
  msg[1] = 0x01; /* number of methods supported */
Packit ae235b
  msg[2] = SOCKS5_AUTH_NONE;
Packit ae235b
Packit ae235b
  /* add support for authentication method */
Packit ae235b
  if (has_auth)
Packit ae235b
    {
Packit ae235b
      msg[1] = 0x02; /* number of methods supported */
Packit ae235b
      msg[3] = SOCKS5_AUTH_USR_PASS;
Packit ae235b
      len++;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return len;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * +----+--------+
Packit ae235b
 * |VER | METHOD |
Packit ae235b
 * +----+--------+
Packit ae235b
 * | 1  |   1    |
Packit ae235b
 * +----+--------+
Packit ae235b
 */
Packit ae235b
#define SOCKS5_NEGO_REP_LEN	  2
Packit ae235b
static gboolean
Packit ae235b
parse_nego_reply (const guint8 *data,
Packit ae235b
		  gboolean     has_auth,
Packit ae235b
		  gboolean    *must_auth,
Packit ae235b
		  GError     **error)
Packit ae235b
{
Packit ae235b
  if (data[0] != SOCKS5_VERSION)
Packit ae235b
    {
Packit ae235b
      g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			   _("The server is not a SOCKSv5 proxy server."));
Packit ae235b
      return FALSE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  switch (data[1])
Packit ae235b
    {
Packit ae235b
      case SOCKS5_AUTH_NONE:
Packit ae235b
	*must_auth = FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_AUTH_USR_PASS:
Packit ae235b
	if (!has_auth)
Packit ae235b
	  {
Packit ae235b
	    g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_NEED_AUTH,
Packit ae235b
			   _("The SOCKSv5 proxy requires authentication."));
Packit ae235b
	    return FALSE;
Packit ae235b
	  }
Packit ae235b
	*must_auth = TRUE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_AUTH_GSSAPI:
Packit ae235b
      case SOCKS5_AUTH_NO_ACCEPT:
Packit ae235b
      default:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_AUTH_FAILED,
Packit ae235b
			     _("The SOCKSv5 proxy requires an authentication "
Packit ae235b
			       "method that is not supported by GLib."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
#define SOCKS5_AUTH_MSG_LEN       515
Packit ae235b
static gint
Packit ae235b
set_auth_msg (guint8	  *msg,
Packit ae235b
	      const gchar *username,
Packit ae235b
	      const gchar *password,
Packit ae235b
	      GError **error)
Packit ae235b
{
Packit ae235b
  gint len = 0;
Packit ae235b
  gint ulen = 0; /* username length */
Packit ae235b
  gint plen = 0; /* Password length */
Packit ae235b
Packit ae235b
  if (username)
Packit ae235b
    ulen = strlen (username);
Packit ae235b
Packit ae235b
  if (password)
Packit ae235b
    plen = strlen (password);
Packit ae235b
Packit ae235b
  if (ulen > SOCKS5_MAX_LEN || plen > SOCKS5_MAX_LEN)
Packit ae235b
    {
Packit ae235b
      g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			   _("Username or password is too long for SOCKSv5 "
Packit ae235b
			     "protocol."));
Packit ae235b
      return -1;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  msg[len++] = SOCKS5_AUTH_VERSION;
Packit ae235b
  msg[len++] = ulen;
Packit ae235b
Packit ae235b
  if (ulen > 0)
Packit ae235b
    memcpy (msg + len, username, ulen);
Packit ae235b
Packit ae235b
  len += ulen;
Packit ae235b
  msg[len++] = plen;
Packit ae235b
Packit ae235b
  if (plen > 0)
Packit ae235b
    memcpy (msg + len, password, plen);
Packit ae235b
Packit ae235b
  len += plen;
Packit ae235b
Packit ae235b
  return len;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
check_auth_status (const guint8 *data, GError **error)
Packit ae235b
{
Packit ae235b
  if (data[0] != SOCKS5_VERSION
Packit ae235b
      || data[1] != SOCKS5_REP_SUCCEEDED)
Packit ae235b
    {
Packit ae235b
      g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_AUTH_FAILED,
Packit ae235b
			   _("SOCKSv5 authentication failed due to wrong "
Packit ae235b
			     "username or password."));
Packit ae235b
      return FALSE;
Packit ae235b
    }
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * |VER | CMD |  RSV  | ATYP | DST.ADDR | DST.PORT |
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * | 1  |  1  | X'00' |  1   | Variable |    2     |
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * DST.ADDR is a string with first byte being the size. So DST.ADDR may not be
Packit ae235b
 * longer then 256 bytes.
Packit ae235b
 */
Packit ae235b
#define SOCKS5_CONN_MSG_LEN	  262
Packit ae235b
static gint
Packit ae235b
set_connect_msg (guint8       *msg,
Packit ae235b
		 const gchar *hostname,
Packit ae235b
		 guint16      port,
Packit ae235b
		 GError     **error)
Packit ae235b
{
Packit ae235b
  guint len = 0;
Packit ae235b
Packit ae235b
  msg[len++] = SOCKS5_VERSION;
Packit ae235b
  msg[len++] = SOCKS5_CMD_CONNECT;
Packit ae235b
  msg[len++] = SOCKS5_RESERVED;
Packit ae235b
Packit ae235b
  if (g_hostname_is_ip_address (hostname))
Packit ae235b
    {
Packit ae235b
      GInetAddress *addr = g_inet_address_new_from_string (hostname);
Packit ae235b
      gsize addr_len = g_inet_address_get_native_size (addr);
Packit ae235b
Packit ae235b
      /* We are cheating for simplicity, here's the logic:
Packit ae235b
       *   1 = IPV4 = 4 bytes / 4
Packit ae235b
       *   4 = IPV6 = 16 bytes / 4 */
Packit ae235b
      msg[len++] = addr_len / 4;
Packit ae235b
      memcpy (msg + len, g_inet_address_to_bytes (addr), addr_len);
Packit ae235b
      len += addr_len;
Packit ae235b
Packit ae235b
      g_object_unref (addr);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      gsize host_len = strlen (hostname);
Packit ae235b
Packit ae235b
      if (host_len > SOCKS5_MAX_LEN)
Packit ae235b
	{
Packit ae235b
	  g_set_error (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
		       _("Hostname “%s” is too long for SOCKSv5 protocol"),
Packit ae235b
		       hostname);
Packit ae235b
	  return -1;
Packit ae235b
	}
Packit ae235b
Packit ae235b
      msg[len++] = SOCKS5_ATYP_DOMAINNAME;
Packit ae235b
      msg[len++] = (guint8) host_len;
Packit ae235b
      memcpy (msg + len, hostname, host_len);
Packit ae235b
      len += host_len;
Packit ae235b
    }
Packit ae235b
Packit ae235b
    {
Packit ae235b
      guint16 hp = g_htons (port);
Packit ae235b
      memcpy (msg + len, &hp, 2);
Packit ae235b
      len += 2;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return len;
Packit ae235b
}
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * |VER | REP |  RSV  | ATYP | BND.ADDR | BND.PORT |
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * | 1  |  1  | X'00' |  1   | Variable |    2     |
Packit ae235b
 * +----+-----+-------+------+----------+----------+
Packit ae235b
 * This reply need to be read by small part to determin size. Buffer
Packit ae235b
 * size is determined in function of the biggest part to read.
Packit ae235b
 *
Packit ae235b
 * The parser only requires 4 bytes.
Packit ae235b
 */
Packit ae235b
#define SOCKS5_CONN_REP_LEN	  255
Packit ae235b
static gboolean
Packit ae235b
parse_connect_reply (const guint8 *data, gint *atype, GError **error)
Packit ae235b
{
Packit ae235b
  if (data[0] != SOCKS5_VERSION)
Packit ae235b
    {
Packit ae235b
      g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			   _("The server is not a SOCKSv5 proxy server."));
Packit ae235b
      return FALSE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  switch (data[1])
Packit ae235b
    {
Packit ae235b
      case SOCKS5_REP_SUCCEEDED:
Packit ae235b
	if (data[2] != SOCKS5_RESERVED)
Packit ae235b
	  {
Packit ae235b
	    g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			   _("The server is not a SOCKSv5 proxy server."));
Packit ae235b
	    return FALSE;
Packit ae235b
	  }
Packit ae235b
Packit ae235b
	switch (data[3])
Packit ae235b
	  {
Packit ae235b
	  case SOCKS5_ATYP_IPV4:
Packit ae235b
	  case SOCKS5_ATYP_IPV6:
Packit ae235b
	  case SOCKS5_ATYP_DOMAINNAME:
Packit ae235b
	    *atype = data[3];
Packit ae235b
	    break;
Packit ae235b
Packit ae235b
	  default:
Packit ae235b
	    g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			   _("The SOCKSv5 proxy server uses unknown address type."));
Packit ae235b
	    return FALSE;
Packit ae235b
	  }
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_SRV_FAILURE:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			     _("Internal SOCKSv5 proxy server error."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_NOT_ALLOWED:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_NOT_ALLOWED,
Packit ae235b
			     _("SOCKSv5 connection not allowed by ruleset."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_TTL_EXPIRED:
Packit ae235b
      case SOCKS5_REP_HOST_UNREACH:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_HOST_UNREACHABLE,
Packit ae235b
			     _("Host unreachable through SOCKSv5 server."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_NET_UNREACH:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NETWORK_UNREACHABLE,
Packit ae235b
			     _("Network unreachable through SOCKSv5 proxy."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_REFUSED:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_CONNECTION_REFUSED,
Packit ae235b
			     _("Connection refused through SOCKSv5 proxy."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_CMD_NOT_SUP:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			     _("SOCKSv5 proxy does not support “connect” command."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      case SOCKS5_REP_ATYPE_NOT_SUP:
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			     _("SOCKSv5 proxy does not support provided address type."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
Packit ae235b
      default: /* Unknown error */
Packit ae235b
	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PROXY_FAILED,
Packit ae235b
			     _("Unknown SOCKSv5 proxy error."));
Packit ae235b
	return FALSE;
Packit ae235b
	break;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static GIOStream *
Packit ae235b
g_socks5_proxy_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
  gboolean has_auth;
Packit ae235b
  GInputStream *in;
Packit ae235b
  GOutputStream *out;
Packit ae235b
  const gchar *hostname;
Packit ae235b
  guint16 port;
Packit ae235b
  const gchar *username;
Packit ae235b
  const gchar *password;
Packit ae235b
Packit ae235b
  hostname = g_proxy_address_get_destination_hostname (proxy_address);
Packit ae235b
  port = g_proxy_address_get_destination_port (proxy_address);
Packit ae235b
  username = g_proxy_address_get_username (proxy_address);
Packit ae235b
  password = g_proxy_address_get_password (proxy_address);
Packit ae235b
Packit ae235b
  has_auth = username || password;
Packit ae235b
  
Packit ae235b
  in = g_io_stream_get_input_stream (io_stream);
Packit ae235b
  out = g_io_stream_get_output_stream (io_stream);
Packit ae235b
Packit ae235b
  /* Send SOCKS5 handshake */
Packit ae235b
    {
Packit ae235b
      guint8 msg[SOCKS5_NEGO_MSG_LEN];
Packit ae235b
      gint len;
Packit ae235b
Packit ae235b
      len = set_nego_msg (msg, has_auth);
Packit ae235b
Packit ae235b
      if (!g_output_stream_write_all (out, msg, len, NULL,
Packit ae235b
				      cancellable, error))
Packit ae235b
	goto error;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  /* Receive SOCKS5 response and reply with authentication if required */
Packit ae235b
    {
Packit ae235b
      guint8 data[SOCKS5_NEGO_REP_LEN];
Packit ae235b
      gboolean must_auth = FALSE;
Packit ae235b
Packit ae235b
      if (!g_input_stream_read_all (in, data, sizeof (data), NULL,
Packit ae235b
				    cancellable, error))
Packit ae235b
	goto error;
Packit ae235b
Packit ae235b
      if (!parse_nego_reply (data, has_auth, &must_auth, error))
Packit ae235b
	  goto error;
Packit ae235b
Packit ae235b
      if (must_auth)
Packit ae235b
	{
Packit ae235b
	  guint8 msg[SOCKS5_AUTH_MSG_LEN];
Packit ae235b
	  gint len;
Packit ae235b
Packit ae235b
	  len = set_auth_msg (msg, username, password, error);
Packit ae235b
Packit ae235b
	  if (len < 0)
Packit ae235b
	    goto error;
Packit ae235b
	  
Packit ae235b
	  if (!g_output_stream_write_all (out, msg, len, NULL,
Packit ae235b
					  cancellable, error))
Packit ae235b
	    goto error;
Packit ae235b
Packit ae235b
	  if (!g_input_stream_read_all (in, data, sizeof (data), NULL,
Packit ae235b
					cancellable, error))
Packit ae235b
	    goto error;
Packit ae235b
Packit ae235b
	  if (!check_auth_status (data, error))
Packit ae235b
	    goto error;
Packit ae235b
	}
Packit ae235b
    }
Packit ae235b
Packit ae235b
  /* Send SOCKS5 connection request */
Packit ae235b
    {
Packit ae235b
      guint8 msg[SOCKS5_CONN_MSG_LEN];
Packit ae235b
      gint len;
Packit ae235b
      
Packit ae235b
      len = set_connect_msg (msg, hostname, port, error);
Packit ae235b
Packit ae235b
      if (len < 0)
Packit ae235b
	goto error;
Packit ae235b
Packit ae235b
      if (!g_output_stream_write_all (out, msg, len, NULL,
Packit ae235b
				      cancellable, error))
Packit ae235b
	goto error;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  /* Read SOCKS5 response */
Packit ae235b
    {
Packit ae235b
      guint8 data[SOCKS5_CONN_REP_LEN];
Packit ae235b
      gint atype;
Packit ae235b
Packit ae235b
      if (!g_input_stream_read_all (in, data, 4, NULL,
Packit ae235b
				    cancellable, error))
Packit ae235b
	goto error;
Packit ae235b
Packit ae235b
      if (!parse_connect_reply (data, &atype, error))
Packit ae235b
	goto error;
Packit ae235b
Packit ae235b
      switch (atype)
Packit ae235b
	{
Packit ae235b
	  case SOCKS5_ATYP_IPV4:
Packit ae235b
	    if (!g_input_stream_read_all (in, data, 6, NULL,
Packit ae235b
					  cancellable, error))
Packit ae235b
	      goto error;
Packit ae235b
	    break;
Packit ae235b
Packit ae235b
	  case SOCKS5_ATYP_IPV6:
Packit ae235b
	    if (!g_input_stream_read_all (in, data, 18, NULL,
Packit ae235b
					  cancellable, error))
Packit ae235b
	      goto error;
Packit ae235b
	    break;
Packit ae235b
Packit ae235b
	  case SOCKS5_ATYP_DOMAINNAME:
Packit ae235b
	    if (!g_input_stream_read_all (in, data, 1, NULL,
Packit ae235b
					  cancellable, error))
Packit ae235b
	      goto error;
Packit ae235b
	    if (!g_input_stream_read_all (in, data, data[0] + 2, NULL,
Packit ae235b
					  cancellable, error))
Packit ae235b
	      goto error;
Packit ae235b
	    break;
Packit ae235b
	}
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return g_object_ref (io_stream);
Packit ae235b
Packit ae235b
error:
Packit ae235b
  return NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
typedef struct
Packit ae235b
{
Packit ae235b
  GIOStream *io_stream;
Packit ae235b
  gchar *hostname;
Packit ae235b
  guint16 port;
Packit ae235b
  gchar *username;
Packit ae235b
  gchar *password;
Packit ae235b
  guint8 *buffer;
Packit ae235b
  gssize length;
Packit ae235b
  gssize offset;
Packit ae235b
} ConnectAsyncData;
Packit ae235b
Packit ae235b
static void nego_msg_write_cb	      (GObject          *source,
Packit ae235b
				       GAsyncResult     *res,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void nego_reply_read_cb	      (GObject          *source,
Packit ae235b
				       GAsyncResult     *res,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void auth_msg_write_cb	      (GObject          *source,
Packit ae235b
				       GAsyncResult     *res,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void auth_reply_read_cb	      (GObject          *source,
Packit ae235b
				       GAsyncResult     *result,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void send_connect_msg	      (GTask            *task);
Packit ae235b
static void connect_msg_write_cb      (GObject          *source,
Packit ae235b
				       GAsyncResult     *result,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void connect_reply_read_cb     (GObject          *source,
Packit ae235b
				       GAsyncResult     *result,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void connect_addr_len_read_cb  (GObject          *source,
Packit ae235b
				       GAsyncResult     *result,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
static void connect_addr_read_cb      (GObject          *source,
Packit ae235b
				       GAsyncResult     *result,
Packit ae235b
				       gpointer          user_data);
Packit ae235b
Packit ae235b
static void
Packit ae235b
free_connect_data (ConnectAsyncData *data)
Packit ae235b
{
Packit ae235b
  g_object_unref (data->io_stream);
Packit ae235b
Packit ae235b
  g_free (data->hostname);
Packit ae235b
  g_free (data->username);
Packit ae235b
  g_free (data->password);
Packit ae235b
  g_free (data->buffer);
Packit ae235b
Packit ae235b
  g_slice_free (ConnectAsyncData, data);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
do_read (GAsyncReadyCallback callback, GTask *task, ConnectAsyncData *data)
Packit ae235b
{
Packit ae235b
   GInputStream *in;
Packit ae235b
   in = g_io_stream_get_input_stream (data->io_stream);
Packit ae235b
   g_input_stream_read_async (in,
Packit ae235b
			      data->buffer + data->offset,
Packit ae235b
			      data->length - data->offset,
Packit ae235b
			      g_task_get_priority (task),
Packit ae235b
			      g_task_get_cancellable (task),
Packit ae235b
			      callback, task);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
do_write (GAsyncReadyCallback callback, GTask *task, ConnectAsyncData *data)
Packit ae235b
{
Packit ae235b
  GOutputStream *out;
Packit ae235b
  out = g_io_stream_get_output_stream (data->io_stream);
Packit ae235b
  g_output_stream_write_async (out,
Packit ae235b
			       data->buffer + data->offset,
Packit ae235b
			       data->length - data->offset,
Packit ae235b
			       g_task_get_priority (task),
Packit ae235b
			       g_task_get_cancellable (task),
Packit ae235b
			       callback, task);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_socks5_proxy_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
  GTask *task;
Packit ae235b
  ConnectAsyncData *data;
Packit ae235b
Packit ae235b
  data = g_slice_new0 (ConnectAsyncData);
Packit ae235b
  data->io_stream = g_object_ref (io_stream);
Packit ae235b
Packit ae235b
  task = g_task_new (proxy, cancellable, callback, user_data);
Packit ae235b
  g_task_set_source_tag (task, g_socks5_proxy_connect_async);
Packit ae235b
  g_task_set_task_data (task, data, (GDestroyNotify) free_connect_data);
Packit ae235b
Packit ae235b
  g_object_get (G_OBJECT (proxy_address),
Packit ae235b
		"destination-hostname", &data->hostname,
Packit ae235b
		"destination-port", &data->port,
Packit ae235b
		"username", &data->username,
Packit ae235b
		"password", &data->password,
Packit ae235b
		NULL);
Packit ae235b
Packit ae235b
  data->buffer = g_malloc0 (SOCKS5_NEGO_MSG_LEN);
Packit ae235b
  data->length = set_nego_msg (data->buffer,
Packit ae235b
			       data->username || data->password);
Packit ae235b
  data->offset = 0;
Packit ae235b
Packit ae235b
  do_write (nego_msg_write_cb, task, data);
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static void
Packit ae235b
nego_msg_write_cb (GObject      *source,
Packit ae235b
		   GAsyncResult *res,
Packit ae235b
		   gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize written;
Packit ae235b
Packit ae235b
  written = g_output_stream_write_finish (G_OUTPUT_STREAM (source),
Packit ae235b
					  res, &error);
Packit ae235b
Packit ae235b
  if (written < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += written;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      g_free (data->buffer);
Packit ae235b
Packit ae235b
      data->buffer = g_malloc0 (SOCKS5_NEGO_REP_LEN);
Packit ae235b
      data->length = SOCKS5_NEGO_REP_LEN;
Packit ae235b
      data->offset = 0;
Packit ae235b
Packit ae235b
      do_read (nego_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_write (nego_msg_write_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
nego_reply_read_cb (GObject      *source,
Packit ae235b
		    GAsyncResult *res,
Packit ae235b
		    gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize read;
Packit ae235b
Packit ae235b
  read = g_input_stream_read_finish (G_INPUT_STREAM (source),
Packit ae235b
				     res, &error);
Packit ae235b
Packit ae235b
  if (read < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += read;
Packit ae235b
  
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      GError *error = NULL;
Packit ae235b
      gboolean must_auth = FALSE;
Packit ae235b
      gboolean has_auth = data->username || data->password;
Packit ae235b
Packit ae235b
      if (!parse_nego_reply (data->buffer, has_auth, &must_auth, &error))
Packit ae235b
	{
Packit ae235b
	  g_task_return_error (task, error);
Packit ae235b
	  g_object_unref (task);
Packit ae235b
	  return;
Packit ae235b
	}
Packit ae235b
Packit ae235b
      if (must_auth)
Packit ae235b
	{
Packit ae235b
	  g_free (data->buffer);
Packit ae235b
Packit ae235b
	  data->buffer = g_malloc0 (SOCKS5_AUTH_MSG_LEN);
Packit ae235b
	  data->length = set_auth_msg (data->buffer,
Packit ae235b
				       data->username,
Packit ae235b
				       data->password,
Packit ae235b
				       &error);
Packit ae235b
	  data->offset = 0;
Packit ae235b
Packit ae235b
	  if (data->length < 0)
Packit ae235b
	    {
Packit ae235b
	      g_task_return_error (task, error);
Packit ae235b
	      g_object_unref (task);
Packit ae235b
	      return;
Packit ae235b
	    }
Packit ae235b
	  
Packit ae235b
	  do_write (auth_msg_write_cb, task, data);
Packit ae235b
	}
Packit ae235b
      else
Packit ae235b
	{
Packit ae235b
	  send_connect_msg (task);
Packit ae235b
	}
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_read (nego_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
auth_msg_write_cb (GObject      *source,
Packit ae235b
		   GAsyncResult *result,
Packit ae235b
		   gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize written;
Packit ae235b
Packit ae235b
  written = g_output_stream_write_finish (G_OUTPUT_STREAM (source),
Packit ae235b
					  result, &error);
Packit ae235b
  
Packit ae235b
  if (written < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += written;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      g_free (data->buffer);
Packit ae235b
Packit ae235b
      data->buffer = g_malloc0 (SOCKS5_NEGO_REP_LEN);
Packit ae235b
      data->length = SOCKS5_NEGO_REP_LEN;
Packit ae235b
      data->offset = 0;
Packit ae235b
Packit ae235b
      do_read (auth_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_write (auth_msg_write_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
auth_reply_read_cb (GObject      *source,
Packit ae235b
		    GAsyncResult *result,
Packit ae235b
		    gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize read;
Packit ae235b
Packit ae235b
  read = g_input_stream_read_finish (G_INPUT_STREAM (source),
Packit ae235b
				     result, &error);
Packit ae235b
Packit ae235b
  if (read < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += read;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      if (!check_auth_status (data->buffer, &error))
Packit ae235b
	{
Packit ae235b
	  g_task_return_error (task, error);
Packit ae235b
	  g_object_unref (task);
Packit ae235b
	  return;
Packit ae235b
	}
Packit ae235b
	
Packit ae235b
      send_connect_msg (task);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_read (auth_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
send_connect_msg (GTask *task)
Packit ae235b
{
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  g_free (data->buffer);
Packit ae235b
Packit ae235b
  data->buffer = g_malloc0 (SOCKS5_CONN_MSG_LEN);
Packit ae235b
  data->length = set_connect_msg (data->buffer,
Packit ae235b
				  data->hostname,
Packit ae235b
				  data->port,
Packit ae235b
				  &error);
Packit ae235b
  data->offset = 0;
Packit ae235b
  
Packit ae235b
  if (data->length < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  do_write (connect_msg_write_cb, task, data);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
connect_msg_write_cb (GObject      *source,
Packit ae235b
		      GAsyncResult *result,
Packit ae235b
		      gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize written;
Packit ae235b
Packit ae235b
  written = g_output_stream_write_finish (G_OUTPUT_STREAM (source),
Packit ae235b
					  result, &error);
Packit ae235b
  
Packit ae235b
  if (written < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += written;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      g_free (data->buffer);
Packit ae235b
Packit ae235b
      data->buffer = g_malloc0 (SOCKS5_CONN_REP_LEN);
Packit ae235b
      data->length = 4;
Packit ae235b
      data->offset = 0;
Packit ae235b
Packit ae235b
      do_read (connect_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_write (connect_msg_write_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
connect_reply_read_cb (GObject       *source,
Packit ae235b
		       GAsyncResult  *result,
Packit ae235b
		       gpointer       user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize read;
Packit ae235b
Packit ae235b
  read = g_input_stream_read_finish (G_INPUT_STREAM (source),
Packit ae235b
				     result, &error);
Packit ae235b
Packit ae235b
  if (read < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += read;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      gint atype;
Packit ae235b
Packit ae235b
      if (!parse_connect_reply (data->buffer, &atype, &error))
Packit ae235b
	{
Packit ae235b
	  g_task_return_error (task, error);
Packit ae235b
	  g_object_unref (task);
Packit ae235b
	  return;
Packit ae235b
	}
Packit ae235b
Packit ae235b
      switch (atype)
Packit ae235b
	{
Packit ae235b
	  case SOCKS5_ATYP_IPV4:
Packit ae235b
	    data->length = 6;
Packit ae235b
	    data->offset = 0;
Packit ae235b
	    do_read (connect_addr_read_cb, task, data);
Packit ae235b
	    break;
Packit ae235b
Packit ae235b
	  case SOCKS5_ATYP_IPV6:
Packit ae235b
	    data->length = 18;
Packit ae235b
	    data->offset = 0;
Packit ae235b
	    do_read (connect_addr_read_cb, task, data);
Packit ae235b
	    break;
Packit ae235b
Packit ae235b
	  case SOCKS5_ATYP_DOMAINNAME:
Packit ae235b
	    data->length = 1;
Packit ae235b
	    data->offset = 0;
Packit ae235b
	    do_read (connect_addr_len_read_cb, task, data);
Packit ae235b
	    break;
Packit ae235b
	}
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_read (connect_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
connect_addr_len_read_cb (GObject      *source,
Packit ae235b
			  GAsyncResult *result,
Packit ae235b
			  gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize read;
Packit ae235b
Packit ae235b
  read = g_input_stream_read_finish (G_INPUT_STREAM (source),
Packit ae235b
				     result, &error);
Packit ae235b
Packit ae235b
  if (read < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->length = data->buffer[0] + 2;
Packit ae235b
  data->offset = 0;
Packit ae235b
Packit ae235b
  do_read (connect_addr_read_cb, task, data);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
connect_addr_read_cb (GObject      *source,
Packit ae235b
		      GAsyncResult *result,
Packit ae235b
		      gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GTask *task = user_data;
Packit ae235b
  ConnectAsyncData *data = g_task_get_task_data (task);
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gssize read;
Packit ae235b
Packit ae235b
  read = g_input_stream_read_finish (G_INPUT_STREAM (source),
Packit ae235b
				     result, &error);
Packit ae235b
Packit ae235b
  if (read < 0)
Packit ae235b
    {
Packit ae235b
      g_task_return_error (task, error);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->offset += read;
Packit ae235b
Packit ae235b
  if (data->offset == data->length)
Packit ae235b
    {
Packit ae235b
      g_task_return_pointer (task, g_object_ref (data->io_stream), g_object_unref);
Packit ae235b
      g_object_unref (task);
Packit ae235b
      return;
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      do_read (connect_reply_read_cb, task, data);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static GIOStream *
Packit ae235b
g_socks5_proxy_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 gboolean
Packit ae235b
g_socks5_proxy_supports_hostname (GProxy *proxy)
Packit ae235b
{
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_socks5_proxy_class_init (GSocks5ProxyClass *class)
Packit ae235b
{
Packit ae235b
  GObjectClass *object_class;
Packit ae235b
Packit ae235b
  object_class = (GObjectClass *) class;
Packit ae235b
  object_class->finalize = g_socks5_proxy_finalize;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
g_socks5_proxy_iface_init (GProxyInterface *proxy_iface)
Packit ae235b
{
Packit ae235b
  proxy_iface->connect  = g_socks5_proxy_connect;
Packit ae235b
  proxy_iface->connect_async = g_socks5_proxy_connect_async;
Packit ae235b
  proxy_iface->connect_finish = g_socks5_proxy_connect_finish;
Packit ae235b
  proxy_iface->supports_hostname = g_socks5_proxy_supports_hostname;
Packit ae235b
}