Tornado Unit Testing - Tornado应用的单元测试

之前在测试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')


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


posted @ 2010-01-01 10:48  CoderZh  阅读(5216)  评论(2编辑  收藏  举报