-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathabstract_test_type.py
132 lines (113 loc) · 4.66 KB
/
abstract_test_type.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
import abc
import copy
import requests
from colorama import Fore
from utils.output_helper import log
class AbstractTestType(metaclass=abc.ABCMeta):
'''
Interface between a test type (json, query, plaintext, etc) and
the rest of BW. A test type defines a number of keys it expects
to find in the benchmark_config.json, and this base class handles extracting
those keys and injecting them into the test. For example, if
benchmark_config.json contains a line `"spam" : "foobar"` and a subclasses X
passes an argument list of ['spam'], then after parsing there will
exist a member `X.spam = 'foobar'`.
'''
def __init__(self,
config,
name,
requires_db=False,
accept_header=None,
args=[]):
self.config = config
self.name = name
self.requires_db = requires_db
self.args = args
self.headers = ""
self.body = ""
if accept_header is None:
self.accept_header = self.accept('json')
else:
self.accept_header = accept_header
self.passed = None
self.failed = None
self.warned = None
@classmethod
def accept(self, content_type):
return {
'json':
'application/json,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7',
'html':
'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
'plaintext':
'text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7'
}[content_type]
def parse(self, test_keys):
'''
Takes the dict of key/value pairs describing a FrameworkTest
and collects all variables needed by this AbstractTestType
Raises AttributeError if required keys are missing
'''
if all(arg in test_keys for arg in self.args):
self.__dict__.update({arg: test_keys[arg] for arg in self.args})
return self
else: # This is quite common - most tests don't support all types
raise AttributeError(
"A %s requires the benchmark_config.json to contain %s" %
(self.name, self.args))
def request_headers_and_body(self, url):
'''
Downloads a URL and returns the HTTP response headers
and body content as a tuple
'''
log("Accessing URL {!s}: ".format(url), color=Fore.CYAN)
headers = {'Accept': self.accept_header}
r = requests.get(url, timeout=15, headers=headers)
self.headers = r.headers
self.body = r.content
return self.headers, self.body
def output_headers_and_body(self):
log(str(self.headers))
log(self.body)
def verify(self, base_url):
'''
Accesses URL used by this test type and checks the return
values for correctness. Most test types run multiple checks,
so this returns a list of results. Each result is a 3-tuple
of (String result, String reason, String urlTested).
- result : 'pass','warn','fail'
- reason : Short human-readable reason if result was
warn or fail. Please do not print the response as part of this,
other parts of BW will do that based upon the current logging
settings if this method indicates a failure happened
- urlTested: The exact URL that was queried
Subclasses should make a best-effort attempt to report as many
failures and warnings as they can to help users avoid needing
to run BW repeatedly while debugging
'''
# TODO make String result into an enum to enforce
raise NotImplementedError("Subclasses must provide verify")
def get_url(self):
'''
Returns the URL for this test, like '/json'
'''
# This is a method because each test type uses a different key
# for their URL so the base class can't know which arg is the URL
raise NotImplementedError("Subclasses must provide get_url")
def get_script_name(self):
'''
Returns the remote script name for running the benchmarking process.
'''
raise NotImplementedError("Subclasses must provide get_script_name")
def get_script_variables(self, name, url, port):
'''
Returns the remote script variables for running the benchmarking process.
'''
raise NotImplementedError(
"Subclasses must provide get_script_variables")
def copy(self):
'''
Returns a copy that can be safely modified.
Use before calling parse
'''
return copy.copy(self)