blob: b056039b8cfb6cdb02a2893c167a6633b3efddf5 [file] [log] [blame]
Brett Cannon66865d22008-03-13 20:27:00 +00001import os
2import unittest
Guido van Rossumb26a1b41998-05-20 17:05:52 +00003import random
Brett Cannon66865d22008-03-13 20:27:00 +00004from test import test_support
Victor Stinner6a102812010-04-27 23:55:59 +00005thread = test_support.import_module('thread')
Guido van Rossumcc544171994-04-14 20:28:41 +00006import time
Gregory P. Smith9e5d1322010-03-01 01:22:39 +00007import sys
Antoine Pitrou59c44f32009-10-30 17:07:08 +00008import weakref
Guido van Rossumcc544171994-04-14 20:28:41 +00009
Antoine Pitrouc98efe02009-11-06 22:34:35 +000010from test import lock_tests
Guido van Rossumcc544171994-04-14 20:28:41 +000011
Brett Cannon66865d22008-03-13 20:27:00 +000012NUMTASKS = 10
13NUMTRIPS = 3
Guido van Rossumd3b68421994-05-23 12:17:36 +000014
Guido van Rossumcc544171994-04-14 20:28:41 +000015
Jeffrey Yasskina1458532008-03-18 04:56:06 +000016_print_mutex = thread.allocate_lock()
17
Brett Cannon66865d22008-03-13 20:27:00 +000018def verbose_print(arg):
19 """Helper function for printing out debugging output."""
20 if test_support.verbose:
Jeffrey Yasskina1458532008-03-18 04:56:06 +000021 with _print_mutex:
22 print arg
Guido van Rossumcc544171994-04-14 20:28:41 +000023
Guido van Rossumcc544171994-04-14 20:28:41 +000024
Brett Cannon66865d22008-03-13 20:27:00 +000025class BasicThreadTest(unittest.TestCase):
Guido van Rossumd3b68421994-05-23 12:17:36 +000026
Brett Cannon66865d22008-03-13 20:27:00 +000027 def setUp(self):
28 self.done_mutex = thread.allocate_lock()
29 self.done_mutex.acquire()
30 self.running_mutex = thread.allocate_lock()
31 self.random_mutex = thread.allocate_lock()
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000032 self.created = 0
Brett Cannon66865d22008-03-13 20:27:00 +000033 self.running = 0
34 self.next_ident = 0
35
36
37class ThreadRunningTests(BasicThreadTest):
38
39 def newtask(self):
40 with self.running_mutex:
41 self.next_ident += 1
42 verbose_print("creating task %s" % self.next_ident)
43 thread.start_new_thread(self.task, (self.next_ident,))
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000044 self.created += 1
Brett Cannon66865d22008-03-13 20:27:00 +000045 self.running += 1
46
47 def task(self, ident):
48 with self.random_mutex:
Jeffrey Yasskina1458532008-03-18 04:56:06 +000049 delay = random.random() / 10000.0
50 verbose_print("task %s will run for %sus" % (ident, round(delay*1e6)))
Brett Cannon66865d22008-03-13 20:27:00 +000051 time.sleep(delay)
52 verbose_print("task %s done" % ident)
53 with self.running_mutex:
54 self.running -= 1
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000055 if self.created == NUMTASKS and self.running == 0:
Brett Cannon66865d22008-03-13 20:27:00 +000056 self.done_mutex.release()
57
58 def test_starting_threads(self):
59 # Basic test for thread creation.
60 for i in range(NUMTASKS):
61 self.newtask()
62 verbose_print("waiting for tasks to complete...")
63 self.done_mutex.acquire()
64 verbose_print("all tasks done")
65
66 def test_stack_size(self):
67 # Various stack size tests.
Florent Xicluna2e6d2622010-03-20 00:17:46 +000068 self.assertEqual(thread.stack_size(), 0, "initial stack size is not 0")
Brett Cannon66865d22008-03-13 20:27:00 +000069
70 thread.stack_size(0)
Florent Xicluna2e6d2622010-03-20 00:17:46 +000071 self.assertEqual(thread.stack_size(), 0, "stack_size not reset to default")
Brett Cannon66865d22008-03-13 20:27:00 +000072
Zachary Ware1f702212013-12-10 14:09:20 -060073 @unittest.skipIf(os.name not in ("nt", "os2", "posix"), 'test meant for nt, os2, and posix')
74 def test_nt_and_posix_stack_size(self):
Brett Cannon66865d22008-03-13 20:27:00 +000075 try:
76 thread.stack_size(4096)
77 except ValueError:
78 verbose_print("caught expected ValueError setting "
79 "stack_size(4096)")
80 except thread.error:
Zachary Ware1f702212013-12-10 14:09:20 -060081 self.skipTest("platform does not support changing thread stack "
82 "size")
Brett Cannon66865d22008-03-13 20:27:00 +000083
Zachary Ware1f702212013-12-10 14:09:20 -060084 fail_msg = "stack_size(%d) failed - should succeed"
85 for tss in (262144, 0x100000, 0):
86 thread.stack_size(tss)
87 self.assertEqual(thread.stack_size(), tss, fail_msg % tss)
88 verbose_print("successfully set stack_size(%d)" % tss)
Brett Cannon66865d22008-03-13 20:27:00 +000089
Zachary Ware1f702212013-12-10 14:09:20 -060090 for tss in (262144, 0x100000):
91 verbose_print("trying stack_size = (%d)" % tss)
92 self.next_ident = 0
93 self.created = 0
94 for i in range(NUMTASKS):
95 self.newtask()
Brett Cannon66865d22008-03-13 20:27:00 +000096
Zachary Ware1f702212013-12-10 14:09:20 -060097 verbose_print("waiting for all tasks to complete")
98 self.done_mutex.acquire()
99 verbose_print("all tasks done")
Brett Cannon66865d22008-03-13 20:27:00 +0000100
Zachary Ware1f702212013-12-10 14:09:20 -0600101 thread.stack_size(0)
Brett Cannon66865d22008-03-13 20:27:00 +0000102
Antoine Pitrou59c44f32009-10-30 17:07:08 +0000103 def test__count(self):
104 # Test the _count() function.
105 orig = thread._count()
106 mut = thread.allocate_lock()
107 mut.acquire()
108 started = []
109 def task():
110 started.append(None)
111 mut.acquire()
112 mut.release()
113 thread.start_new_thread(task, ())
114 while not started:
115 time.sleep(0.01)
Florent Xicluna2e6d2622010-03-20 00:17:46 +0000116 self.assertEqual(thread._count(), orig + 1)
Antoine Pitrou59c44f32009-10-30 17:07:08 +0000117 # Allow the task to finish.
118 mut.release()
119 # The only reliable way to be sure that the thread ended from the
120 # interpreter's point of view is to wait for the function object to be
121 # destroyed.
122 done = []
123 wr = weakref.ref(task, lambda _: done.append(None))
124 del task
125 while not done:
126 time.sleep(0.01)
Florent Xicluna2e6d2622010-03-20 00:17:46 +0000127 self.assertEqual(thread._count(), orig)
Antoine Pitrou59c44f32009-10-30 17:07:08 +0000128
Benjamin Petersonf73813a2012-04-02 11:15:17 -0400129 def test_save_exception_state_on_error(self):
130 # See issue #14474
131 def task():
132 started.release()
Benjamin Petersonf73813a2012-04-02 11:15:17 -0400133 raise SyntaxError
134 def mywrite(self, *args):
135 try:
136 raise ValueError
137 except ValueError:
138 pass
139 real_write(self, *args)
140 c = thread._count()
141 started = thread.allocate_lock()
142 with test_support.captured_output("stderr") as stderr:
143 real_write = stderr.write
144 stderr.write = mywrite
145 started.acquire()
146 thread.start_new_thread(task, ())
147 started.acquire()
148 while thread._count() > c:
Benjamin Peterson19e9beb2012-04-23 10:08:14 -0400149 time.sleep(0.01)
Benjamin Petersonf73813a2012-04-02 11:15:17 -0400150 self.assertIn("Traceback", stderr.getvalue())
151
Brett Cannon66865d22008-03-13 20:27:00 +0000152
153class Barrier:
154 def __init__(self, num_threads):
155 self.num_threads = num_threads
Fred Drake004d5e62000-10-23 17:22:08 +0000156 self.waiting = 0
Brett Cannon66865d22008-03-13 20:27:00 +0000157 self.checkin_mutex = thread.allocate_lock()
158 self.checkout_mutex = thread.allocate_lock()
159 self.checkout_mutex.acquire()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000160
Fred Drake004d5e62000-10-23 17:22:08 +0000161 def enter(self):
Brett Cannon66865d22008-03-13 20:27:00 +0000162 self.checkin_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000163 self.waiting = self.waiting + 1
Brett Cannon66865d22008-03-13 20:27:00 +0000164 if self.waiting == self.num_threads:
165 self.waiting = self.num_threads - 1
166 self.checkout_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000167 return
Brett Cannon66865d22008-03-13 20:27:00 +0000168 self.checkin_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000169
Brett Cannon66865d22008-03-13 20:27:00 +0000170 self.checkout_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000171 self.waiting = self.waiting - 1
172 if self.waiting == 0:
Brett Cannon66865d22008-03-13 20:27:00 +0000173 self.checkin_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000174 return
Brett Cannon66865d22008-03-13 20:27:00 +0000175 self.checkout_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000176
Guido van Rossumd3b68421994-05-23 12:17:36 +0000177
Brett Cannon66865d22008-03-13 20:27:00 +0000178class BarrierTest(BasicThreadTest):
Andrew MacIntyre92913322006-06-13 15:04:24 +0000179
Brett Cannon66865d22008-03-13 20:27:00 +0000180 def test_barrier(self):
181 self.bar = Barrier(NUMTASKS)
182 self.running = NUMTASKS
183 for i in range(NUMTASKS):
184 thread.start_new_thread(self.task2, (i,))
185 verbose_print("waiting for tasks to end")
186 self.done_mutex.acquire()
187 verbose_print("tasks done")
Andrew MacIntyre92913322006-06-13 15:04:24 +0000188
Brett Cannon66865d22008-03-13 20:27:00 +0000189 def task2(self, ident):
190 for i in range(NUMTRIPS):
191 if ident == 0:
192 # give it a good chance to enter the next
193 # barrier before the others are all out
194 # of the current one
Jeffrey Yasskina1458532008-03-18 04:56:06 +0000195 delay = 0
Brett Cannon66865d22008-03-13 20:27:00 +0000196 else:
197 with self.random_mutex:
Jeffrey Yasskina1458532008-03-18 04:56:06 +0000198 delay = random.random() / 10000.0
199 verbose_print("task %s will run for %sus" %
200 (ident, round(delay * 1e6)))
Brett Cannon66865d22008-03-13 20:27:00 +0000201 time.sleep(delay)
202 verbose_print("task %s entering %s" % (ident, i))
203 self.bar.enter()
204 verbose_print("task %s leaving barrier" % ident)
205 with self.running_mutex:
206 self.running -= 1
207 # Must release mutex before releasing done, else the main thread can
208 # exit and set mutex to None as part of global teardown; then
209 # mutex.release() raises AttributeError.
210 finished = self.running == 0
211 if finished:
212 self.done_mutex.release()
Andrew MacIntyre92913322006-06-13 15:04:24 +0000213
Andrew MacIntyre92913322006-06-13 15:04:24 +0000214
Antoine Pitrouc98efe02009-11-06 22:34:35 +0000215class LockTests(lock_tests.LockTests):
216 locktype = thread.allocate_lock
217
218
Gregory P. Smith9e5d1322010-03-01 01:22:39 +0000219class TestForkInThread(unittest.TestCase):
220 def setUp(self):
221 self.read_fd, self.write_fd = os.pipe()
222
Gregory P. Smith7512a902010-03-01 06:01:02 +0000223 @unittest.skipIf(sys.platform.startswith('win'),
Florent Xicluna2e6d2622010-03-20 00:17:46 +0000224 "This test is only appropriate for POSIX-like systems.")
225 @test_support.reap_threads
Gregory P. Smith9e5d1322010-03-01 01:22:39 +0000226 def test_forkinthread(self):
Gregory P. Smith9e5d1322010-03-01 01:22:39 +0000227 def thread1():
228 try:
229 pid = os.fork() # fork in a thread
230 except RuntimeError:
231 sys.exit(0) # exit the child
232
233 if pid == 0: # child
234 os.close(self.read_fd)
235 os.write(self.write_fd, "OK")
236 sys.exit(0)
237 else: # parent
238 os.close(self.write_fd)
239
240 thread.start_new_thread(thread1, ())
241 self.assertEqual(os.read(self.read_fd, 2), "OK",
242 "Unable to fork() in thread")
243
244 def tearDown(self):
245 try:
246 os.close(self.read_fd)
247 except OSError:
248 pass
249
250 try:
251 os.close(self.write_fd)
252 except OSError:
253 pass
254
255
Brett Cannon66865d22008-03-13 20:27:00 +0000256def test_main():
Gregory P. Smith9e5d1322010-03-01 01:22:39 +0000257 test_support.run_unittest(ThreadRunningTests, BarrierTest, LockTests,
258 TestForkInThread)
Andrew MacIntyre92913322006-06-13 15:04:24 +0000259
Brett Cannon66865d22008-03-13 20:27:00 +0000260if __name__ == "__main__":
261 test_main()