5 class headdict(object):
9 def __getitem__(self, key):
10 return self.dict[key.lower()][1]
12 def __setitem__(self, key, val):
13 self.dict[key.lower()] = [key, val]
15 def __contains__(self, key):
16 return key.lower() in self.dict
18 def __delitem__(self, key):
19 del self.dict[key.lower()]
22 return iter((list[0] for list in self.dict.values()))
24 def get(self, key, default=""):
25 if key.lower() in self.dict:
26 return self.dict[key.lower()][1]
29 def getlist(self, key):
30 return self.dict.setdefault(key.lower(), [key])[1:]
32 def add(self, key, val):
33 self.dict.setdefault(key.lower(), [key]).append(val)
36 return repr(self.dict)
47 str = str[:i] + str[i].upper() + str[i + 1:]
54 class shortinput(IOError, EOFError):
56 super().__init__("Unexpected EOF")
58 class limitreader(object):
59 def __init__(self, back, limit, short=False):
64 self.buf = bytearray()
69 def read(self, size=-1):
70 ra = self.limit - self.rb
73 while len(self.buf) < ra:
74 ret = self.bk.read(ra - len(self.buf))
83 ret = bytes(self.buf[:ra])
87 def readline(self, size=-1):
90 p = self.buf.find(b'\n', off)
92 ret = bytes(self.buf[:p + 1])
93 self.buf[:p + 1] = b""
96 if size >= 0 and len(self.buf) >= size:
97 ret = bytes(self.buf[:size])
100 if self.rb == self.limit:
101 ret = bytes(self.buf)
104 ra = self.limit - self.rb
108 ret = self.bk.read(ra)
111 ret = bytes(self.buf)
118 def readlines(self, hint=None):
122 class lineiter(object):
128 raise StopIteration()
140 return self.bk.closed
142 class request(object):
144 return copyrequest(self)
148 new.uriname = self.uriname + self.pathinfo[:n]
149 new.pathinfo = self.pathinfo[n:]
152 class origrequest(request):
153 def __init__(self, env):
155 self.method = env["REQUEST_METHOD"].upper()
156 self.uriname = env["SCRIPT_NAME"]
157 self.filename = env.get("SCRIPT_FILENAME")
158 self.uri = env["REQUEST_URI"]
159 self.pathinfo = env["PATH_INFO"]
160 self.query = env["QUERY_STRING"]
161 self.remoteaddr = env["REMOTE_ADDR"]
162 self.serverport = env["SERVER_PORT"]
163 self.servername = env["SERVER_NAME"]
164 self.https = "HTTPS" in env
165 self.ihead = headdict()
166 if "CONTENT_TYPE" in env:
167 self.ihead["Content-Type"] = env["CONTENT_TYPE"]
168 if "CONTENT_LENGTH" in env:
169 clen = self.ihead["Content-Length"] = env["CONTENT_LENGTH"]
171 self.input = limitreader(env["wsgi.input"], int(clen))
174 self.input = io.BytesIO(b"")
176 # Assume input is chunked and read until ordinary EOF.
177 self.input = env["wsgi.input"]
180 self.ohead = headdict()
181 for k, v in env.items():
183 self.ihead.add(fixcase(k[5:].replace("_", "-")), v)
185 self.statuscode = (200, "OK")
186 self.ohead["Content-Type"] = "text/html"
187 self.resources = set()
191 def status(self, code, msg):
192 self.statuscode = code, msg
196 return self.items[id]
197 self.items[id] = new = id(self)
198 if hasattr(new, "__enter__") and hasattr(new, "__exit__"):
202 def withres(self, res):
203 if res not in self.resources:
207 self.resources.add(res)
208 self.clean.add(res.__exit__)
212 res.__exit__(None, None, None)
213 self.resources.discard(res)
222 clean1(list(self.clean))
224 def oncommit(self, fn):
225 if fn not in self.commitfuns:
226 self.commitfuns.append(fn)
228 def commit(self, startreq):
229 for fun in reversed(self.commitfuns):
232 for nm in self.ohead:
233 for val in self.ohead.getlist(nm):
234 hdrs.append((nm, val))
235 startreq("%s %s" % self.statuscode, hdrs)
240 class copyrequest(request):
241 def __init__(self, p):
243 self.top = p.topreq()
245 self.method = p.method
246 self.uriname = p.uriname
247 self.filename = p.filename
249 self.pathinfo = p.pathinfo
251 self.remoteaddr = p.remoteaddr
252 self.serverport = p.serverport
258 def status(self, code, msg):
259 return self.parent.status(code, msg)
262 return self.top.item(id)
264 def withres(self, res):
265 return self.top.withres(res)
267 def oncommit(self, fn):
268 return self.top.oncommit(fn)
271 return self.parent.topreq()