aboutsummaryrefslogtreecommitdiffstats
path: root/libpathod
diff options
context:
space:
mode:
Diffstat (limited to 'libpathod')
-rw-r--r--libpathod/language/base.py92
-rw-r--r--libpathod/language/http.py73
2 files changed, 75 insertions, 90 deletions
diff --git a/libpathod/language/base.py b/libpathod/language/base.py
index 4e1900a7..4c337a9b 100644
--- a/libpathod/language/base.py
+++ b/libpathod/language/base.py
@@ -4,7 +4,6 @@ import os
import copy
import abc
import contrib.pyparsing as pp
-from netlib import http_uastrings
from .. import utils
from . import generators, exceptions
@@ -234,91 +233,29 @@ class _Component(_Token):
return "".join(i[:] for i in self.values(settings or {}))
-class _Header(_Component):
+class KeyValue(_Component):
+ """
+ A key/value pair.
+ klass.preamble: leader
+ """
def __init__(self, key, value):
self.key, self.value = key, value
- def values(self, settings):
- return [
- self.key.get_generator(settings),
- ": ",
- self.value.get_generator(settings),
- "\r\n",
- ]
-
-
-class Header(_Header):
@classmethod
def expr(klass):
- e = pp.Literal("h").suppress()
+ e = pp.Literal(klass.preamble).suppress()
e += Value
e += pp.Literal("=").suppress()
e += Value
return e.setParseAction(lambda x: klass(*x))
def spec(self):
- return "h%s=%s"%(self.key.spec(), self.value.spec())
+ return "%s%s=%s"%(self.preamble, self.key.spec(), self.value.spec())
def freeze(self, settings):
- return Header(self.key.freeze(settings), self.value.freeze(settings))
-
-
-class ShortcutContentType(_Header):
- def __init__(self, value):
- _Header.__init__(self, ValueLiteral("Content-Type"), value)
-
- @classmethod
- def expr(klass):
- e = pp.Literal("c").suppress()
- e = e + Value
- return e.setParseAction(lambda x: klass(*x))
-
- def spec(self):
- return "c%s"%(self.value.spec())
-
- def freeze(self, settings):
- return ShortcutContentType(self.value.freeze(settings))
-
-
-class ShortcutLocation(_Header):
- def __init__(self, value):
- _Header.__init__(self, ValueLiteral("Location"), value)
-
- @classmethod
- def expr(klass):
- e = pp.Literal("l").suppress()
- e = e + Value
- return e.setParseAction(lambda x: klass(*x))
-
- def spec(self):
- return "l%s"%(self.value.spec())
-
- def freeze(self, settings):
- return ShortcutLocation(self.value.freeze(settings))
-
-
-class ShortcutUserAgent(_Header):
- def __init__(self, value):
- self.specvalue = value
- if isinstance(value, basestring):
- value = ValueLiteral(http_uastrings.get_by_shortcut(value)[2])
- _Header.__init__(self, ValueLiteral("User-Agent"), value)
-
- @classmethod
- def expr(klass):
- e = pp.Literal("u").suppress()
- u = reduce(
- operator.or_,
- [pp.Literal(i[1]) for i in http_uastrings.UASTRINGS]
+ return self.__class__(
+ self.key.freeze(settings), self.value.freeze(settings)
)
- e += u | Value
- return e.setParseAction(lambda x: klass(*x))
-
- def spec(self):
- return "u%s"%self.specvalue
-
- def freeze(self, settings):
- return ShortcutUserAgent(self.value.freeze(settings))
class PathodSpec(_Token):
@@ -407,12 +344,15 @@ class OptionsOrValue(_Component):
"""
Can be any of a specified set of options, or a value specifier.
"""
+ preamble = ""
def __init__(self, value):
# If it's a string, we were passed one of the options, so we upper-case
# it to be canonical. The user can specify a different case by using a
# string value literal.
+ self.option_used = False
if isinstance(value, basestring):
value = ValueLiteral(value.upper())
+ self.option_used = True
self.value = value
@classmethod
@@ -421,6 +361,8 @@ class OptionsOrValue(_Component):
m = pp.MatchFirst(parts)
spec = m | Value.copy()
spec = spec.setParseAction(lambda x: klass(*x))
+ if klass.preamble:
+ spec = pp.Literal(klass.preamble).suppress() + spec
return spec
def values(self, settings):
@@ -432,7 +374,7 @@ class OptionsOrValue(_Component):
s = self.value.spec()
if s[1:-1].lower() in self.options:
s = s[1:-1].lower()
- return "%s"%s
+ return "%s%s"%(self.preamble, s)
def freeze(self, settings):
return self.__class__(self.value.freeze(settings))
@@ -617,10 +559,6 @@ class _Message(object):
def actions(self):
return self.toks(_Action)
- @property
- def headers(self):
- return self.toks(_Header)
-
def length(self, settings):
"""
Calculate the length of the base message without any applied
diff --git a/libpathod/language/http.py b/libpathod/language/http.py
index 30a5fd9f..a759aeb1 100644
--- a/libpathod/language/http.py
+++ b/libpathod/language/http.py
@@ -4,7 +4,7 @@ import abc
import contrib.pyparsing as pp
import netlib.websockets
-from netlib import http_status
+from netlib import http_status, http_uastrings
from . import base, generators, exceptions
@@ -45,6 +45,49 @@ class Method(base.OptionsOrValue):
]
+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
+ )
+
+
def get_header(val, headers):
"""
Header keys may be Values, so we have to "generate" them as we try the
@@ -72,6 +115,10 @@ class _HTTPMessage(base._Message):
def preamble(self, settings): # pragma: no cover
pass
+ @property
+ def headers(self):
+ return self.toks(_HeaderMixin)
+
def values(self, settings):
vals = self.preamble(settings)
vals.append("\r\n")
@@ -86,12 +133,12 @@ class _HTTPMessage(base._Message):
class Response(_HTTPMessage):
comps = (
Body,
- base.Header,
+ Header,
base.PauseAt,
base.DisconnectAt,
base.InjectAt,
- base.ShortcutContentType,
- base.ShortcutLocation,
+ ShortcutContentType,
+ ShortcutLocation,
Raw,
Reason
)
@@ -145,7 +192,7 @@ class Response(_HTTPMessage):
for i in hdrs.lst:
if not get_header(i[0], self.headers):
tokens.append(
- base.Header(
+ Header(
base.ValueLiteral(i[0]),
base.ValueLiteral(i[1]))
)
@@ -156,7 +203,7 @@ class Response(_HTTPMessage):
else:
length = len(self.body.value.get_generator(settings))
tokens.append(
- base.Header(
+ Header(
base.ValueLiteral("Content-Length"),
base.ValueLiteral(str(length)),
)
@@ -193,12 +240,12 @@ class Response(_HTTPMessage):
class Request(_HTTPMessage):
comps = (
Body,
- base.Header,
+ Header,
base.PauseAt,
base.DisconnectAt,
base.InjectAt,
- base.ShortcutContentType,
- base.ShortcutUserAgent,
+ ShortcutContentType,
+ ShortcutUserAgent,
Raw,
base.PathodSpec,
)
@@ -241,7 +288,7 @@ class Request(_HTTPMessage):
for i in netlib.websockets.client_handshake_headers().lst:
if not get_header(i[0], self.headers):
tokens.append(
- base.Header(
+ Header(
base.ValueLiteral(i[0]),
base.ValueLiteral(i[1])
)
@@ -251,7 +298,7 @@ class Request(_HTTPMessage):
if self.body:
length = len(self.body.value.get_generator(settings))
tokens.append(
- base.Header(
+ Header(
base.ValueLiteral("Content-Length"),
base.ValueLiteral(str(length)),
)
@@ -259,7 +306,7 @@ class Request(_HTTPMessage):
if settings.request_host:
if not get_header("Host", self.headers):
tokens.append(
- base.Header(
+ Header(
base.ValueLiteral("Host"),
base.ValueLiteral(settings.request_host)
)
@@ -302,7 +349,7 @@ class PathodErrorResponse(Response):
def make_error_response(reason, body=None):
tokens = [
Code("800"),
- base.Header(
+ Header(
base.ValueLiteral("Content-Type"),
base.ValueLiteral("text/plain")
),