-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathFuncs.py
144 lines (118 loc) · 4.09 KB
/
Funcs.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
"""WebUtils.Funcs
This module provides some basic functions that are useful
in HTML and web development.
You can safely import * from WebUtils.Funcs if you like.
"""
# Aliases for URL encoding and decoding functions:
from urllib.parse import quote_plus as urlEncode, unquote_plus as urlDecode
__all__ = [
'htmlEncode', 'htmlEncodeStr', 'htmlDecode', 'urlEncode', 'urlDecode',
'htmlForDict', 'requestURI', 'normURL']
htmlForNone = '-' # used by htmlEncode
htmlCodes = (
('&', '&'),
('<', '<'),
('>', '>'),
('"', '"'),
)
htmlCodesReversed = tuple(reversed(htmlCodes))
def htmlEncode(what, codes=htmlCodes):
"""Return the HTML encoded version of the given object.
The optional 'codes' parameter allows passing custom translations.
"""
if what is None:
return htmlForNone
if hasattr(what, 'html'):
# allow objects to specify their own translation to html
# via a method, property or attribute
html = what.html
if callable(html):
html = html()
return html
what = str(what)
return htmlEncodeStr(what, codes)
def htmlEncodeStr(s, codes=htmlCodes):
"""Return the HTML encoded version of the given string.
This is useful to display a plain ASCII text string on a web page.
The optional 'codes' parameter allows passing custom translations.
"""
for c, e in codes:
s = s.replace(c, e)
return s
def htmlDecode(s, codes=htmlCodesReversed):
"""Return the ASCII decoded version of the given HTML string.
This does NOT remove normal HTML tags like <p>.
It is the inverse of htmlEncode().
The optional 'codes' parameter allows passing custom translations.
"""
for c, e in codes:
s = s.replace(e, c)
return s
def htmlForDict(d, addSpace=None, filterValueCallBack=None,
maxValueLength=None, topHeading=None, isEncoded=None):
"""Return HTML string with a table where each row is a key/value pair."""
if not d:
return ''
html = ['<table class="NiceTable">\n']
append = html.append
if topHeading:
html.append('<tr class="TopHeading">')
if isinstance(topHeading, tuple):
append('<th>{}</th><th>{}</th>'.format(*topHeading))
else:
append(f'<th colspan="2">{topHeading}</th>')
append('</tr>\n')
for key in sorted(d):
value = d[key]
if addSpace and key in addSpace:
target = addSpace[key]
value = (target + ' ').join(value.split(target))
if filterValueCallBack:
value = filterValueCallBack(value, key, d)
if maxValueLength and not isEncoded:
value = str(value)
if len(value) > maxValueLength:
value = value[:maxValueLength - 3] + '...'
key = htmlEncode(key)
if not isEncoded:
value = htmlEncode(value)
append(f'<tr><th>{key}</th><td>{value}</td></tr>\n')
append('</table>')
return ''.join(html)
def requestURI(env):
"""Return the request URI for a given CGI-style dictionary.
Uses REQUEST_URI if available, otherwise constructs and returns it
from SCRIPT_URL, SCRIPT_NAME, PATH_INFO and QUERY_STRING.
"""
uri = env.get('REQUEST_URI')
if uri is None:
uri = env.get('SCRIPT_URL')
if uri is None:
uri = env.get('SCRIPT_NAME', '') + env.get('PATH_INFO', '')
query = env.get('QUERY_STRING', '')
if query != '':
uri += '?' + query
return uri
def normURL(path):
"""Normalizes a URL path, like os.path.normpath.
Acts on a URL independent of operating system environment.
"""
if not path:
return
startSlash = path.startswith('/')
endSlash = path.endswith('/')
comps = path.split('/')
newComps = []
for comp in comps:
if comp in ('', '.'):
continue
if comp != '..':
newComps.append(comp)
elif newComps:
newComps.pop()
path = '/'.join(newComps)
if path and endSlash:
path += '/'
if startSlash:
path = '/' + path
return path