之前在测试Django应用时,使用了非常方便的django.test.TestCase。在测试Tornado时,我也包装了一个TestCase类,提供和Django一样便捷的测试方法。最终,测试案例的代码将会是这样:
from
testclient
import
TestCase
class QueryTest(TestCase):
def setUp(self):
pass
def test_query(self):
file = open( ' uploadfile.dat ' , ' rb ' )
response = self.client.post( ' /query ' , { ' a ' : ' 1 ' , 'b': '2', 'upload ' : file })
self.failUnlessEqual(response.status_code, 200 )
self.failUnlessEqual(response.content, ' ok ' )
class QueryTest(TestCase):
def setUp(self):
pass
def test_query(self):
file = open( ' uploadfile.dat ' , ' rb ' )
response = self.client.post( ' /query ' , { ' a ' : ' 1 ' , 'b': '2', 'upload ' : file })
self.failUnlessEqual(response.status_code, 200 )
self.failUnlessEqual(response.content, ' ok ' )
testclient.py的代码如下:
代码
#
!/usr/bin/env python
# coding:utf-8
#
# Copyright 2009 CoderZh.com.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
__author__ = ' CoderZh '
import tornado.ioloop
import unittest
import mimetypes
import tornado.httpclient
import tornado.ioloop
TEST_PORT = 8989
def encode_multipart_formdata(fields, files):
"""
fields is a sequence of (name, value) elements for regular form fields.
files is a sequence of (name, filename, value) elements for data to be uploaded as files
Return (content_type, body) ready for httplib.HTTP instance
"""
BOUNDARY = ' ----------ThIs_Is_tHe_bouNdaRY_$ '
CRLF = ' \r\n '
L = []
for (key, value) in fields:
L.append( ' -- ' + BOUNDARY)
L.append( ' Content-Disposition: form-data; name="%s" ' % key)
L.append( '' )
L.append(value)
for (key, filename, value) in files:
L.append( ' -- ' + BOUNDARY)
L.append( ' Content-Disposition: form-data; name="%s"; filename="%s" ' % (key, filename))
L.append( ' Content-Type: %s ' % get_content_type(filename))
L.append( '' )
L.append(value)
L.append( ' -- ' + BOUNDARY + ' -- ' )
L.append( '' )
body = CRLF.join(L)
content_type = ' multipart/form-data; boundary=%s ' % BOUNDARY
return content_type, body
def get_content_type(filename):
return mimetypes.guess_type(filename)[0] or ' application/octet-stream '
class Response:
def __init__ (self, status_code, content):
self.status_code = status_code
self.content = content
class Client:
def handle_request(self, response):
self.response = response
tornado.ioloop.IOLoop.instance().stop()
def post(self, url, data = {}):
url = ' http://127.0.0.1:%s%s ' % (TEST_PORT, url)
fields = []
files = []
for key, value in data.items():
if isinstance(value, file):
files.append([key, value.name, value.read()])
else :
fields.append([key, value])
content_type, body = encode_multipart_formdata(fields, files)
headers = { ' Content-Type ' : content_type}
request = tornado.httpclient.HTTPRequest(url = url,
method = ' POST ' ,
headers = headers,
body = body)
client = tornado.httpclient.AsyncHTTPClient()
client.fetch(request , self.handle_request)
tornado.ioloop.IOLoop.instance().start()
return Response(self.response.code, self.response.body)
class TestCase(unittest.TestCase):
def _pre_setup(self):
pass
def _post_teardown(self):
pass
def __call__ (self, result = None):
"""
Wrapper around default __call__ method to perform My test
set up. This means that user-defined Test Cases aren't required to
include a call to super().setUp().
"""
self.client = Client()
try :
self._pre_setup()
except (KeyboardInterrupt, SystemExit):
raise
except Exception:
import sys
result.addError(self, sys.exc_info())
return
super(TestCase, self). __call__ (result)
try :
self._post_teardown()
except (KeyboardInterrupt, SystemExit):
raise
except Exception:
import sys
result.addError(self, sys.exc_info())
return
# coding:utf-8
#
# Copyright 2009 CoderZh.com.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
__author__ = ' CoderZh '
import tornado.ioloop
import unittest
import mimetypes
import tornado.httpclient
import tornado.ioloop
TEST_PORT = 8989
def encode_multipart_formdata(fields, files):
"""
fields is a sequence of (name, value) elements for regular form fields.
files is a sequence of (name, filename, value) elements for data to be uploaded as files
Return (content_type, body) ready for httplib.HTTP instance
"""
BOUNDARY = ' ----------ThIs_Is_tHe_bouNdaRY_$ '
CRLF = ' \r\n '
L = []
for (key, value) in fields:
L.append( ' -- ' + BOUNDARY)
L.append( ' Content-Disposition: form-data; name="%s" ' % key)
L.append( '' )
L.append(value)
for (key, filename, value) in files:
L.append( ' -- ' + BOUNDARY)
L.append( ' Content-Disposition: form-data; name="%s"; filename="%s" ' % (key, filename))
L.append( ' Content-Type: %s ' % get_content_type(filename))
L.append( '' )
L.append(value)
L.append( ' -- ' + BOUNDARY + ' -- ' )
L.append( '' )
body = CRLF.join(L)
content_type = ' multipart/form-data; boundary=%s ' % BOUNDARY
return content_type, body
def get_content_type(filename):
return mimetypes.guess_type(filename)[0] or ' application/octet-stream '
class Response:
def __init__ (self, status_code, content):
self.status_code = status_code
self.content = content
class Client:
def handle_request(self, response):
self.response = response
tornado.ioloop.IOLoop.instance().stop()
def post(self, url, data = {}):
url = ' http://127.0.0.1:%s%s ' % (TEST_PORT, url)
fields = []
files = []
for key, value in data.items():
if isinstance(value, file):
files.append([key, value.name, value.read()])
else :
fields.append([key, value])
content_type, body = encode_multipart_formdata(fields, files)
headers = { ' Content-Type ' : content_type}
request = tornado.httpclient.HTTPRequest(url = url,
method = ' POST ' ,
headers = headers,
body = body)
client = tornado.httpclient.AsyncHTTPClient()
client.fetch(request , self.handle_request)
tornado.ioloop.IOLoop.instance().start()
return Response(self.response.code, self.response.body)
class TestCase(unittest.TestCase):
def _pre_setup(self):
pass
def _post_teardown(self):
pass
def __call__ (self, result = None):
"""
Wrapper around default __call__ method to perform My test
set up. This means that user-defined Test Cases aren't required to
include a call to super().setUp().
"""
self.client = Client()
try :
self._pre_setup()
except (KeyboardInterrupt, SystemExit):
raise
except Exception:
import sys
result.addError(self, sys.exc_info())
return
super(TestCase, self). __call__ (result)
try :
self._post_teardown()
except (KeyboardInterrupt, SystemExit):
raise
except Exception:
import sys
result.addError(self, sys.exc_info())
return