Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
Zope
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
Zope
Commits
291d1897
Commit
291d1897
authored
May 11, 2004
by
root
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use tagged version of src/ThreadedAsync
parent
db30982b
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
213 additions
and
0 deletions
+213
-0
lib/python/ThreadedAsync/LoopCallback.py
lib/python/ThreadedAsync/LoopCallback.py
+194
-0
lib/python/ThreadedAsync/__init__.py
lib/python/ThreadedAsync/__init__.py
+19
-0
No files found.
lib/python/ThreadedAsync/LoopCallback.py
0 → 100644
View file @
291d1897
##############################################################################
#
# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE
#
##############################################################################
"""Manage the asyncore mainloop in a multi-threaded app
In a multi-threaded application, only a single thread runs the
asyncore mainloop. This thread (the "mainloop thread") may not start
the mainloop before another thread needs to perform an async action
that requires it. As a result, other threads need to coordinate with
the mainloop thread to find out whether the mainloop is running.
This module implements a callback mechanism that allows other threads
to be notified when the mainloop starts. A thread calls
register_loop_callback() to register interest. When the mainloop
thread calls loop(), each registered callback will be called with the
socket map as its first argument.
"""
__version__
=
'$Revision: 1.14 $'
[
11
:
-
2
]
import
asyncore
import
select
import
thread
import
time
from
errno
import
EINTR
_loop_lock
=
thread
.
allocate_lock
()
_looping
=
None
_loop_callbacks
=
[]
def
remove_loop_callback
(
callback
):
"""Remove a callback function registered earlier.
This is useful if loop() was never called.
"""
for
i
in
range
(
len
(
_loop_callbacks
)):
if
_loop_callbacks
[
i
][
0
]
==
callback
:
del
_loop_callbacks
[
i
]
return
def
register_loop_callback
(
callback
,
args
=
(),
kw
=
None
):
"""Register callback function to be called when mainloop starts
The callable object callback will be invokved when the mainloop
starts. If the mainloop is currently running, the callback will
be invoked immediately.
The callback will be called with a single argument, the mainloop
socket map, unless the optional args or kw arguments are used.
args defines a tuple of extra arguments to pass after the socket
map. kw defines a dictionary of keyword arguments.
"""
_loop_lock
.
acquire
()
try
:
if
_looping
is
not
None
:
callback
(
_looping
,
*
args
,
**
(
kw
or
{}))
else
:
_loop_callbacks
.
append
((
callback
,
args
,
kw
))
finally
:
_loop_lock
.
release
()
def
remove_loop_callback
(
callback
):
"""Remove a callback function registered earlier.
This is useful if loop() was never called.
"""
for
i
in
range
(
len
(
_loop_callbacks
)):
if
_loop_callbacks
[
i
][
0
]
==
callback
:
del
_loop_callbacks
[
i
]
return
def
_start_loop
(
map
):
_loop_lock
.
acquire
()
try
:
global
_looping
_looping
=
map
while
_loop_callbacks
:
cb
,
args
,
kw
=
_loop_callbacks
.
pop
()
cb
(
map
,
*
args
,
**
(
kw
or
{}))
finally
:
_loop_lock
.
release
()
def
_stop_loop
():
_loop_lock
.
acquire
()
try
:
global
_looping
_looping
=
None
finally
:
_loop_lock
.
release
()
def
poll
(
timeout
=
0.0
,
map
=
None
):
"""A copy of asyncore.poll() with a bug fixed (see comment).
(asyncore.poll2() and .poll3() don't have this bug.)
"""
if
map
is
None
:
map
=
asyncore
.
socket_map
if
map
:
r
=
[];
w
=
[];
e
=
[]
for
fd
,
obj
in
map
.
items
():
if
obj
.
readable
():
r
.
append
(
fd
)
if
obj
.
writable
():
w
.
append
(
fd
)
if
[]
==
r
==
w
==
e
:
time
.
sleep
(
timeout
)
else
:
try
:
r
,
w
,
e
=
select
.
select
(
r
,
w
,
e
,
timeout
)
except
select
.
error
,
err
:
if
err
[
0
]
!=
EINTR
:
raise
else
:
# This part is missing in asyncore before Python 2.3
return
for
fd
in
r
:
obj
=
map
.
get
(
fd
)
if
obj
is
not
None
:
try
:
obj
.
handle_read_event
()
except
asyncore
.
ExitNow
:
raise
asyncore
.
ExitNow
except
:
obj
.
handle_error
()
for
fd
in
w
:
obj
=
map
.
get
(
fd
)
if
obj
is
not
None
:
try
:
obj
.
handle_write_event
()
except
asyncore
.
ExitNow
:
raise
asyncore
.
ExitNow
except
:
obj
.
handle_error
()
def
loop
(
timeout
=
30.0
,
use_poll
=
0
,
map
=
None
):
"""Invoke asyncore mainloop
This function functions like the regular asyncore.loop() function
except that it also triggers ThreadedAsync callback functions
before starting the loop.
"""
global
exit_status
exit_status
=
None
if
use_poll
:
if
hasattr
(
select
,
'poll'
):
poll_fun
=
asyncore
.
poll3
else
:
poll_fun
=
asyncore
.
poll2
else
:
poll_fun
=
poll
if
map
is
None
:
map
=
asyncore
.
socket_map
_start_loop
(
map
)
while
map
and
exit_status
is
None
:
poll_fun
(
timeout
,
map
)
_stop_loop
()
# This module used to do something evil -- it rebound asyncore.loop to the
# above loop() function. What was evil about this is that if you added some
# debugging to asyncore.loop, you'd spend 6 hours debugging why your debugging
# code wasn't called!
#
# Code should instead explicitly call ThreadedAsync.loop() instead of
# asyncore.loop(). Most of ZODB has been fixed, but ripping this out may
# break 3rd party code. So we'll issue a warning and let it continue -- for
# now.
##def deprecated_loop(*args, **kws):
## import warnings
## warnings.warn("""\
##ThreadedAsync.loop() called through sneaky asyncore.loop() rebinding.
##You should change your code to call ThreadedAsync.loop() explicitly.""",
## DeprecationWarning)
## loop(*args, **kws)
##asyncore.loop = deprecated_loop
# XXX Remove this once we've updated ZODB4 since they share this package
asyncore
.
loop
=
loop
lib/python/ThreadedAsync/__init__.py
0 → 100644
View file @
291d1897
##############################################################################
#
# Copyright (c) 2001, 2002 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE.
#
##############################################################################
"""Manage the asyncore mainloop in a multi-threaded app.
$Id: __init__.py,v 1.9 2004/02/27 10:27:37 philikon Exp $
"""
from
LoopCallback
import
register_loop_callback
,
loop
,
remove_loop_callback
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment