A toolbelt of useful classes and functions to be used with python-requests

Overview

The Requests Toolbelt

This is just a collection of utilities for python-requests, but don't really belong in requests proper. The minimum tested requests version is 2.1.0. In reality, the toolbelt should work with 2.0.1 as well, but some idiosyncracies prevent effective or sane testing on that version.

pip install requests-toolbelt to get started!

multipart/form-data Encoder

The main attraction is a streaming multipart form-data object, MultipartEncoder. Its API looks like this:

from requests_toolbelt import MultipartEncoder
import requests

m = MultipartEncoder(
    fields={'field0': 'value', 'field1': 'value',
            'field2': ('filename', open('file.py', 'rb'), 'text/plain')}
    )

r = requests.post('http://httpbin.org/post', data=m,
                  headers={'Content-Type': m.content_type})

You can also use multipart/form-data encoding for requests that don't require files:

from requests_toolbelt import MultipartEncoder
import requests

m = MultipartEncoder(fields={'field0': 'value', 'field1': 'value'})

r = requests.post('http://httpbin.org/post', data=m,
                  headers={'Content-Type': m.content_type})

Or, you can just create the string and examine the data:

# Assuming `m` is one of the above
m.to_string()  # Always returns unicode

User-Agent constructor

You can easily construct a requests-style User-Agent string:

from requests_toolbelt import user_agent

headers = {
    'User-Agent': user_agent('my_package', '0.0.1')
    }

r = requests.get('https://api.github.com/users', headers=headers)

SSLAdapter

The SSLAdapter was originally published on Cory Benfield's blog. This adapter allows the user to choose one of the SSL protocols made available in Python's ssl module for outgoing HTTPS connections:

from requests_toolbelt import SSLAdapter
import requests
import ssl

s = requests.Session()
s.mount('https://', SSLAdapter(ssl.PROTOCOL_TLSv1))

cookies/ForgetfulCookieJar

The ForgetfulCookieJar prevents a particular requests session from storing cookies:

from requests_toolbelt.cookies.forgetful import ForgetfulCookieJar

session = requests.Session()
session.cookies = ForgetfulCookieJar()

Contributing

Please read the suggested workflow for contributing to this project.

Please report any bugs on the issue tracker

Comments
  • GuessAuth

    GuessAuth

    See https://github.com/kennethreitz/requests/issues/2006

    This currently breaks and i can't explain how it breaks since the same code is used in requests.

    opened by untitaker 42
  • Add AppEngineAdapter for GAE users

    Add AppEngineAdapter for GAE users

    At the moment this will not work because the AppEngineManager does not have the same signature as urllib3's PoolManager.

    Related to: https://github.com/kennethreitz/requests/issues/1905#issuecomment-159380704

    cc @Lukasa @shazow @jonparrott @webmaven @jacobg

    ready 
    opened by sigmavirus24 28
  • Adds a monkeypatch() function that ensures all Sessions use the AppEngineAdapter compatibility class.

    Adds a monkeypatch() function that ensures all Sessions use the AppEngineAdapter compatibility class.

    The goal here is to allow the following code to work:

    from requests_toolbelt.adapters import appengine
    appengine.monkeypatch()
    

    ...and then run all your libraries that have dependencies on requests and sessions buried deep inside, all working properly with URLFetch on Appengine.

    The code doesn't work as-is, as it depends on the following change: https://github.com/shazow/urllib3/pull/763 ...which then would probably need to be released and ported into the requests library as well...not sure how long that takes or what the process is.

    This also is tangentially related to the following issues relating to getting requests playing nicely with AppEngine's URLFetch: https://github.com/kennethreitz/requests/issues/1905 https://github.com/shazow/urllib3/issues/618

    opened by mikelambert 24
  • Disable cert validation when monkeypatching AppEngine

    Disable cert validation when monkeypatching AppEngine

    We have a use case for this in which, due to an issue with AppEngine's dev_appserver.py, certain certificates will fail validation; more on that issue here. The only real workaround when using the emulator is to disable certificate validation.

    Because this is a large code base, in which requests is used both explicitly and within third-party libraries, we are using the monkeypatch; however, to my eye there's no way to set the validate_certificate kwarg in the constructor for AppEngineAdapter to True when monkeypatching; if we could, it would serve our use case by allowing global disabling of certificate validation when a local environment is detected.

    This creates a new AppEngineInsecureAdapter in which cert validation is disabled by default and adds a kwarg to monkeypatch() to trigger its use. Initially I tried to simply make the monkeypatch set partial(AppEngineAdapter, validate_certificate=False) but that was a bit naive as the adapter is not just instantiated.

    opened by bendemaree 20
  • Add HttpFileWrapper as a separate module

    Add HttpFileWrapper as a separate module

    This just provides a basic way to wrap a file object so you can force a specific amount of data to be returned at a minimum.

    So far, basic tests at the interactive console have shown it to work as expected. This still needs real tests, however.

    Closes #75

    Enhancement 
    opened by sigmavirus24 18
  • Allow files as input for StreamingIterator

    Allow files as input for StreamingIterator

    Fixes #38

    The usecase for this is unsized file-like objects like sys.stdin.

    This functionality could've been implemented as a classmethod or helper function. However, when using StreamingIterator with file-like objects, it would've been quite inefficient to

    1.) Read data from the underlying file at a fixed chunk size for the iterator 2.) Then maintain a buffer for the file-like interface provided on top of that iterator

    Enhancement 
    opened by untitaker 17
  • Add pkcs12 support

    Add pkcs12 support

    An adapter to be used in cases where a user needs to authenticate to a service using a .p12/.pfx certificate and it is not desirable or feasible to first convert that certificate into a .pem format. Born out of a use case in which the existence of temporary files containing cert data were deemed too great a risk.

    opened by rashley-iqt 15
  • MultipartEncoder.read() returns less than 8192 bytes even when more bytes are left

    MultipartEncoder.read() returns less than 8192 bytes even when more bytes are left

    Per investigation in: https://github.com/mitsuhiko/flask/issues/1025

    "MultipartEncoder.read() is completely broken - read(8192) (default blocksize) returns less than 8192 bytes even though there is more left."

    opened by SimplicityGuy 14
  • Add support for BasedSession

    Add support for BasedSession

    Following on the suggestion in requests, I've implemented the suggestion. Please let me know if there's anything more you'd like to see to get this accepted. I use this technique all the time and find the code copy/pasted in several places and I'd like to consolidate it somewhere.

    Thanks.

    opened by jaraco 13
  • App Engine: set total timeout so urllib3 passes it to urlfetch

    App Engine: set total timeout so urllib3 passes it to urlfetch

    Closes #145.

    Another plausibe alternative here would be to change this line in urllib3's App Engine adapter from return timeout.total to something like return timeout.total or timeout._read or timeout._connect. I'm not sure of the semantics or conventions for requests vs urllib3, so I'll defer to you all.

    cc @mikelambert

    opened by snarfed 13
  • unhashable type: 'RequestsCookieJar' only on first run

    unhashable type: 'RequestsCookieJar' only on first run

    Edit: I realize this may have nothing to do with requests-toolbelt, it's just the only lead I have at the moment.

    This is an odd issue that I just discovered with PRAW's testing suite as travis recently started failing almost all requests. I have narrowed the issue down to it failing only when .eggs/requests_toolbelt-0.4.0-py3.4.egg does not previously exist. On subsequent runs, when that file already exists, the tests pass just fine.

    Below is a sample run starting with (1) it working, (2) removing the egg and observing the failure, (3) running again and seeing the success:

    (tmp)[email protected]:praw (updates)$ python setup.py test -s tests.test_wiki_page.WikiPageTests.test_revision_by
    running test
    Searching for mock>=1.0.0
    Best match: mock 1.0.1
    Processing mock-1.0.1-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/mock-1.0.1-py3.4.egg
    Searching for betamax-matchers>=0.2.0
    Best match: betamax-matchers 0.2.0
    Processing betamax_matchers-0.2.0-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax_matchers-0.2.0-py3.4.egg
    Searching for betamax>=0.4.2
    Best match: betamax 0.4.2
    Processing betamax-0.4.2-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax-0.4.2-py3.4.egg
    Searching for requests-toolbelt>=0.4.0
    Best match: requests-toolbelt 0.4.0
    Processing requests_toolbelt-0.4.0-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/requests_toolbelt-0.4.0-py3.4.egg
    running egg_info
    writing entry points to praw.egg-info/entry_points.txt
    writing dependency_links to praw.egg-info/dependency_links.txt
    writing requirements to praw.egg-info/requires.txt
    writing praw.egg-info/PKG-INFO
    writing top-level names to praw.egg-info/top_level.txt
    reading manifest file 'praw.egg-info/SOURCES.txt'
    reading manifest template 'MANIFEST.in'
    warning: no files found matching '*' under directory 'praw/tests/files'
    writing manifest file 'praw.egg-info/SOURCES.txt'
    running build_ext
    test_revision_by (tests.test_wiki_page.WikiPageTests) ... ok
    
    ----------------------------------------------------------------------
    Ran 1 test in 0.009s
    
    OK
    (tmp)[email protected]:praw (updates)$ rm .eggs/requests_toolbelt-0.4.0-py3.4.egg 
    (tmp)[email protected]:praw (updates)$ python setup.py test -s tests.test_wiki_page.WikiPageTests.test_revision_by
    running test
    Searching for mock>=1.0.0
    Best match: mock 1.0.1
    Processing mock-1.0.1-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/mock-1.0.1-py3.4.egg
    Searching for betamax-matchers>=0.2.0
    Best match: betamax-matchers 0.2.0
    Processing betamax_matchers-0.2.0-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax_matchers-0.2.0-py3.4.egg
    Searching for betamax>=0.4.2
    Best match: betamax 0.4.2
    Processing betamax-0.4.2-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax-0.4.2-py3.4.egg
    Searching for requests-toolbelt>=0.4.0
    Reading https://pypi.python.org/simple/requests-toolbelt/
    Best match: requests-toolbelt 0.4.0
    Downloading https://pypi.python.org/packages/source/r/requests-toolbelt/requests-toolbelt-0.4.0.tar.gz#md5=2278d650faadf181dd180682591e5926
    Processing requests-toolbelt-0.4.0.tar.gz
    Writing /var/folders/zq/1jxg9xbx211_syhlv6cl2jq40000gn/T/easy_install-kowpbdbs/requests-toolbelt-0.4.0/setup.cfg
    Running requests-toolbelt-0.4.0/setup.py -q bdist_egg --dist-dir /var/folders/zq/1jxg9xbx211_syhlv6cl2jq40000gn/T/easy_install-kowpbdbs/requests-toolbelt-0.4.0/egg-dist-tmp-avhtypx4
    no previously-included directories found matching '*.pyc'
    warning: manifest_maker: MANIFEST.in, line 6: 'recursive-include' expects <dir> <pattern1> <pattern2> ...
    
    warning: manifest_maker: MANIFEST.in, line 7: 'recursive-include' expects <dir> <pattern1> <pattern2> ...
    
    no previously-included directories found matching 'docs/_build'
    zip_safe flag not set; analyzing archive contents...
    Copying requests_toolbelt-0.4.0-py3.4.egg to /Users/bboe/src/praw-dev/praw/.eggs
    
    Installed /Users/bboe/src/praw-dev/praw/.eggs/requests_toolbelt-0.4.0-py3.4.egg
    running egg_info
    writing top-level names to praw.egg-info/top_level.txt
    writing requirements to praw.egg-info/requires.txt
    writing entry points to praw.egg-info/entry_points.txt
    writing dependency_links to praw.egg-info/dependency_links.txt
    writing praw.egg-info/PKG-INFO
    reading manifest file 'praw.egg-info/SOURCES.txt'
    reading manifest template 'MANIFEST.in'
    warning: no files found matching '*' under directory 'praw/tests/files'
    writing manifest file 'praw.egg-info/SOURCES.txt'
    running build_ext
    test_revision_by (tests.test_wiki_page.WikiPageTests) ... ERROR
    
    ======================================================================
    ERROR: test_revision_by (tests.test_wiki_page.WikiPageTests)
    ----------------------------------------------------------------------
    Traceback (most recent call last):
      File "/Users/bboe/src/praw-dev/praw/tests/helper.py", line 120, in betamax_function
        return function(obj)
      File "/Users/bboe/src/praw-dev/praw/tests/test_wiki_page.py", line 64, in test_revision_by
        self.subreddit.get_wiki_pages()))
      File "/Users/bboe/src/praw-dev/praw/praw/decorators.py", line 60, in wrapped
        return function(self.reddit_session, self, *args, **kwargs)
      File "/Users/bboe/src/praw-dev/praw/praw/decorators.py", line 345, in wrapped
        return function(cls, *args, **kwargs)
      File "/Users/bboe/src/praw-dev/praw/praw/__init__.py", line 1052, in get_wiki_pages
        six.text_type(subreddit))
      File "/Users/bboe/src/praw-dev/praw/praw/decorators.py", line 170, in wrapped
        return_value = function(reddit_session, *args, **kwargs)
      File "/Users/bboe/src/praw-dev/praw/praw/__init__.py", line 569, in request_json
        retry_on_error=retry_on_error)
      File "/Users/bboe/src/praw-dev/praw/praw/__init__.py", line 413, in _request
        response = handle_redirect()
      File "/Users/bboe/src/praw-dev/praw/praw/__init__.py", line 383, in handle_redirect
        timeout=timeout, **kwargs)
      File "/Users/bboe/src/praw-dev/praw/praw/handlers.py", line 138, in wrapped
        if _cache_key in cls.cache:
    TypeError: unhashable type: 'RequestsCookieJar'
    
    ----------------------------------------------------------------------
    Ran 1 test in 0.007s
    
    FAILED (errors=1)
    (tmp)[email protected]:praw (updates)$ python setup.py test -s tests.test_wiki_page.WikiPageTests.test_revision_by
    running test
    Searching for mock>=1.0.0
    Best match: mock 1.0.1
    Processing mock-1.0.1-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/mock-1.0.1-py3.4.egg
    Searching for betamax-matchers>=0.2.0
    Best match: betamax-matchers 0.2.0
    Processing betamax_matchers-0.2.0-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax_matchers-0.2.0-py3.4.egg
    Searching for betamax>=0.4.2
    Best match: betamax 0.4.2
    Processing betamax-0.4.2-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/betamax-0.4.2-py3.4.egg
    Searching for requests-toolbelt>=0.4.0
    Best match: requests-toolbelt 0.4.0
    Processing requests_toolbelt-0.4.0-py3.4.egg
    
    Using /Users/bboe/src/praw-dev/praw/.eggs/requests_toolbelt-0.4.0-py3.4.egg
    running egg_info
    writing requirements to praw.egg-info/requires.txt
    writing entry points to praw.egg-info/entry_points.txt
    writing dependency_links to praw.egg-info/dependency_links.txt
    writing top-level names to praw.egg-info/top_level.txt
    writing praw.egg-info/PKG-INFO
    reading manifest file 'praw.egg-info/SOURCES.txt'
    reading manifest template 'MANIFEST.in'
    warning: no files found matching '*' under directory 'praw/tests/files'
    writing manifest file 'praw.egg-info/SOURCES.txt'
    running build_ext
    test_revision_by (tests.test_wiki_page.WikiPageTests) ... ok
    
    ----------------------------------------------------------------------
    Ran 1 test in 0.009s
    
    OK
    

    I really have no idea what to make of this at the moment. Any thoughts? Thanks!

    opened by bboe 13
  • MultipartEncoder can't handle mmap.mmap object correctly

    MultipartEncoder can't handle mmap.mmap object correctly

    MultipartEncoder can't handle mmap.mmap object correctly.

    MultipartEncoder can't handle mmap.mmap object correctly. For example,

    from requests_toolbelt import MultipartEncoder
    import mmap
    
    m = MultipartEncoder(
        fields={
                'field': ('filename', mmap.mmap(-1, 10))}
        )
    
    m.read()
    

    it will fall into an infinity loop in m.read().

    I review those source code, find that the coerce_data function did't consider about the mmap.mmap object. Then the total_len function work different from other io object (always return the origin size). Then the bytes_left_to_write function always return True, fall into the infinity loop.

    In my opinion, judge the mmap.mmap object by isinstance(data, mmap.mmap) or hasattr(data, 'madvise'), or other method?

    Now, I fix my code by manually wrapped with FileWrapper, as

    from requests_toolbelt.multipart.encoder import MultipartEncoder, FileWrapper
    import mmap
    
    m = MultipartEncoder(
        fields={
                'field': ('filename', FileWrapper(mmap.mmap(-1, 10)))}
        )
    
    m.read()
    

    I have little experience to pull requests. Could someone fix it from this package?

    opened by yx1994 1
  • Sending a zero-part multipart document has an uncaught exception; `.parts` should be an empty list instead

    Sending a zero-part multipart document has an uncaught exception; `.parts` should be an empty list instead

    If returning a list-like object, MarkLogic can return a multipart document, one part per returned value. For an empty list, it returns zero parts (i.e. no bytes). Requests Toolbelt crashes in this instance, because the assumption that a content type must exist at all is violated.

    File /usr/local/lib/python3.9/site-packages/caselawclient/Client.py:29 in decode_multipart
    multipart_data = decoder.MultipartDecoder.from_response(response)
    File /usr/local/lib/python3.9/site-packages/requests_toolbelt/multipart/decoder.py:156 in from_response
    return cls(content, content_type, encoding)
    File /usr/local/lib/python3.9/site-packages/requests_toolbelt/multipart/decoder.py:110 in __init__
    self._find_boundary()
    File /usr/local/lib/python3.9/site-packages/requests_toolbelt/multipart/decoder.py:114 in _find_boundary
    ct_info = tuple(x.strip() for x in self.content_type.split(';'))
    AttributeError: 'NoneType' object has no attribute 'split'
    

    We got around this by hardcoding that no content means zero items returned, but this feels like a thing worth solving at the library level. our workaround: https://github.com/nationalarchives/ds-caselaw-custom-api-client/pull/99/files

    opened by dragon-dxw 1
  • Issue341

    Issue341

    I added documentation for the multipart.decoder.MultipartDecoder class. Created a new page on the documentation called "Receiveing and Handling Multipart Data". I added detailed instruction on how to use the parameters for the base initialization of the class.

    On the basic usage examples, I inverted it to show the base initialization first, since this is a more generalized way of using this class. The .from_response method is for a specific use case, so I put it in the end.

    opened by atiagosoares 0
  • There seems to be no documentation on the multipart submodule, or at least on the MultipartDecoder class

    There seems to be no documentation on the multipart submodule, or at least on the MultipartDecoder class

    There isn't a documentation page for the multipart submodule. The MultipartEncoder class is mentioned at times, but I didn't find any mentions to the MultipartDecoder class.

    I have the intentions of adding this documentation myself. Will add a new page called "multipart", with the usage guide for both classes mentioned above. Opening the issue to let maintainer know I'm working on this.

    opened by atiagosoares 3
  • SourceAddressAdapter with certain port should be stricter with pool size and blocking?

    SourceAddressAdapter with certain port should be stricter with pool size and blocking?

    As far as I understand, if SourceAddressAdapter is used with an (IP, PORT) tuple and the port is other than 0, there is no way this adapter can work without passing

    pool_connections=1,
    pool_maxsize=1,
    pool_block=True,
    

    to the underlying HTTPAdapter.

    Shouldn't this either

    • be forced by the implementation
    • raise a warning if this exact values are not specified by the user
    • write more documentation about it

    ?

    opened by n1ngu 0
Releases(0.10.1)
An interactive command-line HTTP and API testing client built on top of HTTPie featuring autocomplete, syntax highlighting, and more. https://twitter.com/httpie

HTTP Prompt HTTP Prompt is an interactive command-line HTTP client featuring autocomplete and syntax highlighting, built on HTTPie and prompt_toolkit.

HTTPie 8.6k Dec 31, 2022
suite de mocks http em json

Ritchie Formula Repo Documentation Contribute to the Ritchie community This repository contains rit formulas which can be executed by the ritchie-cli.

Kaio Fábio Prates Prudêncio 1 Nov 01, 2021
Python HTTP library with thread-safe connection pooling, file post support, user friendly, and more.

urllib3 is a powerful, user-friendly HTTP client for Python. Much of the Python ecosystem already uses urllib3 and you should too. urllib3 brings many

urllib3 3.2k Dec 29, 2022
curl statistics made simple

httpstat httpstat visualizes curl(1) statistics in a way of beauty and clarity. It is a single file 🌟 Python script that has no dependency 👏 and is

Xiao Meng 5.3k Jan 04, 2023
Small, fast HTTP client library for Python. Features persistent connections, cache, and Google App Engine support. Originally written by Joe Gregorio, now supported by community.

Introduction httplib2 is a comprehensive HTTP client library, httplib2.py supports many features left out of other HTTP libraries. HTTP and HTTPS HTTP

457 Dec 10, 2022
HackerNews digest using GitHub actions

HackerNews Digest This script makes use of GitHub actions to send daily newsletters with the top 10 posts from HackerNews of the previous day. How to

Rajkumar S 3 Jan 19, 2022
Asynchronous Python HTTP Requests for Humans using Futures

Asynchronous Python HTTP Requests for Humans Small add-on for the python requests http library. Makes use of python 3.2's concurrent.futures or the ba

Ross McFarland 2k Dec 30, 2022
Python requests like API built on top of Twisted's HTTP client.

treq: High-level Twisted HTTP Client API treq is an HTTP library inspired by requests but written on top of Twisted's Agents. It provides a simple, hi

Twisted Matrix Labs 553 Dec 18, 2022
Asynchronous Python HTTP Requests for Humans using twisted

Asynchronous Python HTTP Requests for Humans Small add-on for the python requests http library. Makes use twisted's ThreadPool, so that the requests'A

Pierre Tardy 32 Oct 27, 2021
Requests + Gevent = <3

GRequests: Asynchronous Requests GRequests allows you to use Requests with Gevent to make asynchronous HTTP Requests easily. Note: You should probably

Spencer Phillip Young 4.2k Dec 30, 2022
💡Python package for HTTP/1.1 style headers. Parse headers to objects. Most advanced available structure for http headers.

HTTP Headers, the Complete Toolkit 🧰 Object-oriented headers. Kind of structured headers. ❓ Why No matter if you are currently dealing with code usin

TAHRI Ahmed R. 103 Dec 02, 2022
Aiosonic - lightweight Python asyncio http client

aiosonic - lightweight Python asyncio http client Very fast, lightweight Python asyncio http client Here is some documentation. There is a performance

Johanderson Mogollon 93 Jan 06, 2023
Pretty fast mass-dmer with multiple tokens support made with python requests

mass-dm-requests - Little preview of the Logger and the Spammer Features Logging User IDS Sending DMs (Embeds are supported) to the logged IDs Includi

karma.meme 14 Nov 18, 2022
r - a small subset of Python Requests

r a small subset of Python Requests a few years ago, when I was first learning Python and looking for http functionality, i found the batteries-includ

Gabriel Sroka 4 Dec 15, 2022
Script to automate PUT HTTP method exploitation to get shell.

Script to automate PUT HTTP method exploitation to get shell.

devploit 116 Nov 10, 2022
PycURL - Python interface to libcurl

PycURL -- A Python Interface To The cURL library PycURL is a Python interface to libcurl, the multiprotocol file transfer library. Similarly to the ur

PycURL 933 Jan 09, 2023
Fast HTTP parser

httptools is a Python binding for the nodejs HTTP parser. The package is available on PyPI: pip install httptools. APIs httptools contains two classes

magicstack 1.1k Jan 07, 2023
Probe and discover HTTP pathname using brute-force methodology and filtered by specific word or 2 words at once

pathprober Probe and discover HTTP pathname using brute-force methodology and filtered by specific word or 2 words at once. Purpose Brute-forcing webs

NFA 41 Jul 06, 2022
Screaming-fast Python 3.5+ HTTP toolkit integrated with pipelining HTTP server based on uvloop and picohttpparser.

Screaming-fast Python 3.5+ HTTP toolkit integrated with pipelining HTTP server based on uvloop and picohttpparser.

Paweł Piotr Przeradowski 8.6k Jan 04, 2023
Python HTTP library with thread-safe connection pooling, file post support, user friendly, and more.

urllib3 is a powerful, user-friendly HTTP client for Python. Much of the Python ecosystem already uses urllib3 and you should too. urllib3 brings many

urllib3 3.2k Jan 02, 2023