SOCKS proxy client for asyncio and aiohttp
==========================================
.. image:: https://travis-ci.org/nibrag/aiosocks.svg?branch=master
  :target: https://travis-ci.org/nibrag/aiosocks
  :align: right
.. image:: https://coveralls.io/repos/github/nibrag/aiosocks/badge.svg?branch=master
  :target: https://coveralls.io/github/nibrag/aiosocks?branch=master
  :align: right
.. image:: https://badge.fury.io/py/aiosocks.svg
  :target: https://badge.fury.io/py/aiosocks
Dependencies
------------
python 3.5+
aiohttp 2.1+
Features
--------
- SOCKS4, SOCKS4a and SOCKS5 version
- ProxyConnector for aiohttp
- SOCKS "CONNECT" command
TODO
----
- UDP associate
- TCP port binding
Installation
------------
You can install it using Pip:
.. code-block::
  pip install aiosocks
If you want the latest development version, you can install it from source:
.. code-block::
  git clone git@github.com:nibrag/aiosocks.git
  cd aiosocks
  python setup.py install
Usage
-----
direct usage
^^^^^^^^^^^^
.. code-block:: python
  import asyncio
  import aiosocks
  async def connect():
    socks5_addr = aiosocks.Socks5Addr('127.0.0.1', 1080)
    socks4_addr = aiosocks.Socks4Addr('127.0.0.1', 1080)
    
    socks5_auth = aiosocks.Socks5Auth('login', 'pwd')
    socks4_auth = aiosocks.Socks4Auth('ident')
  
    dst = ('github.com', 80)
    
    # socks5 connect
    transport, protocol = await aiosocks.create_connection(
        lambda: Protocol, proxy=socks5_addr, proxy_auth=socks5_auth, dst=dst)
    
    # socks4 connect
    transport, protocol = await aiosocks.create_connection(
        lambda: Protocol, proxy=socks4_addr, proxy_auth=socks4_auth, dst=dst)
        
    # socks4 without auth and local domain name resolving
    transport, protocol = await aiosocks.create_connection(
        lambda: Protocol, proxy=socks4_addr, proxy_auth=None, dst=dst, remote_resolve=False)
    # use socks protocol
    transport, protocol = await aiosocks.create_connection(
        None, proxy=socks4_addr, proxy_auth=None, dst=dst)
  
  if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(connect())
    loop.close()
**A wrapper for create_connection() returning a (reader, writer) pair**
.. code-block:: python
    # StreamReader, StreamWriter
    reader, writer = await aiosocks.open_connection(
        proxy=socks5_addr, proxy_auth=socks5_auth, dst=dst, remote_resolve=True)
    data = await reader.read(10)
    writer.write('data')
error handling
^^^^^^^^^^^^^^
`SocksError` is a base class for:
    - `NoAcceptableAuthMethods`
    - `LoginAuthenticationFailed`
    - `InvalidServerVersion`
    - `InvalidServerReply`
.. code-block:: python
    try:
      transport, protocol = await aiosocks.create_connection(
          lambda: Protocol, proxy=socks5_addr, proxy_auth=socks5_auth, dst=dst)
    except aiosocks.SocksConnectionError:
      # connection error
    except aiosocks.LoginAuthenticationFailed:
      # auth failed
    except aiosocks.NoAcceptableAuthMethods:
      # All offered SOCKS5 authentication methods were rejected
    except (aiosocks.InvalidServerVersion, aiosocks.InvalidServerReply):
      # something wrong
    except aiosocks.SocksError:
      # something other
or
.. code-block:: python
    try:
      transport, protocol = await aiosocks.create_connection(
          lambda: Protocol, proxy=socks5_addr, proxy_auth=socks5_auth, dst=dst)
    except aiosocks.SocksConnectionError:
        # connection error
    except aiosocks.SocksError:
        # socks error
aiohttp usage
^^^^^^^^^^^^^
.. code-block:: python
  import asyncio
  import aiohttp
  import aiosocks
  from aiosocks.connector import ProxyConnector, ProxyClientRequest
  async def load_github_main():
    auth5 = aiosocks.Socks5Auth('proxyuser1', password='pwd')
    auth4 = aiosocks.Socks4Auth('proxyuser1')
    ba = aiohttp.BasicAuth('login')
    # remote resolve
    conn = ProxyConnector(remote_resolve=True)
    # or locale resolve
    conn = ProxyConnector(remote_resolve=False)
    try:
      with aiohttp.ClientSession(connector=conn, request_class=ProxyClientRequest) as session:
        # socks5 proxy
        async with session.get('http://github.com/', proxy='socks5://127.0.0.1:1080',
                               proxy_auth=auth5) as resp:
          if resp.status == 200:
            print(await resp.text())
        # socks4 proxy
        async with session.get('http://github.com/', proxy='socks4://127.0.0.1:1081',
                               proxy_auth=auth4) as resp:
          if resp.status == 200:
            print(await resp.text())
        # http proxy
        async with session.get('http://github.com/', proxy='http://127.0.0.1:8080',
                               proxy_auth=ba) as resp:
          if resp.status == 200:
            print(await resp.text())
    except aiohttp.ProxyConnectionError:
      # connection problem
    except aiosocks.SocksError:
      # communication problem
  if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    loop.run_until_complete(load_github_main())
    loop.close()
Proxy from environment
^^^^^^^^^^^^^^^^^^^^^^
.. code-block:: python
  import os
  from aiosocks.connector import ProxyConnector, ProxyClientRequest
  os.environ['socks4_proxy'] = 'socks4://127.0.0.1:333'
  # or
  os.environ['socks5_proxy'] = 'socks5://127.0.0.1:444'
  conn = ProxyConnector()
  with aiohttp.ClientSession(connector=conn, request_class=ProxyClientRequest) as session:
        async with session.get('http://github.com/', proxy_from_env=True) as resp:
          if resp.status == 200:
            print(await resp.text())