Limnoria/src/utils/web.py

240 lines
7.6 KiB
Python
Raw Normal View History

2005-01-19 14:14:38 +01:00
###
2005-01-19 14:33:05 +01:00
# Copyright (c) 2002-2005, Jeremiah Fincher
# Copyright (c) 2009, James McCoy
2005-01-19 14:14:38 +01:00
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright notice,
# this list of conditions, and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions, and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# * Neither the name of the author of this software nor the name of
# contributors to this software may be used to endorse or promote products
# derived from this software without specific prior written consent.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
###
import re
import sys
import base64
2005-01-19 14:14:38 +01:00
import socket
import urllib
import urllib2
import httplib
import urlparse
import htmlentitydefs
from HTMLParser import HTMLParser
2005-01-19 14:14:38 +01:00
sockerrors = (socket.error,)
try:
sockerrors += (socket.sslerror,)
except AttributeError:
pass
from str import normalizeWhitespace
2005-01-19 14:14:38 +01:00
Request = urllib2.Request
urlquote = urllib.quote
urlunquote = urllib.unquote
def urlencode(*args, **kwargs):
return urllib.urlencode(*args, **kwargs).encode()
2005-01-19 14:14:38 +01:00
class Error(Exception):
2005-01-19 14:14:38 +01:00
pass
_octet = r'(?:2(?:[0-4]\d|5[0-5])|1\d\d|\d{1,2})'
_ipAddr = r'%s(?:\.%s){3}' % (_octet, _octet)
# Base domain regex off RFC 1034 and 1738
_label = r'[0-9a-z][-0-9a-z]*[0-9a-z]?'
2011-01-02 13:22:54 +01:00
_domain = r'%s(?:\.%s)*\.[0-9a-z][-0-9a-z]+' % (_label, _label)
_urlRe = r'(\w+://(?:\S+@)?(?:%s|%s)(?::\d+)?(?:/[^\])>\s]*)?)' % (_domain, _ipAddr)
urlRe = re.compile(_urlRe, re.I)
_httpUrlRe = r'(https?://(?:\S+@)?(?:%s|%s)(?::\d+)?(?:/[^\])>\s]*)?)' % (_domain,
_ipAddr)
httpUrlRe = re.compile(_httpUrlRe, re.I)
2005-01-19 14:14:38 +01:00
REFUSED = 'Connection refused.'
TIMED_OUT = 'Connection timed out.'
UNKNOWN_HOST = 'Unknown host.'
RESET_BY_PEER = 'Connection reset by peer.'
FORBIDDEN = 'Client forbidden from accessing URL.'
def strError(e):
try:
n = e.args[0]
except Exception:
return str(e)
if n == 111:
return REFUSED
elif n in (110, 10060):
return TIMED_OUT
elif n == 104:
return RESET_BY_PEER
elif n in (8, 7, 3, 2, -2, -3):
2005-01-19 14:14:38 +01:00
return UNKNOWN_HOST
elif n == 403:
return FORBIDDEN
else:
return str(e)
defaultHeaders = {
'User-agent': 'Mozilla/5.0 (compatible; utils.web python module)'
2005-01-19 14:14:38 +01:00
}
# Other modules should feel free to replace this with an appropriate
# application-specific function. Feel free to use a callable here.
proxy = None
def getUrlFd(url, headers=None, data=None):
"""getUrlFd(url, headers=None, data=None)
Opens the given url and returns a file object. Headers and data are
a dict and string, respectively, as per urllib2.Request's arguments."""
2005-01-19 14:14:38 +01:00
if headers is None:
headers = defaultHeaders
2005-01-19 14:14:38 +01:00
try:
if not isinstance(url, urllib2.Request):
if '#' in url:
url = url[:url.index('#')]
if '@' in url:
scheme, url = url.split('://', 2)
auth, url = url.split('@')
url = scheme + '://' + url
request = urllib2.Request(url, headers=headers, data=data)
2012-10-23 18:15:13 +02:00
if 'auth' in locals():
if sys.version_info[0] >= 3 and isinstance(auth, str):
auth = auth.encode()
auth = base64.b64encode(auth)
if sys.version_info[0] >= 3:
auth = auth.decode()
2012-10-23 18:15:13 +02:00
request.add_header('Authorization',
'Basic ' + auth)
2005-01-19 14:14:38 +01:00
else:
request = url
request.add_data(data)
httpProxy = force(proxy)
2005-01-19 14:14:38 +01:00
if httpProxy:
request.set_proxy(httpProxy, 'http')
fd = urllib2.urlopen(request)
return fd
except socket.timeout, e:
raise Error, TIMED_OUT
except sockerrors, e:
raise Error, strError(e)
2005-01-19 14:14:38 +01:00
except httplib.InvalidURL, e:
raise Error, 'Invalid URL: %s' % e
2005-01-19 14:14:38 +01:00
except urllib2.HTTPError, e:
raise Error, strError(e)
2005-01-19 14:14:38 +01:00
except urllib2.URLError, e:
raise Error, strError(e.reason)
2005-01-19 14:14:38 +01:00
# Raised when urllib doesn't recognize the url type
except ValueError, e:
raise Error, strError(e)
2005-01-19 14:14:38 +01:00
def getUrl(url, size=None, headers=None, data=None):
"""getUrl(url, size=None, headers=None, data=None)
Gets a page. Returns a string that is the page gotten. Size is an integer
number of bytes to read from the URL. Headers and data are dicts as per
urllib2.Request's arguments."""
fd = getUrlFd(url, headers=headers, data=data)
2005-01-19 14:14:38 +01:00
try:
if size is None:
text = fd.read()
else:
text = fd.read(size)
except socket.timeout, e:
raise Error, TIMED_OUT
2005-01-19 14:14:38 +01:00
fd.close()
return text
def getDomain(url):
return urlparse.urlparse(url)[1]
_charset_re = ('<meta[^a-z<>]+charset='
"""(?P<charset>("[^"]+"|'[^']+'))""")
def getEncoding(s):
try:
match = re.search(_charset_re, s, re.MULTILINE)
if match:
return match.group('charset')[1:-1]
except:
match = re.search(_charset_re.encode(), s, re.MULTILINE)
if match:
return match.group('charset').decode()[1:-1]
try:
import charade.universaldetector
u = charade.universaldetector.UniversalDetector()
u.feed(s)
u.close()
return u.result['encoding']
except:
return None
class HtmlToText(HTMLParser, object):
"""Taken from some eff-bot code on c.l.p."""
entitydefs = htmlentitydefs.entitydefs.copy()
entitydefs['nbsp'] = ' '
def __init__(self, tagReplace=' '):
self.data = []
self.tagReplace = tagReplace
super(HtmlToText, self).__init__()
def handle_starttag(self, tag, attr):
self.data.append(self.tagReplace)
def handle_endtag(self, tag):
self.data.append(self.tagReplace)
def handle_data(self, data):
self.data.append(data)
def handle_entityref(self, data):
self.data.append(unichr(htmlentitydefs.name2codepoint[data]))
def getText(self):
text = ''.join(self.data).strip()
return normalizeWhitespace(text)
2013-07-09 14:02:25 +02:00
def htmlToText(s, tagReplace=' '):
"""Turns HTML into text. tagReplace is a string to replace HTML tags with.
"""
encoding = getEncoding(s)
2013-07-09 14:02:25 +02:00
if encoding:
s = s.decode(encoding)
else:
try:
if sys.version_info[0] < 3 or isinstance(s, bytes):
s = s.decode('utf8')
except:
pass
x = HtmlToText(tagReplace)
x.feed(s)
return x.getText()
def mungeEmail(s):
s = s.replace('@', ' AT ')
s = s.replace('.', ' DOT ')
return s
2013-07-09 14:02:25 +02:00
# vim:set shiftwidth=4 softtabstop=4 expandtab textwidth=79:
2005-01-19 14:14:38 +01:00