2008-06-11 13:44:04 -03:00
|
|
|
#
|
|
|
|
# A higher level module for using sockets (or Windows named pipes)
|
|
|
|
#
|
|
|
|
# multiprocessing/connection.py
|
|
|
|
#
|
2010-12-13 21:38:16 -04:00
|
|
|
# Copyright (c) 2006-2008, R Oudkerk
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions
|
|
|
|
# are met:
|
|
|
|
#
|
|
|
|
# 1. Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# 3. Neither the name of author nor the names of any contributors may be
|
|
|
|
# used to endorse or promote products derived from this software
|
|
|
|
# without specific prior written permission.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS "AS IS" AND
|
|
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
# SUCH DAMAGE.
|
2008-06-11 13:44:04 -03:00
|
|
|
#
|
|
|
|
|
|
|
|
__all__ = [ 'Client', 'Listener', 'Pipe' ]
|
|
|
|
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import socket
|
Merged revisions 65459,65472,65481,65518,65536,65581,65609,65637,65641,65644-65645 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r65459 | gregory.p.smith | 2008-08-04 00:13:29 +0000 (Mon, 04 Aug 2008) | 4 lines
- Issue #1857: subprocess.Popen.poll gained an additional _deadstate keyword
argument in python 2.5, this broke code that subclassed Popen to include its
own poll method. Fixed my moving _deadstate to an _internal_poll method.
........
r65472 | andrew.kuchling | 2008-08-04 01:43:43 +0000 (Mon, 04 Aug 2008) | 3 lines
Bug 3228: Explicitly supply the file mode to avoid creating executable files,
and add corresponding tests.
Possible 2.5 backport candidate
........
r65481 | gregory.p.smith | 2008-08-04 07:33:37 +0000 (Mon, 04 Aug 2008) | 22 lines
Adds a sanity check to avoid a *very rare* infinite loop due to a corrupt tls
key list data structure in the thread startup path.
This change is a companion to r60148 which already successfully dealt with a
similar issue on thread shutdown.
In particular this loop has been observed happening from this call path:
#0 in find_key ()
#1 in PyThread_set_key_value ()
#2 in _PyGILState_NoteThreadState ()
#3 in PyThreadState_New ()
#4 in t_bootstrap ()
#5 in pthread_start_thread ()
I don't know how this happens but it does, *very* rarely. On more than
one hardware platform. I have not been able to reproduce it manually.
(A flaky mutex implementation on the system in question is one hypothesis).
As with r60148, the spinning we managed to observe in the wild was due to a
single list element pointing back upon itself.
........
r65518 | mark.dickinson | 2008-08-04 21:30:09 +0000 (Mon, 04 Aug 2008) | 7 lines
Issue #1481296: (again!) Make conversion of a float NaN to an int or
long raise ValueError instead of returning 0. Also, change the error
message for conversion of an infinity to an integer, replacing 'long' by
'integer', so that it's appropriate for both long(float('inf')) and
int(float('inf')).
........
r65536 | andrew.kuchling | 2008-08-05 01:00:57 +0000 (Tue, 05 Aug 2008) | 1 line
Bug 3228: take a test from Niels Gustaebel's patch, and based on his patch, check for having os.stat available
........
r65581 | guido.van.rossum | 2008-08-07 18:51:38 +0000 (Thu, 07 Aug 2008) | 3 lines
Patch by Ian Charnas from issue 3517.
Add F_FULLFSYNC if it exists (OS X only so far).
........
r65609 | antoine.pitrou | 2008-08-09 17:22:25 +0000 (Sat, 09 Aug 2008) | 3 lines
#3205: bz2 iterator fails silently on MemoryError
........
r65637 | georg.brandl | 2008-08-11 09:07:59 +0000 (Mon, 11 Aug 2008) | 3 lines
- Issue #3537: Fix an assertion failure when an empty but presized dict
object was stored in the freelist.
........
r65641 | jesse.noller | 2008-08-11 14:28:07 +0000 (Mon, 11 Aug 2008) | 2 lines
Remove the fqdn call for issue 3270
........
r65644 | antoine.pitrou | 2008-08-11 17:21:36 +0000 (Mon, 11 Aug 2008) | 3 lines
#3134: shutil referenced undefined WindowsError symbol
........
r65645 | jesse.noller | 2008-08-11 19:00:15 +0000 (Mon, 11 Aug 2008) | 2 lines
Fix the connection refused error part of issue 3419, use errno module instead of a static list of possible connection refused messages.
........
2008-08-12 05:35:52 -03:00
|
|
|
import errno
|
2008-06-11 13:44:04 -03:00
|
|
|
import time
|
|
|
|
import tempfile
|
|
|
|
import itertools
|
|
|
|
|
|
|
|
import _multiprocessing
|
2008-08-24 22:53:32 -03:00
|
|
|
from multiprocessing import current_process, AuthenticationError
|
2008-06-11 13:44:04 -03:00
|
|
|
from multiprocessing.util import get_temp_dir, Finalize, sub_debug, debug
|
|
|
|
from multiprocessing.forking import duplicate, close
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
|
|
|
|
BUFSIZE = 8192
|
2009-11-13 18:35:18 -04:00
|
|
|
# A very generous timeout when it comes to local connections...
|
|
|
|
CONNECTION_TIMEOUT = 20.
|
2008-06-11 13:44:04 -03:00
|
|
|
|
|
|
|
_mmap_counter = itertools.count()
|
|
|
|
|
|
|
|
default_family = 'AF_INET'
|
|
|
|
families = ['AF_INET']
|
|
|
|
|
|
|
|
if hasattr(socket, 'AF_UNIX'):
|
|
|
|
default_family = 'AF_UNIX'
|
|
|
|
families += ['AF_UNIX']
|
|
|
|
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
default_family = 'AF_PIPE'
|
|
|
|
families += ['AF_PIPE']
|
|
|
|
|
2009-11-13 18:35:18 -04:00
|
|
|
|
|
|
|
def _init_timeout(timeout=CONNECTION_TIMEOUT):
|
|
|
|
return time.time() + timeout
|
|
|
|
|
|
|
|
def _check_timeout(t):
|
|
|
|
return time.time() > t
|
|
|
|
|
2008-06-11 13:44:04 -03:00
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
|
|
|
|
def arbitrary_address(family):
|
|
|
|
'''
|
|
|
|
Return an arbitrary free address for the given family
|
|
|
|
'''
|
|
|
|
if family == 'AF_INET':
|
|
|
|
return ('localhost', 0)
|
|
|
|
elif family == 'AF_UNIX':
|
|
|
|
return tempfile.mktemp(prefix='listener-', dir=get_temp_dir())
|
|
|
|
elif family == 'AF_PIPE':
|
|
|
|
return tempfile.mktemp(prefix=r'\\.\pipe\pyc-%d-%d-' %
|
|
|
|
(os.getpid(), next(_mmap_counter)))
|
|
|
|
else:
|
|
|
|
raise ValueError('unrecognized family')
|
|
|
|
|
2012-04-01 12:19:09 -03:00
|
|
|
def _validate_family(family):
|
|
|
|
'''
|
|
|
|
Checks if the family is valid for the current environment.
|
|
|
|
'''
|
|
|
|
if sys.platform != 'win32' and family == 'AF_PIPE':
|
|
|
|
raise ValueError('Family %s is not recognized.' % family)
|
|
|
|
|
2012-04-03 15:12:23 -03:00
|
|
|
if sys.platform == 'win32' and family == 'AF_UNIX':
|
|
|
|
# double check
|
|
|
|
if not hasattr(socket, family):
|
|
|
|
raise ValueError('Family %s is not recognized.' % family)
|
2008-06-11 13:44:04 -03:00
|
|
|
|
|
|
|
def address_type(address):
|
|
|
|
'''
|
|
|
|
Return the types of the address
|
|
|
|
|
|
|
|
This can be 'AF_INET', 'AF_UNIX', or 'AF_PIPE'
|
|
|
|
'''
|
|
|
|
if type(address) == tuple:
|
|
|
|
return 'AF_INET'
|
|
|
|
elif type(address) is str and address.startswith('\\\\'):
|
|
|
|
return 'AF_PIPE'
|
|
|
|
elif type(address) is str:
|
|
|
|
return 'AF_UNIX'
|
|
|
|
else:
|
|
|
|
raise ValueError('address type of %r unrecognized' % address)
|
|
|
|
|
|
|
|
#
|
|
|
|
# Public functions
|
|
|
|
#
|
|
|
|
|
|
|
|
class Listener(object):
|
|
|
|
'''
|
|
|
|
Returns a listener object.
|
|
|
|
|
|
|
|
This is a wrapper for a bound socket which is 'listening' for
|
|
|
|
connections, or for a Windows named pipe.
|
|
|
|
'''
|
|
|
|
def __init__(self, address=None, family=None, backlog=1, authkey=None):
|
|
|
|
family = family or (address and address_type(address)) \
|
|
|
|
or default_family
|
|
|
|
address = address or arbitrary_address(family)
|
|
|
|
|
2012-04-01 12:19:09 -03:00
|
|
|
_validate_family(family)
|
2008-06-11 13:44:04 -03:00
|
|
|
if family == 'AF_PIPE':
|
|
|
|
self._listener = PipeListener(address, backlog)
|
|
|
|
else:
|
|
|
|
self._listener = SocketListener(address, family, backlog)
|
|
|
|
|
|
|
|
if authkey is not None and not isinstance(authkey, bytes):
|
|
|
|
raise TypeError('authkey should be a byte string')
|
|
|
|
|
|
|
|
self._authkey = authkey
|
|
|
|
|
|
|
|
def accept(self):
|
|
|
|
'''
|
|
|
|
Accept a connection on the bound socket or named pipe of `self`.
|
|
|
|
|
|
|
|
Returns a `Connection` object.
|
|
|
|
'''
|
|
|
|
c = self._listener.accept()
|
|
|
|
if self._authkey:
|
|
|
|
deliver_challenge(c, self._authkey)
|
|
|
|
answer_challenge(c, self._authkey)
|
|
|
|
return c
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
'''
|
|
|
|
Close the bound socket or named pipe of `self`.
|
|
|
|
'''
|
|
|
|
return self._listener.close()
|
|
|
|
|
|
|
|
address = property(lambda self: self._listener._address)
|
|
|
|
last_accepted = property(lambda self: self._listener._last_accepted)
|
|
|
|
|
|
|
|
|
|
|
|
def Client(address, family=None, authkey=None):
|
|
|
|
'''
|
|
|
|
Returns a connection to the address of a `Listener`
|
|
|
|
'''
|
|
|
|
family = family or address_type(address)
|
2012-04-01 12:19:09 -03:00
|
|
|
_validate_family(family)
|
2008-06-11 13:44:04 -03:00
|
|
|
if family == 'AF_PIPE':
|
|
|
|
c = PipeClient(address)
|
|
|
|
else:
|
|
|
|
c = SocketClient(address)
|
|
|
|
|
|
|
|
if authkey is not None and not isinstance(authkey, bytes):
|
|
|
|
raise TypeError('authkey should be a byte string')
|
|
|
|
|
|
|
|
if authkey is not None:
|
|
|
|
answer_challenge(c, authkey)
|
|
|
|
deliver_challenge(c, authkey)
|
|
|
|
|
|
|
|
return c
|
|
|
|
|
|
|
|
|
|
|
|
if sys.platform != 'win32':
|
|
|
|
|
|
|
|
def Pipe(duplex=True):
|
|
|
|
'''
|
|
|
|
Returns pair of connection objects at either end of a pipe
|
|
|
|
'''
|
|
|
|
if duplex:
|
|
|
|
s1, s2 = socket.socketpair()
|
2012-07-27 10:06:11 -03:00
|
|
|
s1.setblocking(True)
|
|
|
|
s2.setblocking(True)
|
2008-06-11 13:44:04 -03:00
|
|
|
c1 = _multiprocessing.Connection(os.dup(s1.fileno()))
|
|
|
|
c2 = _multiprocessing.Connection(os.dup(s2.fileno()))
|
|
|
|
s1.close()
|
|
|
|
s2.close()
|
|
|
|
else:
|
|
|
|
fd1, fd2 = os.pipe()
|
|
|
|
c1 = _multiprocessing.Connection(fd1, writable=False)
|
|
|
|
c2 = _multiprocessing.Connection(fd2, readable=False)
|
|
|
|
|
|
|
|
return c1, c2
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
2010-08-04 12:47:24 -03:00
|
|
|
from _multiprocessing import win32
|
2008-06-11 13:44:04 -03:00
|
|
|
|
|
|
|
def Pipe(duplex=True):
|
|
|
|
'''
|
|
|
|
Returns pair of connection objects at either end of a pipe
|
|
|
|
'''
|
|
|
|
address = arbitrary_address('AF_PIPE')
|
|
|
|
if duplex:
|
|
|
|
openmode = win32.PIPE_ACCESS_DUPLEX
|
|
|
|
access = win32.GENERIC_READ | win32.GENERIC_WRITE
|
|
|
|
obsize, ibsize = BUFSIZE, BUFSIZE
|
|
|
|
else:
|
|
|
|
openmode = win32.PIPE_ACCESS_INBOUND
|
|
|
|
access = win32.GENERIC_WRITE
|
|
|
|
obsize, ibsize = 0, BUFSIZE
|
|
|
|
|
|
|
|
h1 = win32.CreateNamedPipe(
|
|
|
|
address, openmode,
|
|
|
|
win32.PIPE_TYPE_MESSAGE | win32.PIPE_READMODE_MESSAGE |
|
|
|
|
win32.PIPE_WAIT,
|
|
|
|
1, obsize, ibsize, win32.NMPWAIT_WAIT_FOREVER, win32.NULL
|
|
|
|
)
|
|
|
|
h2 = win32.CreateFile(
|
|
|
|
address, access, 0, win32.NULL, win32.OPEN_EXISTING, 0, win32.NULL
|
|
|
|
)
|
|
|
|
win32.SetNamedPipeHandleState(
|
|
|
|
h2, win32.PIPE_READMODE_MESSAGE, None, None
|
|
|
|
)
|
|
|
|
|
|
|
|
try:
|
|
|
|
win32.ConnectNamedPipe(h1, win32.NULL)
|
|
|
|
except WindowsError as e:
|
|
|
|
if e.args[0] != win32.ERROR_PIPE_CONNECTED:
|
|
|
|
raise
|
|
|
|
|
|
|
|
c1 = _multiprocessing.PipeConnection(h1, writable=duplex)
|
|
|
|
c2 = _multiprocessing.PipeConnection(h2, readable=duplex)
|
|
|
|
|
|
|
|
return c1, c2
|
|
|
|
|
|
|
|
#
|
|
|
|
# Definitions for connections based on sockets
|
|
|
|
#
|
|
|
|
|
|
|
|
class SocketListener(object):
|
|
|
|
'''
|
Merged revisions 65437,65469,65476,65480,65502,65528,65539,65543,65558,65561-65562,65565,65591,65601,65608,65610,65639 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r65437 | georg.brandl | 2008-08-03 22:28:55 +0000 (Sun, 03 Aug 2008) | 2 lines
Note the removal of several committers.
........
r65469 | gregory.p.smith | 2008-08-04 01:03:50 +0000 (Mon, 04 Aug 2008) | 3 lines
issue1606: Add warnings to the subprocess documentation about common pitfalls
of using pipes that cause deadlocks.
........
r65476 | georg.brandl | 2008-08-04 06:29:36 +0000 (Mon, 04 Aug 2008) | 2 lines
Fix markup.
........
r65480 | georg.brandl | 2008-08-04 07:31:50 +0000 (Mon, 04 Aug 2008) | 3 lines
Clarify the meaning of the select() parameters and sync
names with docstring.
........
r65502 | gregory.p.smith | 2008-08-04 18:34:07 +0000 (Mon, 04 Aug 2008) | 2 lines
more cleanup ups of the recently added warnings in the subprocess docs.
........
r65528 | brett.cannon | 2008-08-04 21:52:25 +0000 (Mon, 04 Aug 2008) | 4 lines
Add a note about all the modules/packages changed to silence -3 warnings. More
changes are needed once some decisions are made, but this is the work up to this
point.
........
r65539 | andrew.kuchling | 2008-08-05 01:38:08 +0000 (Tue, 05 Aug 2008) | 6 lines
#3367 from Kristjan Valur Jonsson:
If a PyTokenizer_FromString() is called with an empty string, the
tokenizer's line_start member never gets initialized. Later, it is
compared with the token pointer 'a' in parsetok.c:193 and that behavior
can result in undefined behavior.
........
r65543 | andrew.kuchling | 2008-08-05 02:05:23 +0000 (Tue, 05 Aug 2008) | 1 line
#3367: revert rev. 65539: this change causes test_parser to fail
........
r65558 | georg.brandl | 2008-08-06 17:20:41 +0000 (Wed, 06 Aug 2008) | 2 lines
Fix longstringitem definition. #3505.
........
r65561 | mark.dickinson | 2008-08-06 20:12:30 +0000 (Wed, 06 Aug 2008) | 2 lines
Docstring typo
........
r65562 | mark.dickinson | 2008-08-06 21:36:57 +0000 (Wed, 06 Aug 2008) | 2 lines
Remove duplicate import
........
r65565 | andrew.kuchling | 2008-08-07 01:47:34 +0000 (Thu, 07 Aug 2008) | 1 line
Add some items
........
r65591 | georg.brandl | 2008-08-08 06:42:20 +0000 (Fri, 08 Aug 2008) | 2 lines
#3519: callee is an expression too.
........
r65601 | georg.brandl | 2008-08-08 15:34:34 +0000 (Fri, 08 Aug 2008) | 2 lines
Remove mention of backquotes in the tutorial.
........
r65608 | guido.van.rossum | 2008-08-09 14:55:34 +0000 (Sat, 09 Aug 2008) | 2 lines
Add news item about _sre.compile() re-bytecode validator.
........
r65610 | antoine.pitrou | 2008-08-09 17:27:23 +0000 (Sat, 09 Aug 2008) | 3 lines
move NEWS entry to the appropriate section (oops!)
........
r65639 | georg.brandl | 2008-08-11 10:27:31 +0000 (Mon, 11 Aug 2008) | 2 lines
#3540: fix exception name.
........
2008-08-12 05:18:18 -03:00
|
|
|
Representation of a socket which is bound to an address and listening
|
2008-06-11 13:44:04 -03:00
|
|
|
'''
|
|
|
|
def __init__(self, address, family, backlog=1):
|
|
|
|
self._socket = socket.socket(getattr(socket, family))
|
2012-02-04 09:55:53 -04:00
|
|
|
try:
|
|
|
|
self._socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
2012-07-27 10:06:11 -03:00
|
|
|
self._socket.setblocking(True)
|
2012-02-04 09:55:53 -04:00
|
|
|
self._socket.bind(address)
|
|
|
|
self._socket.listen(backlog)
|
|
|
|
self._address = self._socket.getsockname()
|
|
|
|
except socket.error:
|
|
|
|
self._socket.close()
|
|
|
|
raise
|
2008-06-11 13:44:04 -03:00
|
|
|
self._family = family
|
|
|
|
self._last_accepted = None
|
|
|
|
|
|
|
|
if family == 'AF_UNIX':
|
|
|
|
self._unlink = Finalize(
|
Merged revisions 64722,64729,64753,64845-64846,64849,64871,64880-64882,64885,64888,64897,64900-64901,64915,64926-64929,64938-64941,64944,64961,64966,64973 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
r64722 | georg.brandl | 2008-07-05 12:13:36 +0200 (Sat, 05 Jul 2008) | 4 lines
#2663: support an *ignore* argument to shutil.copytree(). Patch by Tarek Ziade.
This is a new feature, but Barry authorized adding it in the beta period.
........
r64729 | mark.dickinson | 2008-07-05 13:33:52 +0200 (Sat, 05 Jul 2008) | 5 lines
Issue 3188: accept float('infinity') as well as float('inf'). This
makes the float constructor behave in the same way as specified
by various other language standards, including C99, IEEE 754r,
and the IBM Decimal standard.
........
r64753 | gregory.p.smith | 2008-07-06 05:35:58 +0200 (Sun, 06 Jul 2008) | 4 lines
- Issue #2862: Make int and float freelist management consistent with other
freelists. Changes their CompactFreeList apis into ClearFreeList apis and
calls them via gc.collect().
........
r64845 | raymond.hettinger | 2008-07-10 16:03:19 +0200 (Thu, 10 Jul 2008) | 1 line
Issue 3301: Bisect functions behaved badly when lo was negative.
........
r64846 | raymond.hettinger | 2008-07-10 16:34:57 +0200 (Thu, 10 Jul 2008) | 1 line
Issue 3285: Fractions from_float() and from_decimal() accept Integral arguments.
........
r64849 | andrew.kuchling | 2008-07-10 16:43:31 +0200 (Thu, 10 Jul 2008) | 1 line
Wording changes
........
r64871 | raymond.hettinger | 2008-07-11 14:00:21 +0200 (Fri, 11 Jul 2008) | 1 line
Add cautionary note on the use of PySequence_Fast_ITEMS.
........
r64880 | amaury.forgeotdarc | 2008-07-11 23:28:25 +0200 (Fri, 11 Jul 2008) | 5 lines
#3317 in zipfile module, restore the previous names of global variables:
some applications relied on them.
Also remove duplicated lines.
........
r64881 | amaury.forgeotdarc | 2008-07-11 23:45:06 +0200 (Fri, 11 Jul 2008) | 3 lines
#3342: In tracebacks, printed source lines were not indented since r62555.
#3343: Py_DisplaySourceLine should be a private function. Rename it to _Py_DisplaySourceLine.
........
r64882 | josiah.carlson | 2008-07-12 00:17:14 +0200 (Sat, 12 Jul 2008) | 2 lines
Fix for the AttributeError in test_asynchat.
........
r64885 | josiah.carlson | 2008-07-12 01:26:59 +0200 (Sat, 12 Jul 2008) | 2 lines
Fixed test for asyncore.
........
r64888 | matthias.klose | 2008-07-12 09:51:48 +0200 (Sat, 12 Jul 2008) | 2 lines
- Fix bashisms in Tools/faqwiz/move-faqwiz.sh
........
r64897 | benjamin.peterson | 2008-07-12 22:16:19 +0200 (Sat, 12 Jul 2008) | 1 line
fix various doc typos #3320
........
r64900 | alexandre.vassalotti | 2008-07-13 00:06:53 +0200 (Sun, 13 Jul 2008) | 2 lines
Fixed typo.
........
r64901 | benjamin.peterson | 2008-07-13 01:41:19 +0200 (Sun, 13 Jul 2008) | 1 line
#1778443 robotparser fixes from Aristotelis Mikropoulos
........
r64915 | nick.coghlan | 2008-07-13 16:52:36 +0200 (Sun, 13 Jul 2008) | 1 line
Fix issue 3221 by emitting a RuntimeWarning instead of raising SystemError when the parent module can't be found during an absolute import (likely due to non-PEP 361 aware code which sets a module level __package__ attribute)
........
r64926 | martin.v.loewis | 2008-07-13 22:31:49 +0200 (Sun, 13 Jul 2008) | 2 lines
Add turtle into the module index.
........
r64927 | alexandre.vassalotti | 2008-07-13 22:42:44 +0200 (Sun, 13 Jul 2008) | 3 lines
Issue #3274: Use a less common identifier for the temporary variable
in Py_CLEAR().
........
r64928 | andrew.kuchling | 2008-07-13 23:43:25 +0200 (Sun, 13 Jul 2008) | 1 line
Re-word
........
r64929 | andrew.kuchling | 2008-07-13 23:43:52 +0200 (Sun, 13 Jul 2008) | 1 line
Add various items; move ctypes items into a subsection of their own
........
r64938 | andrew.kuchling | 2008-07-14 02:35:32 +0200 (Mon, 14 Jul 2008) | 1 line
Typo fixes
........
r64939 | andrew.kuchling | 2008-07-14 02:40:55 +0200 (Mon, 14 Jul 2008) | 1 line
Typo fix
........
r64940 | andrew.kuchling | 2008-07-14 03:18:16 +0200 (Mon, 14 Jul 2008) | 1 line
Typo fix
........
r64941 | andrew.kuchling | 2008-07-14 03:18:31 +0200 (Mon, 14 Jul 2008) | 1 line
Expand the multiprocessing section
........
r64944 | gregory.p.smith | 2008-07-14 08:06:48 +0200 (Mon, 14 Jul 2008) | 7 lines
Fix posix.fork1() / os.fork1() to only call PyOS_AfterFork() in the child
process rather than both parent and child.
Does anyone actually use fork1()? It appears to be a Solaris thing
but if Python is built with pthreads on Solaris, fork1() and fork()
should be the same.
........
r64961 | jesse.noller | 2008-07-15 15:47:33 +0200 (Tue, 15 Jul 2008) | 1 line
multiprocessing/connection.py patch to remove fqdn oddness for issue 3270
........
r64966 | nick.coghlan | 2008-07-15 17:40:22 +0200 (Tue, 15 Jul 2008) | 1 line
Add missing NEWS entry for r64962
........
r64973 | jesse.noller | 2008-07-15 20:29:18 +0200 (Tue, 15 Jul 2008) | 1 line
Revert 3270 patch: self._address is in pretty widespread use, need to revisit
........
2008-07-16 09:55:28 -03:00
|
|
|
self, os.unlink, args=(address,), exitpriority=0
|
2008-06-11 13:44:04 -03:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
self._unlink = None
|
|
|
|
|
|
|
|
def accept(self):
|
|
|
|
s, self._last_accepted = self._socket.accept()
|
2012-07-27 10:06:11 -03:00
|
|
|
s.setblocking(True)
|
2008-06-11 13:44:04 -03:00
|
|
|
fd = duplicate(s.fileno())
|
|
|
|
conn = _multiprocessing.Connection(fd)
|
|
|
|
s.close()
|
|
|
|
return conn
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
self._socket.close()
|
|
|
|
if self._unlink is not None:
|
|
|
|
self._unlink()
|
|
|
|
|
|
|
|
|
|
|
|
def SocketClient(address):
|
|
|
|
'''
|
|
|
|
Return a connection object connected to the socket given by `address`
|
|
|
|
'''
|
|
|
|
family = address_type(address)
|
2011-01-03 11:47:59 -04:00
|
|
|
with socket.socket( getattr(socket, family) ) as s:
|
2012-07-27 10:06:11 -03:00
|
|
|
s.setblocking(True)
|
2011-01-03 11:47:59 -04:00
|
|
|
t = _init_timeout()
|
|
|
|
|
|
|
|
while 1:
|
|
|
|
try:
|
|
|
|
s.connect(address)
|
|
|
|
except socket.error as e:
|
|
|
|
if e.args[0] != errno.ECONNREFUSED or _check_timeout(t):
|
|
|
|
debug('failed to connect to address %s', address)
|
|
|
|
raise
|
|
|
|
time.sleep(0.01)
|
|
|
|
else:
|
|
|
|
break
|
2008-06-11 13:44:04 -03:00
|
|
|
else:
|
2011-01-03 11:47:59 -04:00
|
|
|
raise
|
2008-06-11 13:44:04 -03:00
|
|
|
|
2011-01-03 11:47:59 -04:00
|
|
|
fd = duplicate(s.fileno())
|
2008-06-11 13:44:04 -03:00
|
|
|
conn = _multiprocessing.Connection(fd)
|
|
|
|
return conn
|
|
|
|
|
|
|
|
#
|
|
|
|
# Definitions for connections based on named pipes
|
|
|
|
#
|
|
|
|
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
|
|
|
|
class PipeListener(object):
|
|
|
|
'''
|
|
|
|
Representation of a named pipe
|
|
|
|
'''
|
|
|
|
def __init__(self, address, backlog=None):
|
|
|
|
self._address = address
|
|
|
|
handle = win32.CreateNamedPipe(
|
|
|
|
address, win32.PIPE_ACCESS_DUPLEX,
|
|
|
|
win32.PIPE_TYPE_MESSAGE | win32.PIPE_READMODE_MESSAGE |
|
|
|
|
win32.PIPE_WAIT,
|
|
|
|
win32.PIPE_UNLIMITED_INSTANCES, BUFSIZE, BUFSIZE,
|
|
|
|
win32.NMPWAIT_WAIT_FOREVER, win32.NULL
|
|
|
|
)
|
|
|
|
self._handle_queue = [handle]
|
|
|
|
self._last_accepted = None
|
|
|
|
|
|
|
|
sub_debug('listener created with address=%r', self._address)
|
|
|
|
|
|
|
|
self.close = Finalize(
|
|
|
|
self, PipeListener._finalize_pipe_listener,
|
|
|
|
args=(self._handle_queue, self._address), exitpriority=0
|
|
|
|
)
|
|
|
|
|
|
|
|
def accept(self):
|
|
|
|
newhandle = win32.CreateNamedPipe(
|
|
|
|
self._address, win32.PIPE_ACCESS_DUPLEX,
|
|
|
|
win32.PIPE_TYPE_MESSAGE | win32.PIPE_READMODE_MESSAGE |
|
|
|
|
win32.PIPE_WAIT,
|
|
|
|
win32.PIPE_UNLIMITED_INSTANCES, BUFSIZE, BUFSIZE,
|
|
|
|
win32.NMPWAIT_WAIT_FOREVER, win32.NULL
|
|
|
|
)
|
|
|
|
self._handle_queue.append(newhandle)
|
|
|
|
handle = self._handle_queue.pop(0)
|
|
|
|
try:
|
|
|
|
win32.ConnectNamedPipe(handle, win32.NULL)
|
|
|
|
except WindowsError as e:
|
2012-05-05 16:41:23 -03:00
|
|
|
# ERROR_NO_DATA can occur if a client has already connected,
|
|
|
|
# written data and then disconnected -- see Issue 14725.
|
|
|
|
if e.args[0] not in (win32.ERROR_PIPE_CONNECTED,
|
|
|
|
win32.ERROR_NO_DATA):
|
2008-06-11 13:44:04 -03:00
|
|
|
raise
|
|
|
|
return _multiprocessing.PipeConnection(handle)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _finalize_pipe_listener(queue, address):
|
|
|
|
sub_debug('closing listener with address=%r', address)
|
|
|
|
for handle in queue:
|
|
|
|
close(handle)
|
|
|
|
|
|
|
|
def PipeClient(address):
|
|
|
|
'''
|
|
|
|
Return a connection object connected to the pipe given by `address`
|
|
|
|
'''
|
2009-11-13 18:35:18 -04:00
|
|
|
t = _init_timeout()
|
2008-06-11 13:44:04 -03:00
|
|
|
while 1:
|
|
|
|
try:
|
|
|
|
win32.WaitNamedPipe(address, 1000)
|
|
|
|
h = win32.CreateFile(
|
|
|
|
address, win32.GENERIC_READ | win32.GENERIC_WRITE,
|
|
|
|
0, win32.NULL, win32.OPEN_EXISTING, 0, win32.NULL
|
|
|
|
)
|
|
|
|
except WindowsError as e:
|
|
|
|
if e.args[0] not in (win32.ERROR_SEM_TIMEOUT,
|
2009-11-13 18:35:18 -04:00
|
|
|
win32.ERROR_PIPE_BUSY) or _check_timeout(t):
|
2008-06-11 13:44:04 -03:00
|
|
|
raise
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
raise
|
|
|
|
|
|
|
|
win32.SetNamedPipeHandleState(
|
|
|
|
h, win32.PIPE_READMODE_MESSAGE, None, None
|
|
|
|
)
|
|
|
|
return _multiprocessing.PipeConnection(h)
|
|
|
|
|
|
|
|
#
|
|
|
|
# Authentication stuff
|
|
|
|
#
|
|
|
|
|
|
|
|
MESSAGE_LENGTH = 20
|
|
|
|
|
2008-06-25 09:54:22 -03:00
|
|
|
CHALLENGE = b'#CHALLENGE#'
|
|
|
|
WELCOME = b'#WELCOME#'
|
|
|
|
FAILURE = b'#FAILURE#'
|
2008-06-11 13:44:04 -03:00
|
|
|
|
|
|
|
def deliver_challenge(connection, authkey):
|
|
|
|
import hmac
|
|
|
|
assert isinstance(authkey, bytes)
|
|
|
|
message = os.urandom(MESSAGE_LENGTH)
|
|
|
|
connection.send_bytes(CHALLENGE + message)
|
|
|
|
digest = hmac.new(authkey, message).digest()
|
|
|
|
response = connection.recv_bytes(256) # reject large message
|
|
|
|
if response == digest:
|
|
|
|
connection.send_bytes(WELCOME)
|
|
|
|
else:
|
|
|
|
connection.send_bytes(FAILURE)
|
|
|
|
raise AuthenticationError('digest received was wrong')
|
|
|
|
|
|
|
|
def answer_challenge(connection, authkey):
|
|
|
|
import hmac
|
|
|
|
assert isinstance(authkey, bytes)
|
|
|
|
message = connection.recv_bytes(256) # reject large message
|
|
|
|
assert message[:len(CHALLENGE)] == CHALLENGE, 'message = %r' % message
|
|
|
|
message = message[len(CHALLENGE):]
|
|
|
|
digest = hmac.new(authkey, message).digest()
|
|
|
|
connection.send_bytes(digest)
|
|
|
|
response = connection.recv_bytes(256) # reject large message
|
|
|
|
if response != WELCOME:
|
|
|
|
raise AuthenticationError('digest sent was rejected')
|
|
|
|
|
|
|
|
#
|
|
|
|
# Support for using xmlrpclib for serialization
|
|
|
|
#
|
|
|
|
|
|
|
|
class ConnectionWrapper(object):
|
|
|
|
def __init__(self, conn, dumps, loads):
|
|
|
|
self._conn = conn
|
|
|
|
self._dumps = dumps
|
|
|
|
self._loads = loads
|
|
|
|
for attr in ('fileno', 'close', 'poll', 'recv_bytes', 'send_bytes'):
|
|
|
|
obj = getattr(conn, attr)
|
|
|
|
setattr(self, attr, obj)
|
|
|
|
def send(self, obj):
|
|
|
|
s = self._dumps(obj)
|
|
|
|
self._conn.send_bytes(s)
|
|
|
|
def recv(self):
|
|
|
|
s = self._conn.recv_bytes()
|
|
|
|
return self._loads(s)
|
|
|
|
|
|
|
|
def _xml_dumps(obj):
|
|
|
|
return xmlrpclib.dumps((obj,), None, None, None, 1).encode('utf8')
|
|
|
|
|
|
|
|
def _xml_loads(s):
|
|
|
|
(obj,), method = xmlrpclib.loads(s.decode('utf8'))
|
|
|
|
return obj
|
|
|
|
|
|
|
|
class XmlListener(Listener):
|
|
|
|
def accept(self):
|
|
|
|
global xmlrpclib
|
|
|
|
import xmlrpc.client as xmlrpclib
|
|
|
|
obj = Listener.accept(self)
|
|
|
|
return ConnectionWrapper(obj, _xml_dumps, _xml_loads)
|
|
|
|
|
|
|
|
def XmlClient(*args, **kwds):
|
|
|
|
global xmlrpclib
|
|
|
|
import xmlrpc.client as xmlrpclib
|
|
|
|
return ConnectionWrapper(Client(*args, **kwds), _xml_dumps, _xml_loads)
|