|
|
@ -29,8 +29,12 @@ import zlib
|
|
|
|
from email.message import Message
|
|
|
|
from email.message import Message
|
|
|
|
from http.cookiejar import CookieJar
|
|
|
|
from http.cookiejar import CookieJar
|
|
|
|
|
|
|
|
|
|
|
|
from test.conftest import validate_and_send
|
|
|
|
from test.helper import (
|
|
|
|
from test.helper import FakeYDL, http_server_port, verify_address_availability
|
|
|
|
FakeYDL,
|
|
|
|
|
|
|
|
http_server_port,
|
|
|
|
|
|
|
|
validate_and_send,
|
|
|
|
|
|
|
|
verify_address_availability,
|
|
|
|
|
|
|
|
)
|
|
|
|
from yt_dlp.cookies import YoutubeDLCookieJar
|
|
|
|
from yt_dlp.cookies import YoutubeDLCookieJar
|
|
|
|
from yt_dlp.dependencies import brotli, curl_cffi, requests, urllib3
|
|
|
|
from yt_dlp.dependencies import brotli, curl_cffi, requests, urllib3
|
|
|
|
from yt_dlp.networking import (
|
|
|
|
from yt_dlp.networking import (
|
|
|
@ -64,21 +68,6 @@ from yt_dlp.utils.networking import HTTPHeaderDict, std_headers
|
|
|
|
TEST_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
|
|
TEST_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _build_proxy_handler(name):
|
|
|
|
|
|
|
|
class HTTPTestRequestHandler(http.server.BaseHTTPRequestHandler):
|
|
|
|
|
|
|
|
proxy_name = name
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def log_message(self, format, *args):
|
|
|
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def do_GET(self):
|
|
|
|
|
|
|
|
self.send_response(200)
|
|
|
|
|
|
|
|
self.send_header('Content-Type', 'text/plain; charset=utf-8')
|
|
|
|
|
|
|
|
self.end_headers()
|
|
|
|
|
|
|
|
self.wfile.write(f'{self.proxy_name}: {self.path}'.encode())
|
|
|
|
|
|
|
|
return HTTPTestRequestHandler
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class HTTPTestRequestHandler(http.server.BaseHTTPRequestHandler):
|
|
|
|
class HTTPTestRequestHandler(http.server.BaseHTTPRequestHandler):
|
|
|
|
protocol_version = 'HTTP/1.1'
|
|
|
|
protocol_version = 'HTTP/1.1'
|
|
|
|
default_request_version = 'HTTP/1.1'
|
|
|
|
default_request_version = 'HTTP/1.1'
|
|
|
@ -319,8 +308,9 @@ class TestRequestHandlerBase:
|
|
|
|
cls.https_server_thread.start()
|
|
|
|
cls.https_server_thread.start()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_verify_cert(self, handler):
|
|
|
|
def test_verify_cert(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
with pytest.raises(CertificateVerifyError):
|
|
|
|
with pytest.raises(CertificateVerifyError):
|
|
|
@ -331,7 +321,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert r.status == 200
|
|
|
|
assert r.status == 200
|
|
|
|
r.close()
|
|
|
|
r.close()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_ssl_error(self, handler):
|
|
|
|
def test_ssl_error(self, handler):
|
|
|
|
# HTTPS server with too old TLS version
|
|
|
|
# HTTPS server with too old TLS version
|
|
|
|
# XXX: is there a better way to test this than to create a new server?
|
|
|
|
# XXX: is there a better way to test this than to create a new server?
|
|
|
@ -349,7 +338,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{https_port}/headers'))
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{https_port}/headers'))
|
|
|
|
assert not issubclass(exc_info.type, CertificateVerifyError)
|
|
|
|
assert not issubclass(exc_info.type, CertificateVerifyError)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_percent_encode(self, handler):
|
|
|
|
def test_percent_encode(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
# Unicode characters should be encoded with uppercase percent-encoding
|
|
|
|
# Unicode characters should be encoded with uppercase percent-encoding
|
|
|
@ -361,7 +349,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.status == 200
|
|
|
|
assert res.status == 200
|
|
|
|
res.close()
|
|
|
|
res.close()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('path', [
|
|
|
|
@pytest.mark.parametrize('path', [
|
|
|
|
'/a/b/./../../headers',
|
|
|
|
'/a/b/./../../headers',
|
|
|
|
'/redirect_dotsegments',
|
|
|
|
'/redirect_dotsegments',
|
|
|
@ -377,15 +364,13 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.url == f'http://127.0.0.1:{self.http_port}/headers'
|
|
|
|
assert res.url == f'http://127.0.0.1:{self.http_port}/headers'
|
|
|
|
res.close()
|
|
|
|
res.close()
|
|
|
|
|
|
|
|
|
|
|
|
# Not supported by CurlCFFI (non-standard)
|
|
|
|
@pytest.mark.skip_handler('CurlCFFI', 'not supported by curl-cffi (non-standard)')
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests'], indirect=True)
|
|
|
|
|
|
|
|
def test_unicode_path_redirection(self, handler):
|
|
|
|
def test_unicode_path_redirection(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
r = validate_and_send(rh, Request(f'http://127.0.0.1:{self.http_port}/302-non-ascii-redirect'))
|
|
|
|
r = validate_and_send(rh, Request(f'http://127.0.0.1:{self.http_port}/302-non-ascii-redirect'))
|
|
|
|
assert r.url == f'http://127.0.0.1:{self.http_port}/%E4%B8%AD%E6%96%87.html'
|
|
|
|
assert r.url == f'http://127.0.0.1:{self.http_port}/%E4%B8%AD%E6%96%87.html'
|
|
|
|
r.close()
|
|
|
|
r.close()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_raise_http_error(self, handler):
|
|
|
|
def test_raise_http_error(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
for bad_status in (400, 500, 599, 302):
|
|
|
|
for bad_status in (400, 500, 599, 302):
|
|
|
@ -395,7 +380,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
# Should not raise an error
|
|
|
|
# Should not raise an error
|
|
|
|
validate_and_send(rh, Request('http://127.0.0.1:%d/gen_200' % self.http_port)).close()
|
|
|
|
validate_and_send(rh, Request('http://127.0.0.1:%d/gen_200' % self.http_port)).close()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_response_url(self, handler):
|
|
|
|
def test_response_url(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
# Response url should be that of the last url in redirect chain
|
|
|
|
# Response url should be that of the last url in redirect chain
|
|
|
@ -407,7 +391,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
res2.close()
|
|
|
|
res2.close()
|
|
|
|
|
|
|
|
|
|
|
|
# Covers some basic cases we expect some level of consistency between request handlers for
|
|
|
|
# Covers some basic cases we expect some level of consistency between request handlers for
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('redirect_status,method,expected', [
|
|
|
|
@pytest.mark.parametrize('redirect_status,method,expected', [
|
|
|
|
# A 303 must either use GET or HEAD for subsequent request
|
|
|
|
# A 303 must either use GET or HEAD for subsequent request
|
|
|
|
(303, 'POST', ('', 'GET', False)),
|
|
|
|
(303, 'POST', ('', 'GET', False)),
|
|
|
@ -449,7 +432,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert expected[1] == res.headers.get('method')
|
|
|
|
assert expected[1] == res.headers.get('method')
|
|
|
|
assert expected[2] == ('content-length' in headers.decode().lower())
|
|
|
|
assert expected[2] == ('content-length' in headers.decode().lower())
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_request_cookie_header(self, handler):
|
|
|
|
def test_request_cookie_header(self, handler):
|
|
|
|
# We should accept a Cookie header being passed as in normal headers and handle it appropriately.
|
|
|
|
# We should accept a Cookie header being passed as in normal headers and handle it appropriately.
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
@ -482,19 +464,16 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert b'cookie: test=ytdlp' not in data.lower()
|
|
|
|
assert b'cookie: test=ytdlp' not in data.lower()
|
|
|
|
assert b'cookie: test=test3' in data.lower()
|
|
|
|
assert b'cookie: test=test3' in data.lower()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_redirect_loop(self, handler):
|
|
|
|
def test_redirect_loop(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
with pytest.raises(HTTPError, match='redirect loop'):
|
|
|
|
with pytest.raises(HTTPError, match='redirect loop'):
|
|
|
|
validate_and_send(rh, Request(f'http://127.0.0.1:{self.http_port}/redirect_loop'))
|
|
|
|
validate_and_send(rh, Request(f'http://127.0.0.1:{self.http_port}/redirect_loop'))
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_incompleteread(self, handler):
|
|
|
|
def test_incompleteread(self, handler):
|
|
|
|
with handler(timeout=2) as rh:
|
|
|
|
with handler(timeout=2) as rh:
|
|
|
|
with pytest.raises(IncompleteRead, match='13 bytes read, 234221 more expected'):
|
|
|
|
with pytest.raises(IncompleteRead, match='13 bytes read, 234221 more expected'):
|
|
|
|
validate_and_send(rh, Request('http://127.0.0.1:%d/incompleteread' % self.http_port)).read()
|
|
|
|
validate_and_send(rh, Request('http://127.0.0.1:%d/incompleteread' % self.http_port)).read()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_cookies(self, handler):
|
|
|
|
def test_cookies(self, handler):
|
|
|
|
cookiejar = YoutubeDLCookieJar()
|
|
|
|
cookiejar = YoutubeDLCookieJar()
|
|
|
|
cookiejar.set_cookie(http.cookiejar.Cookie(
|
|
|
|
cookiejar.set_cookie(http.cookiejar.Cookie(
|
|
|
@ -511,7 +490,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
rh, Request(f'http://127.0.0.1:{self.http_port}/headers', extensions={'cookiejar': cookiejar})).read()
|
|
|
|
rh, Request(f'http://127.0.0.1:{self.http_port}/headers', extensions={'cookiejar': cookiejar})).read()
|
|
|
|
assert b'cookie: test=ytdlp' in data.lower()
|
|
|
|
assert b'cookie: test=ytdlp' in data.lower()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_headers(self, handler):
|
|
|
|
def test_headers(self, handler):
|
|
|
|
|
|
|
|
|
|
|
|
with handler(headers=HTTPHeaderDict({'test1': 'test', 'test2': 'test2'})) as rh:
|
|
|
|
with handler(headers=HTTPHeaderDict({'test1': 'test', 'test2': 'test2'})) as rh:
|
|
|
@ -527,7 +505,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert b'test2: test2' not in data
|
|
|
|
assert b'test2: test2' not in data
|
|
|
|
assert b'test3: test3' in data
|
|
|
|
assert b'test3: test3' in data
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_read_timeout(self, handler):
|
|
|
|
def test_read_timeout(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
# Default timeout is 20 seconds, so this should go through
|
|
|
|
# Default timeout is 20 seconds, so this should go through
|
|
|
@ -543,7 +520,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
validate_and_send(
|
|
|
|
validate_and_send(
|
|
|
|
rh, Request(f'http://127.0.0.1:{self.http_port}/timeout_1', extensions={'timeout': 4}))
|
|
|
|
rh, Request(f'http://127.0.0.1:{self.http_port}/timeout_1', extensions={'timeout': 4}))
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_connect_timeout(self, handler):
|
|
|
|
def test_connect_timeout(self, handler):
|
|
|
|
# nothing should be listening on this port
|
|
|
|
# nothing should be listening on this port
|
|
|
|
connect_timeout_url = 'http://10.255.255.255'
|
|
|
|
connect_timeout_url = 'http://10.255.255.255'
|
|
|
@ -562,7 +538,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
rh, Request(connect_timeout_url, extensions={'timeout': 0.01}))
|
|
|
|
rh, Request(connect_timeout_url, extensions={'timeout': 0.01}))
|
|
|
|
assert 0.01 <= time.time() - now < 20
|
|
|
|
assert 0.01 <= time.time() - now < 20
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_source_address(self, handler):
|
|
|
|
def test_source_address(self, handler):
|
|
|
|
source_address = f'127.0.0.{random.randint(5, 255)}'
|
|
|
|
source_address = f'127.0.0.{random.randint(5, 255)}'
|
|
|
|
# on some systems these loopback addresses we need for testing may not be available
|
|
|
|
# on some systems these loopback addresses we need for testing may not be available
|
|
|
@ -574,13 +549,13 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert source_address == data
|
|
|
|
assert source_address == data
|
|
|
|
|
|
|
|
|
|
|
|
# Not supported by CurlCFFI
|
|
|
|
# Not supported by CurlCFFI
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests'], indirect=True)
|
|
|
|
@pytest.mark.skip_handler('CurlCFFI', 'not supported by curl-cffi')
|
|
|
|
def test_gzip_trailing_garbage(self, handler):
|
|
|
|
def test_gzip_trailing_garbage(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
data = validate_and_send(rh, Request(f'http://localhost:{self.http_port}/trailing_garbage')).read().decode()
|
|
|
|
data = validate_and_send(rh, Request(f'http://localhost:{self.http_port}/trailing_garbage')).read().decode()
|
|
|
|
assert data == '<html><video src="/vid.mp4" /></html>'
|
|
|
|
assert data == '<html><video src="/vid.mp4" /></html>'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests'], indirect=True)
|
|
|
|
@pytest.mark.skip_handler('CurlCFFI', 'not applicable to curl-cffi')
|
|
|
|
@pytest.mark.skipif(not brotli, reason='brotli support is not installed')
|
|
|
|
@pytest.mark.skipif(not brotli, reason='brotli support is not installed')
|
|
|
|
def test_brotli(self, handler):
|
|
|
|
def test_brotli(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
@ -591,7 +566,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.headers.get('Content-Encoding') == 'br'
|
|
|
|
assert res.headers.get('Content-Encoding') == 'br'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_deflate(self, handler):
|
|
|
|
def test_deflate(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
res = validate_and_send(
|
|
|
|
res = validate_and_send(
|
|
|
@ -601,7 +575,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.headers.get('Content-Encoding') == 'deflate'
|
|
|
|
assert res.headers.get('Content-Encoding') == 'deflate'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_gzip(self, handler):
|
|
|
|
def test_gzip(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
res = validate_and_send(
|
|
|
|
res = validate_and_send(
|
|
|
@ -611,7 +584,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.headers.get('Content-Encoding') == 'gzip'
|
|
|
|
assert res.headers.get('Content-Encoding') == 'gzip'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_multiple_encodings(self, handler):
|
|
|
|
def test_multiple_encodings(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
for pair in ('gzip,deflate', 'deflate, gzip', 'gzip, gzip', 'deflate, deflate'):
|
|
|
|
for pair in ('gzip,deflate', 'deflate, gzip', 'gzip, gzip', 'deflate, deflate'):
|
|
|
@ -622,8 +594,7 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.headers.get('Content-Encoding') == pair
|
|
|
|
assert res.headers.get('Content-Encoding') == pair
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
assert res.read() == b'<html><video src="/vid.mp4" /></html>'
|
|
|
|
|
|
|
|
|
|
|
|
# Not supported by curl_cffi
|
|
|
|
@pytest.mark.skip_handler('CurlCFFI', 'not supported by curl-cffi')
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests'], indirect=True)
|
|
|
|
|
|
|
|
def test_unsupported_encoding(self, handler):
|
|
|
|
def test_unsupported_encoding(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
res = validate_and_send(
|
|
|
|
res = validate_and_send(
|
|
|
@ -633,7 +604,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.headers.get('Content-Encoding') == 'unsupported'
|
|
|
|
assert res.headers.get('Content-Encoding') == 'unsupported'
|
|
|
|
assert res.read() == b'raw'
|
|
|
|
assert res.read() == b'raw'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_read(self, handler):
|
|
|
|
def test_read(self, handler):
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
|
res = validate_and_send(
|
|
|
|
res = validate_and_send(
|
|
|
@ -644,7 +614,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert res.read().decode().endswith('\n\n')
|
|
|
|
assert res.read().decode().endswith('\n\n')
|
|
|
|
assert res.read() == b''
|
|
|
|
assert res.read() == b''
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_request_disable_proxy(self, handler):
|
|
|
|
def test_request_disable_proxy(self, handler):
|
|
|
|
for proxy_proto in handler._SUPPORTED_PROXY_SCHEMES or ['http']:
|
|
|
|
for proxy_proto in handler._SUPPORTED_PROXY_SCHEMES or ['http']:
|
|
|
|
# Given the handler is configured with a proxy
|
|
|
|
# Given the handler is configured with a proxy
|
|
|
@ -656,7 +625,6 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
res.close()
|
|
|
|
res.close()
|
|
|
|
assert res.status == 200
|
|
|
|
assert res.status == 200
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
@pytest.mark.skip_handlers_if(
|
|
|
|
@pytest.mark.skip_handlers_if(
|
|
|
|
lambda _, handler: Features.NO_PROXY not in handler._SUPPORTED_FEATURES, 'handler does not support NO_PROXY')
|
|
|
|
lambda _, handler: Features.NO_PROXY not in handler._SUPPORTED_FEATURES, 'handler does not support NO_PROXY')
|
|
|
|
def test_noproxy(self, handler):
|
|
|
|
def test_noproxy(self, handler):
|
|
|
@ -671,24 +639,23 @@ class TestHTTPRequestHandler(TestRequestHandlerBase):
|
|
|
|
assert nop_response.status == 200
|
|
|
|
assert nop_response.status == 200
|
|
|
|
nop_response.close()
|
|
|
|
nop_response.close()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
@pytest.mark.skip_handlers_if(
|
|
|
|
@pytest.mark.skip_handlers_if(
|
|
|
|
lambda _, handler: Features.ALL_PROXY not in handler._SUPPORTED_FEATURES, 'handler does not support ALL_PROXY')
|
|
|
|
lambda _, handler: Features.ALL_PROXY not in handler._SUPPORTED_FEATURES, 'handler does not support ALL_PROXY')
|
|
|
|
def test_allproxy(self, handler):
|
|
|
|
def test_allproxy(self, handler):
|
|
|
|
# This is a bit of a hacky test, but it should be enough to check whether the handler is using the proxy.
|
|
|
|
# This is a bit of a hacky test, but it should be enough to check whether the handler is using the proxy.
|
|
|
|
# 0.1s might not be enough of a timeout if proxy is not used in all cases, but should still get failures.
|
|
|
|
# 0.1s might not be enough of a timeout if proxy is not used in all cases, but should still get failures.
|
|
|
|
with handler(proxies={'all': f'http://10.255.255.255'}, timeout=0.1) as rh:
|
|
|
|
with handler(proxies={'all': 'http://10.255.255.255'}, timeout=0.1) as rh:
|
|
|
|
with pytest.raises(TransportError):
|
|
|
|
with pytest.raises(TransportError):
|
|
|
|
validate_and_send(
|
|
|
|
validate_and_send(rh, Request(f'http://127.0.0.1:{self.http_port}/headers')).close()
|
|
|
|
rh, Request(f'http://127.0.0.1:{self.http_port}/headers')).close()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
with handler(timeout=0.1) as rh:
|
|
|
|
with handler(timeout=0.1) as rh:
|
|
|
|
with pytest.raises(TransportError):
|
|
|
|
with pytest.raises(TransportError):
|
|
|
|
validate_and_send(
|
|
|
|
validate_and_send(
|
|
|
|
rh, Request(
|
|
|
|
rh, Request(
|
|
|
|
f'http://127.0.0.1:{self.http_port}/headers', proxies={'all': f'http://10.255.255.255'})).close()
|
|
|
|
f'http://127.0.0.1:{self.http_port}/headers', proxies={'all': 'http://10.255.255.255'})).close()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
class TestClientCertificate:
|
|
|
|
class TestClientCertificate:
|
|
|
|
@classmethod
|
|
|
|
@classmethod
|
|
|
|
def setup_class(cls):
|
|
|
|
def setup_class(cls):
|
|
|
@ -715,27 +682,23 @@ class TestClientCertificate:
|
|
|
|
) as rh:
|
|
|
|
) as rh:
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{self.port}/video.html')).read().decode()
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{self.port}/video.html')).read().decode()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_certificate_combined_nopass(self, handler):
|
|
|
|
def test_certificate_combined_nopass(self, handler):
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'clientwithkey.crt'),
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'clientwithkey.crt'),
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_certificate_nocombined_nopass(self, handler):
|
|
|
|
def test_certificate_nocombined_nopass(self, handler):
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'client.crt'),
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'client.crt'),
|
|
|
|
'client_certificate_key': os.path.join(self.certdir, 'client.key'),
|
|
|
|
'client_certificate_key': os.path.join(self.certdir, 'client.key'),
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_certificate_combined_pass(self, handler):
|
|
|
|
def test_certificate_combined_pass(self, handler):
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'clientwithencryptedkey.crt'),
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'clientwithencryptedkey.crt'),
|
|
|
|
'client_certificate_password': 'foobar',
|
|
|
|
'client_certificate_password': 'foobar',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_certificate_nocombined_pass(self, handler):
|
|
|
|
def test_certificate_nocombined_pass(self, handler):
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
self._run_test(handler, client_cert={
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'client.crt'),
|
|
|
|
'client_certificate': os.path.join(self.certdir, 'client.crt'),
|
|
|
@ -775,8 +738,8 @@ class TestRequestHandlerMisc:
|
|
|
|
assert len(logging_handlers) == before_count
|
|
|
|
assert len(logging_handlers) == before_count
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib'], indirect=True)
|
|
|
|
class TestUrllibRequestHandler(TestRequestHandlerBase):
|
|
|
|
class TestUrllibRequestHandler(TestRequestHandlerBase):
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib'], indirect=True)
|
|
|
|
|
|
|
|
def test_file_urls(self, handler):
|
|
|
|
def test_file_urls(self, handler):
|
|
|
|
# See https://github.com/ytdl-org/youtube-dl/issues/8227
|
|
|
|
# See https://github.com/ytdl-org/youtube-dl/issues/8227
|
|
|
|
tf = tempfile.NamedTemporaryFile(delete=False)
|
|
|
|
tf = tempfile.NamedTemporaryFile(delete=False)
|
|
|
@ -798,7 +761,6 @@ class TestUrllibRequestHandler(TestRequestHandlerBase):
|
|
|
|
|
|
|
|
|
|
|
|
os.unlink(tf.name)
|
|
|
|
os.unlink(tf.name)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib'], indirect=True)
|
|
|
|
|
|
|
|
def test_http_error_returns_content(self, handler):
|
|
|
|
def test_http_error_returns_content(self, handler):
|
|
|
|
# urllib HTTPError will try close the underlying response if reference to the HTTPError object is lost
|
|
|
|
# urllib HTTPError will try close the underlying response if reference to the HTTPError object is lost
|
|
|
|
def get_response():
|
|
|
|
def get_response():
|
|
|
@ -811,7 +773,6 @@ class TestUrllibRequestHandler(TestRequestHandlerBase):
|
|
|
|
|
|
|
|
|
|
|
|
assert get_response().read() == b'<html></html>'
|
|
|
|
assert get_response().read() == b'<html></html>'
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib'], indirect=True)
|
|
|
|
|
|
|
|
def test_verify_cert_error_text(self, handler):
|
|
|
|
def test_verify_cert_error_text(self, handler):
|
|
|
|
# Check the output of the error message
|
|
|
|
# Check the output of the error message
|
|
|
|
with handler() as rh:
|
|
|
|
with handler() as rh:
|
|
|
@ -821,7 +782,6 @@ class TestUrllibRequestHandler(TestRequestHandlerBase):
|
|
|
|
):
|
|
|
|
):
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{self.https_port}/headers'))
|
|
|
|
validate_and_send(rh, Request(f'https://127.0.0.1:{self.https_port}/headers'))
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib'], indirect=True)
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('req,match,version_check', [
|
|
|
|
@pytest.mark.parametrize('req,match,version_check', [
|
|
|
|
# https://github.com/python/cpython/blob/987b712b4aeeece336eed24fcc87a950a756c3e2/Lib/http/client.py#L1256
|
|
|
|
# https://github.com/python/cpython/blob/987b712b4aeeece336eed24fcc87a950a756c3e2/Lib/http/client.py#L1256
|
|
|
|
# bpo-39603: Check implemented in 3.7.9+, 3.8.5+
|
|
|
|
# bpo-39603: Check implemented in 3.7.9+, 3.8.5+
|
|
|
@ -1153,7 +1113,7 @@ class TestRequestHandlerValidation:
|
|
|
|
]
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
PROXY_SCHEME_TESTS = [
|
|
|
|
PROXY_SCHEME_TESTS = [
|
|
|
|
# scheme, expected to fail
|
|
|
|
# proxy scheme, expected to fail
|
|
|
|
('Urllib', 'http', [
|
|
|
|
('Urllib', 'http', [
|
|
|
|
('http', False),
|
|
|
|
('http', False),
|
|
|
|
('https', UnsupportedRequest),
|
|
|
|
('https', UnsupportedRequest),
|
|
|
@ -1179,30 +1139,41 @@ class TestRequestHandlerValidation:
|
|
|
|
('socks5', False),
|
|
|
|
('socks5', False),
|
|
|
|
('socks5h', False),
|
|
|
|
('socks5h', False),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
|
|
|
|
('Websockets', 'ws', [
|
|
|
|
|
|
|
|
('http', UnsupportedRequest),
|
|
|
|
|
|
|
|
('https', UnsupportedRequest),
|
|
|
|
|
|
|
|
('socks4', False),
|
|
|
|
|
|
|
|
('socks4a', False),
|
|
|
|
|
|
|
|
('socks5', False),
|
|
|
|
|
|
|
|
('socks5h', False),
|
|
|
|
|
|
|
|
]),
|
|
|
|
(NoCheckRH, 'http', [('http', False)]),
|
|
|
|
(NoCheckRH, 'http', [('http', False)]),
|
|
|
|
(HTTPSupportedRH, 'http', [('http', UnsupportedRequest)]),
|
|
|
|
(HTTPSupportedRH, 'http', [('http', UnsupportedRequest)]),
|
|
|
|
('Websockets', 'ws', [('http', UnsupportedRequest)]),
|
|
|
|
|
|
|
|
(NoCheckRH, 'http', [('http', False)]),
|
|
|
|
(NoCheckRH, 'http', [('http', False)]),
|
|
|
|
(HTTPSupportedRH, 'http', [('http', UnsupportedRequest)]),
|
|
|
|
(HTTPSupportedRH, 'http', [('http', UnsupportedRequest)]),
|
|
|
|
]
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
PROXY_KEY_TESTS = [
|
|
|
|
PROXY_KEY_TESTS = [
|
|
|
|
# key, expected to fail
|
|
|
|
# proxy key, proxy scheme, expected to fail
|
|
|
|
('Urllib', [
|
|
|
|
('Urllib', 'http', [
|
|
|
|
('all', False),
|
|
|
|
('all', 'http', False),
|
|
|
|
('unrelated', False),
|
|
|
|
('unrelated', 'http', False),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
('Requests', [
|
|
|
|
('Requests', 'http', [
|
|
|
|
('all', False),
|
|
|
|
('all', 'http', False),
|
|
|
|
('unrelated', False),
|
|
|
|
('unrelated', 'http', False),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
('CurlCFFI', [
|
|
|
|
('CurlCFFI', 'http', [
|
|
|
|
('all', False),
|
|
|
|
('all', 'http', False),
|
|
|
|
('unrelated', False),
|
|
|
|
('unrelated', 'http', False),
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
(NoCheckRH, [('all', False)]),
|
|
|
|
('Websockets', 'ws', [
|
|
|
|
(HTTPSupportedRH, [('all', UnsupportedRequest)]),
|
|
|
|
('all', 'socks5', False),
|
|
|
|
(HTTPSupportedRH, [('no', UnsupportedRequest)]),
|
|
|
|
('unrelated', 'socks5', False),
|
|
|
|
|
|
|
|
]),
|
|
|
|
|
|
|
|
(NoCheckRH, 'http', [('all', 'http', False)]),
|
|
|
|
|
|
|
|
(HTTPSupportedRH, 'http', [('all', 'http', UnsupportedRequest)]),
|
|
|
|
|
|
|
|
(HTTPSupportedRH, 'http', [('no', 'http', UnsupportedRequest)]),
|
|
|
|
]
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
EXTENSION_TESTS = [
|
|
|
|
EXTENSION_TESTS = [
|
|
|
@ -1244,28 +1215,54 @@ class TestRequestHandlerValidation:
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
]
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,fail,scheme', [
|
|
|
|
|
|
|
|
('Urllib', False, 'http'),
|
|
|
|
|
|
|
|
('Requests', False, 'http'),
|
|
|
|
|
|
|
|
('CurlCFFI', False, 'http'),
|
|
|
|
|
|
|
|
('Websockets', False, 'ws')
|
|
|
|
|
|
|
|
], indirect=['handler'])
|
|
|
|
|
|
|
|
def test_no_proxy(self, handler, fail, scheme):
|
|
|
|
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://', proxies={'no': '127.0.0.1,github.com'}))
|
|
|
|
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://'), proxies={'no': '127.0.0.1,github.com'})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,scheme', [
|
|
|
|
|
|
|
|
('Urllib', 'http'),
|
|
|
|
|
|
|
|
(HTTPSupportedRH, 'http'),
|
|
|
|
|
|
|
|
('Requests', 'http'),
|
|
|
|
|
|
|
|
('CurlCFFI', 'http'),
|
|
|
|
|
|
|
|
('Websockets', 'ws')
|
|
|
|
|
|
|
|
], indirect=['handler'])
|
|
|
|
|
|
|
|
def test_empty_proxy(self, handler, scheme):
|
|
|
|
|
|
|
|
run_validation(handler, False, Request(f'{scheme}://', proxies={scheme: None}))
|
|
|
|
|
|
|
|
run_validation(handler, False, Request(f'{scheme}://'), proxies={scheme: None})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('proxy_url', ['//example.com', 'example.com', '127.0.0.1', '/a/b/c'])
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,scheme', [
|
|
|
|
|
|
|
|
('Urllib', 'http'),
|
|
|
|
|
|
|
|
(HTTPSupportedRH, 'http'),
|
|
|
|
|
|
|
|
('Requests', 'http'),
|
|
|
|
|
|
|
|
('CurlCFFI', 'http'),
|
|
|
|
|
|
|
|
('Websockets', 'ws')
|
|
|
|
|
|
|
|
], indirect=['handler'])
|
|
|
|
|
|
|
|
def test_invalid_proxy_url(self, handler, scheme, proxy_url):
|
|
|
|
|
|
|
|
run_validation(handler, UnsupportedRequest, Request(f'{scheme}://', proxies={scheme: proxy_url}))
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,scheme,fail,handler_kwargs', [
|
|
|
|
@pytest.mark.parametrize('handler,scheme,fail,handler_kwargs', [
|
|
|
|
(handler_tests[0], scheme, fail, handler_kwargs)
|
|
|
|
(handler_tests[0], scheme, fail, handler_kwargs)
|
|
|
|
for handler_tests in URL_SCHEME_TESTS
|
|
|
|
for handler_tests in URL_SCHEME_TESTS
|
|
|
|
for scheme, fail, handler_kwargs in handler_tests[1]
|
|
|
|
for scheme, fail, handler_kwargs in handler_tests[1]
|
|
|
|
|
|
|
|
|
|
|
|
], indirect=['handler'])
|
|
|
|
], indirect=['handler'])
|
|
|
|
def test_url_scheme(self, handler, scheme, fail, handler_kwargs):
|
|
|
|
def test_url_scheme(self, handler, scheme, fail, handler_kwargs):
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://'), **(handler_kwargs or {}))
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://'), **(handler_kwargs or {}))
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,fail', [('Urllib', False), ('Requests', False), ('CurlCFFI', False)], indirect=['handler'])
|
|
|
|
@pytest.mark.parametrize('handler,scheme,proxy_key,proxy_scheme,fail', [
|
|
|
|
def test_no_proxy(self, handler, fail):
|
|
|
|
(handler_tests[0], handler_tests[1], proxy_key, proxy_scheme, fail)
|
|
|
|
run_validation(handler, fail, Request('http://', proxies={'no': '127.0.0.1,github.com'}))
|
|
|
|
|
|
|
|
run_validation(handler, fail, Request('http://'), proxies={'no': '127.0.0.1,github.com'})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,proxy_key,fail', [
|
|
|
|
|
|
|
|
(handler_tests[0], proxy_key, fail)
|
|
|
|
|
|
|
|
for handler_tests in PROXY_KEY_TESTS
|
|
|
|
for handler_tests in PROXY_KEY_TESTS
|
|
|
|
for proxy_key, fail in handler_tests[1]
|
|
|
|
for proxy_key, proxy_scheme, fail in handler_tests[2]
|
|
|
|
], indirect=['handler'])
|
|
|
|
], indirect=['handler'])
|
|
|
|
def test_proxy_key(self, handler, proxy_key, fail):
|
|
|
|
def test_proxy_key(self, handler, scheme, proxy_key, proxy_scheme, fail):
|
|
|
|
run_validation(handler, fail, Request('http://', proxies={proxy_key: 'http://example.com'}))
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://', proxies={proxy_key: f'{proxy_scheme}://example.com'}))
|
|
|
|
run_validation(handler, fail, Request('http://'), proxies={proxy_key: 'http://example.com'})
|
|
|
|
run_validation(handler, fail, Request(f'{scheme}://'), proxies={proxy_key: f'{proxy_scheme}://example.com'})
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,req_scheme,scheme,fail', [
|
|
|
|
@pytest.mark.parametrize('handler,req_scheme,scheme,fail', [
|
|
|
|
(handler_tests[0], handler_tests[1], scheme, fail)
|
|
|
|
(handler_tests[0], handler_tests[1], scheme, fail)
|
|
|
@ -1276,16 +1273,6 @@ class TestRequestHandlerValidation:
|
|
|
|
run_validation(handler, fail, Request(f'{req_scheme}://', proxies={req_scheme: f'{scheme}://example.com'}))
|
|
|
|
run_validation(handler, fail, Request(f'{req_scheme}://', proxies={req_scheme: f'{scheme}://example.com'}))
|
|
|
|
run_validation(handler, fail, Request(f'{req_scheme}://'), proxies={req_scheme: f'{scheme}://example.com'})
|
|
|
|
run_validation(handler, fail, Request(f'{req_scheme}://'), proxies={req_scheme: f'{scheme}://example.com'})
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', HTTPSupportedRH, 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_empty_proxy(self, handler):
|
|
|
|
|
|
|
|
run_validation(handler, False, Request('http://', proxies={'http': None}))
|
|
|
|
|
|
|
|
run_validation(handler, False, Request('http://'), proxies={'http': None})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('proxy_url', ['//example.com', 'example.com', '127.0.0.1', '/a/b/c'])
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler', ['Urllib', 'Requests', 'CurlCFFI'], indirect=True)
|
|
|
|
|
|
|
|
def test_invalid_proxy_url(self, handler, proxy_url):
|
|
|
|
|
|
|
|
run_validation(handler, UnsupportedRequest, Request('http://', proxies={'http': proxy_url}))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('handler,scheme,extensions,fail', [
|
|
|
|
@pytest.mark.parametrize('handler,scheme,extensions,fail', [
|
|
|
|
(handler_tests[0], handler_tests[1], extensions, fail)
|
|
|
|
(handler_tests[0], handler_tests[1], extensions, fail)
|
|
|
|
for handler_tests in EXTENSION_TESTS
|
|
|
|
for handler_tests in EXTENSION_TESTS
|
|
|
|