@@ -134,6 +134,10 @@ def Event(self):
134
134
return threading .Event ()
135
135
136
136
137
+ class MainThreadOnlyExecModel (ThreadExecModel ):
138
+ backend = "main_thread_only"
139
+
140
+
137
141
class EventletExecModel (ExecModel ):
138
142
backend = "eventlet"
139
143
@@ -254,6 +258,8 @@ def get_execmodel(backend):
254
258
return backend
255
259
if backend == "thread" :
256
260
return ThreadExecModel ()
261
+ elif backend == "main_thread_only" :
262
+ return MainThreadOnlyExecModel ()
257
263
elif backend == "eventlet" :
258
264
return EventletExecModel ()
259
265
elif backend == "gevent" :
@@ -322,7 +328,7 @@ def __init__(self, execmodel, hasprimary=False):
322
328
self ._shuttingdown = False
323
329
self ._waitall_events = []
324
330
if hasprimary :
325
- if self .execmodel .backend != "thread" :
331
+ if self .execmodel .backend not in ( "thread" , "main_thread_only" ) :
326
332
raise ValueError ("hasprimary=True requires thread model" )
327
333
self ._primary_thread_task_ready = self .execmodel .Event ()
328
334
else :
@@ -332,7 +338,7 @@ def integrate_as_primary_thread(self):
332
338
"""integrate the thread with which we are called as a primary
333
339
thread for executing functions triggered with spawn().
334
340
"""
335
- assert self .execmodel .backend == "thread" , self .execmodel
341
+ assert self .execmodel .backend in ( "thread" , "main_thread_only" ) , self .execmodel
336
342
primary_thread_task_ready = self ._primary_thread_task_ready
337
343
# interacts with code at REF1
338
344
while 1 :
@@ -345,7 +351,11 @@ def integrate_as_primary_thread(self):
345
351
with self ._running_lock :
346
352
if self ._shuttingdown :
347
353
break
348
- primary_thread_task_ready .clear ()
354
+ # Only clear if _try_send_to_primary_thread has not
355
+ # yet set the next self._primary_thread_task reply
356
+ # after waiting for this one to complete.
357
+ if reply is self ._primary_thread_task :
358
+ primary_thread_task_ready .clear ()
349
359
350
360
def trigger_shutdown (self ):
351
361
with self ._running_lock :
@@ -376,6 +386,19 @@ def _try_send_to_primary_thread(self, reply):
376
386
# wake up primary thread
377
387
primary_thread_task_ready .set ()
378
388
return True
389
+ elif (
390
+ self .execmodel .backend == "main_thread_only"
391
+ and self ._primary_thread_task is not None
392
+ ):
393
+ self ._primary_thread_task .waitfinish ()
394
+ self ._primary_thread_task = reply
395
+ # wake up primary thread (it's okay if this is already set
396
+ # because we waited for the previous task to finish above
397
+ # and integrate_as_primary_thread will not clear it when
398
+ # it enters self._running_lock if it detects that a new
399
+ # task is available)
400
+ primary_thread_task_ready .set ()
401
+ return True
379
402
return False
380
403
381
404
def spawn (self , func , * args , ** kwargs ):
@@ -1132,7 +1155,7 @@ def serve(self):
1132
1155
def trace (msg ):
1133
1156
self ._trace ("[serve] " + msg )
1134
1157
1135
- hasprimary = self .execmodel .backend == "thread"
1158
+ hasprimary = self .execmodel .backend in ( "thread" , "main_thread_only" )
1136
1159
self ._execpool = WorkerPool (self .execmodel , hasprimary = hasprimary )
1137
1160
trace ("spawning receiver thread" )
1138
1161
self ._initreceive ()
0 commit comments