2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
import abc
|
|
|
|
|
|
|
|
import contrib.pyparsing as pp
|
|
|
|
|
|
|
|
import netlib.websockets
|
2015-05-02 10:32:57 +00:00
|
|
|
from netlib import http_status, http_uastrings
|
2015-05-02 04:17:00 +00:00
|
|
|
from . import base, generators, exceptions
|
|
|
|
|
|
|
|
|
2015-05-02 09:27:11 +00:00
|
|
|
class WS(base.CaselessLiteral):
|
|
|
|
TOK = "ws"
|
|
|
|
|
|
|
|
|
|
|
|
class Raw(base.CaselessLiteral):
|
|
|
|
TOK = "r"
|
|
|
|
|
|
|
|
|
|
|
|
class Path(base.SimpleValue):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2015-05-02 09:42:09 +00:00
|
|
|
class Code(base.Integer):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class Reason(base.PreValue):
|
|
|
|
preamble = "m"
|
|
|
|
|
|
|
|
|
|
|
|
class Body(base.PreValue):
|
|
|
|
preamble = "b"
|
|
|
|
|
|
|
|
|
2015-05-02 09:27:11 +00:00
|
|
|
class Method(base.OptionsOrValue):
|
|
|
|
options = [
|
|
|
|
"get",
|
|
|
|
"head",
|
|
|
|
"post",
|
|
|
|
"put",
|
|
|
|
"delete",
|
|
|
|
"options",
|
|
|
|
"trace",
|
|
|
|
"connect",
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2015-05-02 10:32:57 +00:00
|
|
|
class _HeaderMixin(object):
|
|
|
|
def format_header(self, key, value):
|
|
|
|
return [key, ": ", value, "\r\n"]
|
|
|
|
|
|
|
|
def values(self, settings):
|
|
|
|
return self.format_header(
|
|
|
|
self.key.get_generator(settings),
|
|
|
|
self.value.get_generator(settings),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class Header(_HeaderMixin, base.KeyValue):
|
|
|
|
preamble = "h"
|
|
|
|
|
|
|
|
|
|
|
|
class ShortcutContentType(_HeaderMixin, base.PreValue):
|
|
|
|
preamble = "c"
|
|
|
|
key = base.ValueLiteral("Content-Type")
|
|
|
|
|
|
|
|
|
|
|
|
class ShortcutLocation(_HeaderMixin, base.PreValue):
|
|
|
|
preamble = "l"
|
|
|
|
key = base.ValueLiteral("Location")
|
|
|
|
|
|
|
|
|
|
|
|
class ShortcutUserAgent(_HeaderMixin, base.OptionsOrValue):
|
|
|
|
preamble = "u"
|
|
|
|
options = [i[1] for i in http_uastrings.UASTRINGS]
|
|
|
|
key = base.ValueLiteral("User-Agent")
|
|
|
|
|
|
|
|
def values(self, settings):
|
|
|
|
if self.option_used:
|
|
|
|
value = http_uastrings.get_by_shortcut(
|
|
|
|
self.value.val.lower()
|
|
|
|
)[2]
|
|
|
|
else:
|
|
|
|
value = self.value
|
|
|
|
return self.format_header(
|
|
|
|
self.key.get_generator(settings),
|
|
|
|
value
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2015-05-02 04:17:00 +00:00
|
|
|
def get_header(val, headers):
|
|
|
|
"""
|
|
|
|
Header keys may be Values, so we have to "generate" them as we try the
|
|
|
|
match.
|
|
|
|
"""
|
|
|
|
for h in headers:
|
|
|
|
k = h.key.get_generator({})
|
|
|
|
if len(k) == len(val) and k[:].lower() == val.lower():
|
|
|
|
return h
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
class _HTTPMessage(base._Message):
|
|
|
|
version = "HTTP/1.1"
|
2015-05-02 09:42:09 +00:00
|
|
|
|
2015-05-02 09:27:11 +00:00
|
|
|
@property
|
|
|
|
def raw(self):
|
|
|
|
return bool(self.tok(Raw))
|
|
|
|
|
2015-05-02 09:42:09 +00:00
|
|
|
@property
|
|
|
|
def body(self):
|
|
|
|
return self.tok(Body)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@abc.abstractmethod
|
|
|
|
def preamble(self, settings): # pragma: no cover
|
|
|
|
pass
|
|
|
|
|
2015-05-02 10:32:57 +00:00
|
|
|
@property
|
|
|
|
def headers(self):
|
|
|
|
return self.toks(_HeaderMixin)
|
|
|
|
|
2015-05-02 04:17:00 +00:00
|
|
|
def values(self, settings):
|
|
|
|
vals = self.preamble(settings)
|
|
|
|
vals.append("\r\n")
|
|
|
|
for h in self.headers:
|
|
|
|
vals.extend(h.values(settings))
|
|
|
|
vals.append("\r\n")
|
|
|
|
if self.body:
|
|
|
|
vals.append(self.body.value.get_generator(settings))
|
|
|
|
return vals
|
|
|
|
|
|
|
|
|
|
|
|
class Response(_HTTPMessage):
|
|
|
|
comps = (
|
2015-05-02 09:42:09 +00:00
|
|
|
Body,
|
2015-05-02 10:32:57 +00:00
|
|
|
Header,
|
2015-05-02 04:17:00 +00:00
|
|
|
base.PauseAt,
|
|
|
|
base.DisconnectAt,
|
|
|
|
base.InjectAt,
|
2015-05-02 10:32:57 +00:00
|
|
|
ShortcutContentType,
|
|
|
|
ShortcutLocation,
|
2015-05-02 09:27:11 +00:00
|
|
|
Raw,
|
2015-05-02 09:42:09 +00:00
|
|
|
Reason
|
2015-05-02 04:17:00 +00:00
|
|
|
)
|
|
|
|
logattrs = ["code", "reason", "version", "body"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def ws(self):
|
2015-05-02 09:27:11 +00:00
|
|
|
return self.tok(WS)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def code(self):
|
2015-05-02 09:42:09 +00:00
|
|
|
return self.tok(Code)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def reason(self):
|
2015-05-02 09:42:09 +00:00
|
|
|
return self.tok(Reason)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
def preamble(self, settings):
|
|
|
|
l = [self.version, " "]
|
|
|
|
l.extend(self.code.values(settings))
|
2015-05-02 09:42:09 +00:00
|
|
|
code = int(self.code.value)
|
2015-05-02 04:17:00 +00:00
|
|
|
l.append(" ")
|
|
|
|
if self.reason:
|
|
|
|
l.extend(self.reason.values(settings))
|
|
|
|
else:
|
|
|
|
l.append(
|
|
|
|
generators.LiteralGenerator(
|
|
|
|
http_status.RESPONSES.get(
|
|
|
|
code,
|
|
|
|
"Unknown code"
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return l
|
|
|
|
|
|
|
|
def resolve(self, settings, msg=None):
|
|
|
|
tokens = self.tokens[:]
|
|
|
|
if self.ws:
|
|
|
|
if not settings.websocket_key:
|
|
|
|
raise exceptions.RenderError(
|
|
|
|
"No websocket key - have we seen a client handshake?"
|
|
|
|
)
|
|
|
|
if not self.code:
|
|
|
|
tokens.insert(
|
|
|
|
1,
|
2015-05-02 09:42:09 +00:00
|
|
|
Code(101)
|
2015-05-02 04:17:00 +00:00
|
|
|
)
|
|
|
|
hdrs = netlib.websockets.server_handshake_headers(
|
|
|
|
settings.websocket_key
|
|
|
|
)
|
|
|
|
for i in hdrs.lst:
|
|
|
|
if not get_header(i[0], self.headers):
|
|
|
|
tokens.append(
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral(i[0]),
|
|
|
|
base.ValueLiteral(i[1]))
|
|
|
|
)
|
|
|
|
if not self.raw:
|
|
|
|
if not get_header("Content-Length", self.headers):
|
|
|
|
if not self.body:
|
|
|
|
length = 0
|
|
|
|
else:
|
|
|
|
length = len(self.body.value.get_generator(settings))
|
|
|
|
tokens.append(
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral("Content-Length"),
|
|
|
|
base.ValueLiteral(str(length)),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
intermediate = self.__class__(tokens)
|
|
|
|
return self.__class__(
|
|
|
|
[i.resolve(settings, intermediate) for i in tokens]
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def expr(klass):
|
|
|
|
parts = [i.expr() for i in klass.comps]
|
|
|
|
atom = pp.MatchFirst(parts)
|
|
|
|
resp = pp.And(
|
|
|
|
[
|
|
|
|
pp.MatchFirst(
|
|
|
|
[
|
2015-05-02 09:27:11 +00:00
|
|
|
WS.expr() + pp.Optional(
|
2015-05-02 09:42:09 +00:00
|
|
|
base.Sep + Code.expr()
|
2015-05-02 04:17:00 +00:00
|
|
|
),
|
2015-05-02 09:42:09 +00:00
|
|
|
Code.expr(),
|
2015-05-02 04:17:00 +00:00
|
|
|
]
|
|
|
|
),
|
|
|
|
pp.ZeroOrMore(base.Sep + atom)
|
|
|
|
]
|
|
|
|
)
|
|
|
|
resp = resp.setParseAction(klass)
|
|
|
|
return resp
|
|
|
|
|
|
|
|
def spec(self):
|
|
|
|
return ":".join([i.spec() for i in self.tokens])
|
|
|
|
|
|
|
|
|
|
|
|
class Request(_HTTPMessage):
|
|
|
|
comps = (
|
2015-05-02 09:42:09 +00:00
|
|
|
Body,
|
2015-05-02 10:32:57 +00:00
|
|
|
Header,
|
2015-05-02 04:17:00 +00:00
|
|
|
base.PauseAt,
|
|
|
|
base.DisconnectAt,
|
|
|
|
base.InjectAt,
|
2015-05-02 10:32:57 +00:00
|
|
|
ShortcutContentType,
|
|
|
|
ShortcutUserAgent,
|
2015-05-02 09:27:11 +00:00
|
|
|
Raw,
|
2015-05-02 04:17:00 +00:00
|
|
|
base.PathodSpec,
|
|
|
|
)
|
|
|
|
logattrs = ["method", "path", "body"]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def ws(self):
|
2015-05-02 09:27:11 +00:00
|
|
|
return self.tok(WS)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def method(self):
|
2015-05-02 09:27:11 +00:00
|
|
|
return self.tok(Method)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def path(self):
|
2015-05-02 09:27:11 +00:00
|
|
|
return self.tok(Path)
|
2015-05-02 04:17:00 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def pathodspec(self):
|
|
|
|
return self.tok(base.PathodSpec)
|
|
|
|
|
|
|
|
def preamble(self, settings):
|
|
|
|
v = self.method.values(settings)
|
|
|
|
v.append(" ")
|
|
|
|
v.extend(self.path.values(settings))
|
|
|
|
if self.pathodspec:
|
|
|
|
v.append(self.pathodspec.parsed.spec())
|
|
|
|
v.append(" ")
|
|
|
|
v.append(self.version)
|
|
|
|
return v
|
|
|
|
|
|
|
|
def resolve(self, settings, msg=None):
|
|
|
|
tokens = self.tokens[:]
|
|
|
|
if self.ws:
|
|
|
|
if not self.method:
|
|
|
|
tokens.insert(
|
|
|
|
1,
|
2015-05-02 09:27:11 +00:00
|
|
|
Method("get")
|
2015-05-02 04:17:00 +00:00
|
|
|
)
|
|
|
|
for i in netlib.websockets.client_handshake_headers().lst:
|
|
|
|
if not get_header(i[0], self.headers):
|
|
|
|
tokens.append(
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral(i[0]),
|
|
|
|
base.ValueLiteral(i[1])
|
|
|
|
)
|
|
|
|
)
|
|
|
|
if not self.raw:
|
|
|
|
if not get_header("Content-Length", self.headers):
|
|
|
|
if self.body:
|
|
|
|
length = len(self.body.value.get_generator(settings))
|
|
|
|
tokens.append(
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral("Content-Length"),
|
|
|
|
base.ValueLiteral(str(length)),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
if settings.request_host:
|
|
|
|
if not get_header("Host", self.headers):
|
|
|
|
tokens.append(
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral("Host"),
|
|
|
|
base.ValueLiteral(settings.request_host)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
intermediate = self.__class__(tokens)
|
|
|
|
return self.__class__(
|
|
|
|
[i.resolve(settings, intermediate) for i in tokens]
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def expr(klass):
|
|
|
|
parts = [i.expr() for i in klass.comps]
|
|
|
|
atom = pp.MatchFirst(parts)
|
|
|
|
resp = pp.And(
|
|
|
|
[
|
|
|
|
pp.MatchFirst(
|
|
|
|
[
|
2015-05-02 09:27:11 +00:00
|
|
|
WS.expr() + pp.Optional(
|
|
|
|
base.Sep + Method.expr()
|
2015-05-02 04:17:00 +00:00
|
|
|
),
|
2015-05-02 09:27:11 +00:00
|
|
|
Method.expr(),
|
2015-05-02 04:17:00 +00:00
|
|
|
]
|
|
|
|
),
|
|
|
|
base.Sep,
|
2015-05-02 09:27:11 +00:00
|
|
|
Path.expr(),
|
2015-05-02 04:17:00 +00:00
|
|
|
pp.ZeroOrMore(base.Sep + atom)
|
|
|
|
]
|
|
|
|
)
|
|
|
|
resp = resp.setParseAction(klass)
|
|
|
|
return resp
|
|
|
|
|
|
|
|
def spec(self):
|
|
|
|
return ":".join([i.spec() for i in self.tokens])
|
|
|
|
|
|
|
|
|
|
|
|
class PathodErrorResponse(Response):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
def make_error_response(reason, body=None):
|
|
|
|
tokens = [
|
2015-05-02 09:42:09 +00:00
|
|
|
Code("800"),
|
2015-05-02 10:32:57 +00:00
|
|
|
Header(
|
2015-05-02 04:17:00 +00:00
|
|
|
base.ValueLiteral("Content-Type"),
|
|
|
|
base.ValueLiteral("text/plain")
|
|
|
|
),
|
2015-05-02 09:42:09 +00:00
|
|
|
Reason(base.ValueLiteral(reason)),
|
|
|
|
Body(base.ValueLiteral("pathod error: " + (body or reason))),
|
2015-05-02 04:17:00 +00:00
|
|
|
]
|
|
|
|
return PathodErrorResponse(tokens)
|