add --source-address command-line option (closes #2206)

This commit is contained in:
Mike Fährmann 2022-01-20 23:16:00 +01:00
parent 698f35215e
commit de754590e0
No known key found for this signature in database
GPG Key ID: 5680CA389D365A88
3 changed files with 56 additions and 7 deletions

View File

@ -463,6 +463,20 @@ Description
otherwise ``http://`` is assumed. otherwise ``http://`` is assumed.
extractor.*.source-address
--------------------------
Type
* ``string``
* ``list`` with 1 ``string`` and 1 ``integer`` as elements
Example
* ``"192.168.178.20"``
* ``["192.168.178.20", 8080]``
Description
Client-side IP address to bind to.
| Can be either a simple ``string`` with just the local IP address
| or a ``list`` with IP and explicit port number as elements.
extractor.*.user-agent extractor.*.user-agent
---------------------- ----------------------
Type Type

View File

@ -220,6 +220,14 @@ class Extractor():
headers = session.headers headers = session.headers
headers.clear() headers.clear()
source_address = self.config("source-address")
if source_address:
if isinstance(source_address, str):
source_address = (source_address, 0)
else:
source_address = (source_address[0], source_address[1])
session.mount("http://", SourceAdapter(source_address))
browser = self.config("browser") or self.browser browser = self.config("browser") or self.browser
if browser and isinstance(browser, str): if browser and isinstance(browser, str):
browser, _, platform = browser.lower().partition(":") browser, _, platform = browser.lower().partition(":")
@ -235,10 +243,12 @@ class Extractor():
platform = "Macintosh; Intel Mac OS X 11.5" platform = "Macintosh; Intel Mac OS X 11.5"
if browser == "chrome": if browser == "chrome":
_emulate_browser_chrome(session, platform) _emulate_browser_chrome(session, platform, source_address)
else: else:
_emulate_browser_firefox(session, platform) _emulate_browser_firefox(session, platform, source_address)
else: else:
if source_address:
session.mount("https://", SourceAdapter(source_address))
headers["User-Agent"] = self.config("user-agent", ( headers["User-Agent"] = self.config("user-agent", (
"Mozilla/5.0 (Windows NT 10.0; Win64; x64; " "Mozilla/5.0 (Windows NT 10.0; Win64; x64; "
"rv:91.0) Gecko/20100101 Firefox/91.0")) "rv:91.0) Gecko/20100101 Firefox/91.0"))
@ -605,26 +615,44 @@ class BaseExtractor(Extractor):
) )
class SourceAdapter(HTTPAdapter):
def __init__(self, source_address):
self.source_address = source_address
HTTPAdapter.__init__(self)
def init_poolmanager(self, *args, **kwargs):
kwargs["source_address"] = self.source_address
return HTTPAdapter.init_poolmanager(self, *args, **kwargs)
def proxy_manager_for(self, *args, **kwargs):
kwargs["source_address"] = self.source_address
return HTTPAdapter.proxy_manager_for(self, *args, **kwargs)
class HTTPSAdapter(HTTPAdapter): class HTTPSAdapter(HTTPAdapter):
def __init__(self, ciphers): def __init__(self, ciphers, source_address=None):
context = self.ssl_context = ssl.create_default_context() context = self.ssl_context = ssl.create_default_context()
context.options |= (ssl.OP_NO_SSLv2 | ssl.OP_NO_SSLv3 | context.options |= (ssl.OP_NO_SSLv2 | ssl.OP_NO_SSLv3 |
ssl.OP_NO_TLSv1 | ssl.OP_NO_TLSv1_1) ssl.OP_NO_TLSv1 | ssl.OP_NO_TLSv1_1)
context.set_ecdh_curve("prime256v1") context.set_ecdh_curve("prime256v1")
context.set_ciphers(ciphers) context.set_ciphers(ciphers)
self.source_address = source_address
HTTPAdapter.__init__(self) HTTPAdapter.__init__(self)
def init_poolmanager(self, *args, **kwargs): def init_poolmanager(self, *args, **kwargs):
kwargs["ssl_context"] = self.ssl_context kwargs["ssl_context"] = self.ssl_context
kwargs["source_address"] = self.source_address
return HTTPAdapter.init_poolmanager(self, *args, **kwargs) return HTTPAdapter.init_poolmanager(self, *args, **kwargs)
def proxy_manager_for(self, *args, **kwargs): def proxy_manager_for(self, *args, **kwargs):
kwargs["ssl_context"] = self.ssl_context kwargs["ssl_context"] = self.ssl_context
kwargs["source_address"] = self.source_address
return HTTPAdapter.proxy_manager_for(self, *args, **kwargs) return HTTPAdapter.proxy_manager_for(self, *args, **kwargs)
def _emulate_browser_firefox(session, platform): def _emulate_browser_firefox(session, platform, source_address):
headers = session.headers headers = session.headers
headers["User-Agent"] = ("Mozilla/5.0 (" + platform + "; rv:91.0) " headers["User-Agent"] = ("Mozilla/5.0 (" + platform + "; rv:91.0) "
"Gecko/20100101 Firefox/91.0") "Gecko/20100101 Firefox/91.0")
@ -654,11 +682,12 @@ def _emulate_browser_firefox(session, platform):
"DHE-RSA-AES256-SHA:" "DHE-RSA-AES256-SHA:"
"AES128-SHA:" "AES128-SHA:"
"AES256-SHA:" "AES256-SHA:"
"DES-CBC3-SHA" "DES-CBC3-SHA",
source_address
)) ))
def _emulate_browser_chrome(session, platform): def _emulate_browser_chrome(session, platform, source_address):
if platform.startswith("Macintosh"): if platform.startswith("Macintosh"):
platform = platform.replace(".", "_") + "_2" platform = platform.replace(".", "_") + "_2"
@ -690,7 +719,8 @@ def _emulate_browser_chrome(session, platform):
"AES256-GCM-SHA384:" "AES256-GCM-SHA384:"
"AES128-SHA:" "AES128-SHA:"
"AES256-SHA:" "AES256-SHA:"
"DES-CBC3-SHA" "DES-CBC3-SHA",
source_address
)) ))

View File

@ -123,6 +123,11 @@ def build_parser():
dest="proxy", metavar="URL", action=ConfigAction, dest="proxy", metavar="URL", action=ConfigAction,
help="Use the specified proxy", help="Use the specified proxy",
) )
general.add_argument(
"--source-address",
dest="source-address", metavar="IP", action=ConfigAction,
help="Client-side IP address to bind to",
)
general.add_argument( general.add_argument(
"--clear-cache", "--clear-cache",
dest="clear_cache", metavar="MODULE", dest="clear_cache", metavar="MODULE",