/[thuban]/branches/WIP-pyshapelib-bramz/test/support.py
ViewVC logotype

Annotation of /branches/WIP-pyshapelib-bramz/test/support.py

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1605 - (hide annotations)
Tue Aug 19 11:00:40 2003 UTC (21 years, 6 months ago) by bh
Original Path: trunk/thuban/test/support.py
File MIME type: text/x-python
File size: 11129 byte(s)
Add very basic postgis database support and the corresponding test
cases. The test cases require a PostgreSQL + postgis installation
but no existing database. The database will be created
automatically by the test cases

* test/README: Add note about skipped tests and the requirements
of the postgis tests.

* Thuban/Model/postgisdb.py: New. Basic postgis database support.

* test/test_postgis_db.py: New. Test cases for the postgis
support.

* Thuban/Model/wellknowntext.py: New. Parser for well-known-text
format

* test/test_wellknowntext.py: New. Test cases for the
wellknowntext parser

* test/postgissupport.py: New. Support module for tests involving
a postgis database.

* test/support.py (execute_as_testsuite): Shut down the postmaster
if it's still running after the tests

* Thuban/Model/data.py (RAW_WKT): New constant for raw data in
well known text format

1 bh 596 # Copyright (c) 2002, 2003 by Intevation GmbH
2 bh 292 # Authors:
3     # Bernhard Herzog <[email protected]>
4     #
5     # This program is free software under the GPL (>=v2)
6     # Read the file COPYING coming with Thuban for details.
7    
8     """
9     Support classes and function for the test suite
10     """
11    
12     __version__ = "$Revision$"
13     # $Source$
14     # $Id$
15    
16     import os, sys
17     import unittest
18 bh 1601 import traceback
19 bh 292
20 bh 1605 import postgissupport
21    
22    
23 bh 292 def thuban_dir():
24     """Return the directory containing the Thuban package"""
25     thisdir = os.path.dirname(__file__)
26     return os.path.join(thisdir, os.pardir)
27    
28 bh 1245 def resource_dir():
29     return os.path.join(thuban_dir(), "Resources")
30 bh 292
31     def add_thuban_dir_to_path():
32     """Insert the Thuban directory at the beginning of the python path.
33    
34     If it's already part of the path, remove later occurrences.
35     """
36     dir = thuban_dir()
37     while 1:
38     try:
39     sys.path.remove(dir)
40     except ValueError:
41     break
42     sys.path.insert(0, dir)
43    
44    
45     _initthuban_done = 0
46     def initthuban():
47     """Initialize the interpreter for using Thuban modules
48     """
49     global _initthuban_done
50     if not _initthuban_done:
51     add_thuban_dir_to_path()
52     import thubaninit
53     _initthuban_done = 1
54    
55 bh 1555
56     #
57     # Special test runner and result that support skipping tests
58     #
59    
60     class SkipTest(Exception):
61     """Exception to raise in tests that are skipped for some reason
62    
63     For instance, since gdal support is optional, test cases that
64     require gdal raise this exception to indicate that they are skipped.
65     Skipped is different from failure or error in that it is expected
66     under certain circumstances.
67     """
68    
69     class ThubanTestResult(unittest._TextTestResult):
70    
71     def __init__(self, stream, descriptions, verbosity):
72     unittest._TextTestResult.__init__(self, stream, descriptions,
73     verbosity)
74     self.skipped_tests = {}
75    
76     def add_skipped_test(self, test, exc):
77     reason = str(exc)
78     self.skipped_tests.setdefault(reason, []).append(test)
79    
80     def count_skipped(self):
81     sum = 0
82     for tests in self.skipped_tests.values():
83     sum += len(tests)
84     return sum
85    
86     def addError(self, test, err):
87     """Extend inherited method to handle SkipTest exceptions specially
88     """
89     #print "addError", test, err
90     if isinstance(err[1], SkipTest):
91     self.add_skipped_test(test, err[1])
92     if self.showAll:
93     self.stream.writeln("skipped")
94     elif self.dots:
95     self.stream.write('S')
96     else:
97     unittest._TextTestResult.addError(self, test, err)
98    
99     def printErrors(self):
100     if self.skipped_tests:
101     if self.dots or self.showAll:
102     self.stream.writeln()
103     self.stream.writeln("Skipped tests:")
104     for reason, tests in self.skipped_tests.items():
105 bh 1601 self.stream.writeln(" %s:" % reason)
106 bh 1555 for test in tests:
107     self.stream.writeln(" " + test.id())
108     unittest._TextTestResult.printErrors(self)
109    
110    
111     class ThubanTestRunner(unittest.TextTestRunner):
112    
113     def _makeResult(self):
114     return ThubanTestResult(self.stream, self.descriptions, self.verbosity)
115    
116     def run(self, test):
117     result = unittest.TextTestRunner.run(self, test)
118     self.stream.writeln("skipped = %d" % result.count_skipped())
119     return result
120    
121    
122     class ThubanTestProgram(unittest.TestProgram):
123    
124     def runTests(self):
125     """Extend inherited method so that we use a ThubanTestRunner"""
126     print "ThubanTestProgram.runTests"
127     self.testRunner = ThubanTestRunner(verbosity = self.verbosity)
128     unittest.TestProgram.runTests(self)
129    
130    
131 bh 1601 def execute_as_testsuite(callable, *args, **kw):
132     """Call callable with args as if it were the entry point to the test suite
133 bh 292
134 bh 1601 Return watever callable returns.
135    
136     This is a helper function for run_tests and runtests.py. Call
137     callable in a try-finally block and run some cleanup and print some
138     additional information in the finally block.
139    
140     The additionaly information include:
141    
142     - A list of uncollected objects (after an explicit garbage
143     collector call)
144    
145     - any unsubscribed messages
146 bh 596 """
147     try:
148 bh 1601 return callable(*args, **kw)
149 bh 596 finally:
150     # This has to be in a finally clause because unittest.main()
151     # ends with a sys.exit to make sure that the process exits with
152     # an appropriate exit code
153 bh 1601
154 bh 1605 # Shutdown the postgis server if it's running
155     try:
156     postgissupport.shutdown_test_server()
157     except:
158     traceback.print_exc()
159    
160 bh 1601 # Print additional information
161 bh 1245 print_additional_summary()
162 bh 292
163 bh 1601 def run_tests():
164     """Frontend for unittest.main that prints some additional debug information
165    
166     After calling unittest.main, run the garbage collector and print
167     uncollected objects. Also print any un-unsubscribed messages.
168     """
169     execute_as_testsuite(ThubanTestProgram)
170    
171    
172 bh 1245 def print_additional_summary():
173     """Print some additional summary information after tests have been run"""
174     print_garbage_information()
175     import xmlsupport
176     xmlsupport.print_summary_message()
177    
178 bh 596 def print_garbage_information():
179     """Print information about things that haven't been cleaned up.
180    
181     Run the garbage collector and print uncollected objects. Also print
182     any un-unsubscribed messages.
183     """
184     import gc, Thuban.Lib.connector
185     gc.collect()
186     if gc.garbage:
187     print
188     print "There are %d uncollected objects:" % len(gc.garbage)
189     print gc.garbage
190     Thuban.Lib.connector._the_connector.print_connections()
191    
192 bh 1555 #
193 bh 596
194 bh 292 def create_temp_dir():
195     """Create a temporary directory and return its name.
196    
197     The temporary directory is always called temp and is created in the
198     directory where support module is located.
199    
200     If the temp directory already exists, just return the name.
201     """
202     name = os.path.abspath(os.path.join(os.path.dirname(__file__), "temp"))
203    
204     # if the directory already exists, we're done
205     if os.path.isdir(name):
206     return name
207    
208     # create the directory
209     os.mkdir(name)
210     return name
211    
212    
213     class FileTestMixin:
214    
215     """Mixin class for tests that use files in the temporary directory
216     """
217    
218     def temp_file_name(self, basename):
219     """Return the full name of the file named basename in the temp. dir"""
220     return os.path.join(create_temp_dir(), basename)
221    
222     def temp_dir(self):
223     """Return the name of the directory for the temporary files"""
224     return create_temp_dir()
225 bh 324
226    
227 bh 956
228     class FileLoadTestCase(unittest.TestCase, FileTestMixin):
229    
230     """Base class for test case that test loading files.
231    
232     This base class provides some common infrastructure for testing the
233     reading of files.
234    
235     Each test case should be its own class derived from this one. There
236     is one file associated with each class. The contents are defined by
237     the file_contents class attribute and its name by the filename
238     method.
239    
240     Derived classes usually only have to provide appropriate values for
241     the file_contents and file_extension class attributes.
242     """
243    
244     file_contents = None
245     file_extension = ""
246    
247     def filename(self):
248     """Return the name of the test file to use.
249    
250     The default implementation simply calls self.volatile_file_name
251     with a basename derived from the class name by stripping off a
252     leading 'test_' and appending self.file_extension.
253     """
254     name = self.__class__.__name__
255     if name.startswith("test_"):
256     name = name[5:]
257     return self.temp_file_name(name + self.file_extension)
258    
259     def setUp(self):
260     """Create the volatile file for the test.
261    
262     Write self.contents (which should be a string) to the file named
263     by self.filename().
264     """
265     filename = self.filename()
266     file = open(filename, "w")
267     file.write(self.file_contents)
268     file.close()
269    
270    
271 bh 1421 class FloatComparisonMixin:
272 bh 324
273     """
274     Mixin class for tests comparing floating point numbers.
275    
276     This class provides a few methods for testing floating point
277     operations.
278     """
279    
280     fp_epsilon = 1e-6
281 jonathan 1263 fp_inf = float('1e1000') # FIXME: hack for infinite
282 bh 1574
283 jonathan 1397 def assertFloatEqual(self, test, value, epsilon = None):
284 bh 324 """Assert equality of test and value with some tolerance.
285    
286     Assert that the absolute difference between test and value is
287     less than self.fp_epsilon.
288     """
289 jonathan 1397 if epsilon is None:
290     epsilon = self.fp_epsilon
291 jonathan 1263 if abs(test) == self.fp_inf:
292     self.assertEqual(test, value)
293     else:
294 jonathan 1397 self.assert_(epsilon > abs(test - value),
295     "abs(%g - %g) >= %g" % (test, value, epsilon))
296 bh 324
297     def assertFloatSeqEqual(self, test, value, epsilon = None):
298     """Assert equality of the sequences test and value with some tolerance.
299    
300     Assert that the absolute difference between each corresponding
301     value in test and value is less than the optional parameter
302     epsilon. If epsilon is not given use self.fp_epsilon.
303     """
304     for i in range(len(test)):
305 jonathan 1397 self.assertFloatEqual(test[i], value[i], epsilon)
306 bh 324
307 bh 1574 def assertPointListEquals(self, test, value):
308     """Assert equality of two lists of lists of tuples of float
309    
310     This assertion is usually used to compare the geometry of shapes
311     as returned by a Shape object's Points() method, hence the name.
312     """
313     for i in range(len(test)):
314     self.assertEquals(len(test[i]), len(value[i]))
315     for j in range(len(test[i])):
316     self.assertFloatSeqEqual(test[i][j], value[i][j])
317    
318    
319 bh 324 class SubscriberMixin:
320    
321     """Mixin class for tests for messages sent through the Connector
322    
323     The SubscriberMixin has some methods that can be used as subscribers
324     of events that when called append information about the message into
325     a list of messages received.
326    
327     A derived class should call the clear_messages() method in both its
328     setUp and tearDown methods to clear the list of messages received.
329     """
330    
331     def clear_messages(self):
332     """Clear the list of received messages.
333    
334     Call this at least in the tests setUp and tearDown methods. It's
335     important to do it in tearDown too because otherwise there may
336     be cyclic references.
337     """
338     self.received_messages = []
339    
340     def subscribe_no_params(self):
341     """Method for subscriptions without parameters.
342    
343     Add an empty tuple to the list of received messages.
344     """
345     self.received_messages.append(())
346    
347     def subscribe_with_params(self, *args):
348     """Method for subscriptions with parameters.
349    
350     Append the tuple will all arguments to this function (except for
351     the self argument) to the list of received messages.
352     """
353     self.received_messages.append(args)
354    
355     def check_messages(self, messages):
356     """Check whether the messages received match the list messages"""
357     self.assertEquals(messages, self.received_messages)
358 jonathan 904

Properties

Name Value
svn:eol-style native
svn:keywords Author Date Id Revision

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26