urllibcompat.py
156 lines
| 3.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / urllibcompat.py
Augie Fackler
|
r34466 | # urllibcompat.py - adapters to ease using urllib2 on Py2 and urllib on Py3 | ||
# | ||||
# Copyright 2017 Google, Inc. | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Matt Harbison
|
r52755 | from __future__ import annotations | ||
Gregory Szorc
|
r49751 | import http.server | ||
import urllib.error | ||||
import urllib.parse | ||||
import urllib.request | ||||
import urllib.response | ||||
Augie Fackler
|
r34466 | from . import pycompat | ||
Augie Fackler
|
r34468 | _sysstr = pycompat.sysstr | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class _pycompatstub: | ||
Augie Fackler
|
r34468 | def __init__(self): | ||
self._aliases = {} | ||||
def _registeraliases(self, origin, items): | ||||
"""Add items that will be populated at the first access""" | ||||
items = map(_sysstr, items) | ||||
self._aliases.update( | ||||
Augie Fackler
|
r43906 | (item.replace('_', '').lower(), (origin, item)) for item in items | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r34468 | |||
def _registeralias(self, origin, attr, name): | ||||
"""Alias ``origin``.``attr`` as ``name``""" | ||||
self._aliases[_sysstr(name)] = (origin, _sysstr(attr)) | ||||
def __getattr__(self, name): | ||||
try: | ||||
origin, item = self._aliases[name] | ||||
except KeyError: | ||||
raise AttributeError(name) | ||||
self.__dict__[name] = obj = getattr(origin, item) | ||||
return obj | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r34468 | httpserver = _pycompatstub() | ||
urlreq = _pycompatstub() | ||||
urlerr = _pycompatstub() | ||||
Gregory Szorc
|
r49751 | urlreq._registeraliases( | ||
urllib.parse, | ||||
( | ||||
b"splitattr", | ||||
b"splitpasswd", | ||||
b"splitport", | ||||
b"splituser", | ||||
b"urlparse", | ||||
b"urlunparse", | ||||
), | ||||
) | ||||
urlreq._registeralias(urllib.parse, b"parse_qs", b"parseqs") | ||||
urlreq._registeralias(urllib.parse, b"parse_qsl", b"parseqsl") | ||||
urlreq._registeralias(urllib.parse, b"unquote_to_bytes", b"unquote") | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49751 | urlreq._registeraliases( | ||
urllib.request, | ||||
( | ||||
b"AbstractHTTPHandler", | ||||
b"BaseHandler", | ||||
b"build_opener", | ||||
b"FileHandler", | ||||
b"FTPHandler", | ||||
b"ftpwrapper", | ||||
Joerg Sonnenberger
|
r52722 | b"HTTPCookieProcessor", | ||
Gregory Szorc
|
r49751 | b"HTTPHandler", | ||
b"HTTPSHandler", | ||||
b"install_opener", | ||||
b"pathname2url", | ||||
b"HTTPBasicAuthHandler", | ||||
b"HTTPDigestAuthHandler", | ||||
b"HTTPPasswordMgrWithDefaultRealm", | ||||
b"ProxyHandler", | ||||
b"Request", | ||||
b"url2pathname", | ||||
b"urlopen", | ||||
), | ||||
) | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49751 | urlreq._registeraliases( | ||
urllib.response, | ||||
( | ||||
b"addclosehook", | ||||
b"addinfourl", | ||||
), | ||||
) | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49751 | urlerr._registeraliases( | ||
urllib.error, | ||||
( | ||||
b"HTTPError", | ||||
b"URLError", | ||||
), | ||||
) | ||||
httpserver._registeraliases( | ||||
http.server, | ||||
( | ||||
b"HTTPServer", | ||||
b"BaseHTTPRequestHandler", | ||||
b"SimpleHTTPRequestHandler", | ||||
b"CGIHTTPRequestHandler", | ||||
), | ||||
) | ||||
Matt Harbison
|
r47543 | |||
Raphaël Gomès
|
r52595 | |||
Gregory Szorc
|
r49751 | # urllib.parse.quote() accepts both str and bytes, decodes bytes | ||
# (if necessary), and returns str. This is wonky. We provide a custom | ||||
# implementation that only accepts bytes and emits bytes. | ||||
def quote(s, safe='/'): | ||||
# bytestr has an __iter__ that emits characters. quote_from_bytes() | ||||
# does an iteration and expects ints. We coerce to bytes to appease it. | ||||
if isinstance(s, pycompat.bytestr): | ||||
s = bytes(s) | ||||
s = urllib.parse.quote_from_bytes(s, safe=safe) | ||||
return s.encode('ascii', 'strict') | ||||
# urllib.parse.urlencode() returns str. We use this function to make | ||||
# sure we return bytes. | ||||
def urlencode(query, doseq=False): | ||||
s = urllib.parse.urlencode(query, doseq=doseq) | ||||
return s.encode('ascii') | ||||
Augie Fackler
|
r34466 | |||
Gregory Szorc
|
r49751 | urlreq.quote = quote | ||
urlreq.urlencode = urlencode | ||||
Augie Fackler
|
r34466 | |||
Gregory Szorc
|
r49751 | def getfullurl(req): | ||
return req.full_url | ||||
def gethost(req): | ||||
return req.host | ||||
Augie Fackler
|
r34466 | |||
Gregory Szorc
|
r49751 | |||
def getselector(req): | ||||
return req.selector | ||||
Augie Fackler
|
r34466 | |||
Gregory Szorc
|
r49751 | def getdata(req): | ||
return req.data | ||||
Augie Fackler
|
r34466 | |||
Gregory Szorc
|
r49751 | |||
def hasdata(req): | ||||
return req.data is not None | ||||