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
|
from BaseHTTPServer import (
HTTPServer,
BaseHTTPRequestHandler,
)
import httplib
import os
from signal import SIGKILL
import simplejson
from StringIO import StringIO
from unittest import TestCase
from urlparse import urlparse
from urlparse import parse_qs
from testtools import ExpectedException
from grackle.client import (
GrackleClient,
)
class ForkedFake:
def __init__(self, port, messages=None):
self.pid = None
self.port = port
self.messages = messages
self.read_end, self.write_end = os.pipe()
def is_ready(self):
os.write(self.write_end, 'asdf')
def __enter__(self):
pid = os.fork()
if pid == 0:
self.start_server()
self.pid = pid
os.read(self.read_end, 1)
return
def start_server(self):
service = HTTPServer(('', self.port), FakeGrackleRequestHandler)
service.messages = self.messages
self.is_ready()
service.serve_forever()
def __exit__(self, exc_type, exc_val, traceback):
os.kill(self.pid, SIGKILL)
class FakeGrackleRequestHandler(BaseHTTPRequestHandler):
def do_POST(self):
message = self.rfile.read(int(self.headers['content-length']))
if message == 'This is a message':
self.send_response(httplib.CREATED)
self.end_headers()
self.wfile.close()
else:
self.send_error(httplib.BAD_REQUEST)
def do_GET(self):
scheme, netloc, path, params, query_string, fragments = (
urlparse(self.path))
archive = os.path.split(path)[1]
query = parse_qs(query_string)
parameters = simplejson.loads(query['parameters'][0])
self.send_response(httplib.OK)
self.end_headers()
messages = [m for m in self.server.messages[archive] if 'message_ids'
not in parameters or m['message_id'] in
parameters['message_ids']]
response = {
'messages': messages,
'next_memo': None,
'previous_memo': None
}
self.wfile.write(simplejson.dumps(response))
def fake_grackle_service(client, messages=None):
if messages is None:
messages = {}
return ForkedFake(client.port, messages)
class TestPutMessage(TestCase):
def test_put_message(self):
client = GrackleClient('localhost', 8436)
with fake_grackle_service(client):
client.put_message('arch1', 'asdf', StringIO('This is a message'))
with ExpectedException(Exception, 'wtf'):
client.put_message('arch1', 'asdf',
StringIO('This is not a message'))
class TestGetMessages(TestCase):
def test_get_messages(self):
client = GrackleClient('localhost', 8435)
with fake_grackle_service(client,
{'baz':
[{'message_id': 'foo'},
{'message_id': 'bar'}]}):
response = client.get_messages('baz')
self.assertEqual(['bar', 'foo'], sorted(m['message_id'] for m in
response['messages']))
self.assertIs(None, response['next_memo'])
self.assertIs(None, response['previous_memo'])
def test_get_messages_by_id(self):
client = GrackleClient('localhost', 8437)
with fake_grackle_service(client,
{'baz':
[{'message_id': 'foo'},
{'message_id': 'bar'}]}):
response = client.get_messages('baz', message_ids=['foo'])
message, = response['messages']
self.assertEqual('foo', message['message_id'])
|