]> git.saurik.com Git - wxWidgets.git/blame - wxPython/samples/ide/activegrid/tool/process.py
removed code inside USE_SIZABLE_CALENDAR, we should allow making the main calendar...
[wxWidgets.git] / wxPython / samples / ide / activegrid / tool / process.py
CommitLineData
1f780e48
RD
1#!/usr/bin/env python
2# Copyright (c) 2002-2003 ActiveState
3# See LICENSE.txt for license details.
4""" Contents of LICENSE.txt:
5Permission is hereby granted, free of charge, to any person obtaining a
6copy of this software and associated documentation files (the
7"Software"), to deal in the Software without restriction, including
8without limitation the rights to use, copy, modify, merge, publish,
9distribute, sublicense, and/or sell copies of the Software, and to
10permit persons to whom the Software is furnished to do so, subject to
11the following conditions:
12
13The above copyright notice and this permission notice shall be included
14in all copies or substantial portions of the Software.
15
16THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
17OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
19IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
20CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
21TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
22SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23"""
24
25r"""
26 Python interface for process control.
27
28 This module defines three Process classes for spawning,
29 communicating and control processes. They are: Process, ProcessOpen,
30 ProcessProxy. All of the classes allow one to specify the command (cmd),
31 starting working directory (cwd), and environment to create for the
32 new process (env) and to "wait" for termination of the child and
33 "kill" the child.
34
35 Process:
36 Use this class to simply launch a process (either a GUI app or a
37 console app in a new console) with which you do not intend to
38 communicate via it std handles.
39
40 ProcessOpen:
41 Think of this as a super version of Python's os.popen3() method.
42 This spawns the given command and sets up pipes for
43 stdin/stdout/stderr which can then be used to communicate with
44 the child.
45
46 ProcessProxy:
47 This is a heavy-weight class that, similar to ProcessOpen,
48 spawns the given commands and sets up pipes to the child's
49 stdin/stdout/stderr. However, it also starts three threads to
50 proxy communication between each of the child's and parent's std
51 handles. At the parent end of this communication are, by
52 default, IOBuffer objects. You may specify your own objects here
53 (usually sub-classing from IOBuffer, which handles some
54 synchronization issues for you). The result is that it is
55 possible to have your own IOBuffer instance that gets, say, a
56 .write() "event" for every write that the child does on its
57 stdout.
58
59 Understanding ProcessProxy is pretty complex. Some examples
60 below attempt to help show some uses. Here is a diagram of the
61 comminucation:
62
63 <parent process>
64 ,---->->->------' ^ `------>->->----,
65 | | v
66 IOBuffer IOBuffer IOBuffer
67 (p.stdout) (p.stderr) (p.stdin)
68 | | |
69 _OutFileProxy _OutFileProxy _InFileProxy
70 thread thread thread
71 | ^ |
72 `----<-<-<------, | ,------<-<-<----'
73 <child process>
74
75 Usage:
76 import process
77 p = process.<Process class>(cmd='echo hi', ...)
78 #... use the various methods and attributes
79
80 Examples:
81 A simple 'hello world':
82 >>> import process
83 >>> p = process.ProcessOpen(['echo', 'hello'])
84 >>> p.stdout.read()
85 'hello\r\n'
86 >>> p.wait() # .wait() returns the child's exit status
87 0
88
89 Redirecting the stdout handler:
90 >>> import sys
91 >>> p = process.ProcessProxy(['echo', 'hello'], stdout=sys.stdout)
92 hello
93
94 Using stdin (need to use ProcessProxy here because it defaults to
95 text-mode translation on Windows, ProcessOpen does not support
96 this):
97 >>> p = process.ProcessProxy(['sort'])
98 >>> p.stdin.write('5\n')
99 >>> p.stdin.write('2\n')
100 >>> p.stdin.write('7\n')
101 >>> p.stdin.close()
102 >>> p.stdout.read()
103 '2\n5\n7\n'
104
105 Specifying environment variables:
106 >>> p = process.ProcessOpen(['perl', '-e', 'print $ENV{FOO}'])
107 >>> p.stdout.read()
108 ''
109 >>> p = process.ProcessOpen(['perl', '-e', 'print $ENV{FOO}'],
110 ... env={'FOO':'bar'})
111 >>> p.stdout.read()
112 'bar'
113
114 Killing a long running process (On Linux, to poll you must use
115 p.wait(os.WNOHANG)):
116 >>> p = ProcessOpen(['perl', '-e', 'while (1) {}'])
117 >>> try:
118 ... p.wait(os.WNOHANG) # poll to see if is process still running
119 ... except ProcessError, ex:
120 ... if ex.errno == ProcessProxy.WAIT_TIMEOUT:
121 ... print "process is still running"
122 ...
123 process is still running
124 >>> p.kill(42)
125 >>> p.wait()
126 42
127
128 Providing objects for stdin/stdout/stderr:
129 XXX write this, mention IOBuffer subclassing.
130"""
131#TODO:
132# - Discuss the decision to NOT have the stdout/stderr _OutFileProxy's
133# wait for process termination before closing stdin. It will just
134# close stdin when stdout is seen to have been closed. That is
135# considered Good Enough (tm). Theoretically it would be nice to
136# only abort the stdin proxying when the process terminates, but
137# watching for process termination in any of the parent's thread
138# adds the undesired condition that the parent cannot exit with the
139# child still running. That sucks.
140# XXX Note that I don't even know if the current stdout proxy even
141# closes the stdin proxy at all.
142# - DavidA: if I specify "unbuffered" for my stdin handler (in the
143# ProcessProxy constructor) then the stdin IOBuffer should do a
144# fparent.read() rather than a fparent.readline(). TrentM: can I do
145# that? What happens?
146#
147
148import os
149import sys
150import threading
151import types
152import pprint
153if sys.platform.startswith("win"):
154 import msvcrt
155 import win32api
156 import win32file
157 import win32pipe
158 import pywintypes
159 import win32process
160 import win32event
161 # constants pulled from win32con to save memory
162 VER_PLATFORM_WIN32_WINDOWS = 1
163 CTRL_BREAK_EVENT = 1
164 SW_SHOWDEFAULT = 10
165 WM_CLOSE = 0x10
166 DUPLICATE_SAME_ACCESS = 2
167
168else:
169 import signal
170
171
172#---- exceptions
173
174class ProcessError(Exception):
175 def __init__(self, msg, errno=-1):
176 Exception.__init__(self, msg)
177 self.errno = errno
178
179
180#---- internal logging facility
181
182class Logger:
183 DEBUG, INFO, WARN, ERROR, FATAL = range(5)
184 def __init__(self, name, level=None, streamOrFileName=sys.stderr):
185 self.name = name
186 if level is None:
187 self.level = self.WARN
188 else:
189 self.level = level
190 if type(streamOrFileName) == types.StringType:
191 self.stream = open(streamOrFileName, 'w')
192 self._opennedStream = 1
193 else:
194 self.stream = streamOrFileName
195 self._opennedStream = 0
196 def __del__(self):
197 if self._opennedStream:
198 self.stream.close()
199 def _getLevelName(self, level):
200 levelNameMap = {
201 self.DEBUG: "DEBUG",
202 self.INFO: "INFO",
203 self.WARN: "WARN",
204 self.ERROR: "ERROR",
205 self.FATAL: "FATAL",
206 }
207 return levelNameMap[level]
208 def log(self, level, msg, *args):
209 if level < self.level:
210 return
211 message = "%s: %s:" % (self.name, self._getLevelName(level).lower())
212 message = message + (msg % args) + "\n"
213 self.stream.write(message)
214 self.stream.flush()
215 def debug(self, msg, *args):
216 self.log(self.DEBUG, msg, *args)
217 def info(self, msg, *args):
218 self.log(self.INFO, msg, *args)
219 def warn(self, msg, *args):
220 self.log(self.WARN, msg, *args)
221 def error(self, msg, *args):
222 self.log(self.ERROR, msg, *args)
223 def fatal(self, msg, *args):
224 self.log(self.FATAL, msg, *args)
225
226# Loggers:
227# - 'log' to log normal process handling
228# - 'logres' to track system resource life
229# - 'logfix' to track wait/kill proxying in _ThreadFixer
230if 1: # normal/production usage
231 log = Logger("process", Logger.WARN)
232else: # development/debugging usage
233 log = Logger("process", Logger.DEBUG, sys.stdout)
234if 1: # normal/production usage
235 logres = Logger("process.res", Logger.WARN)
236else: # development/debugging usage
237 logres = Logger("process.res", Logger.DEBUG, sys.stdout)
238if 1: # normal/production usage
239 logfix = Logger("process.waitfix", Logger.WARN)
240else: # development/debugging usage
241 logfix = Logger("process.waitfix", Logger.DEBUG, sys.stdout)
242
243
244
245#---- globals
246
247_version_ = (0, 5, 0)
248
249# List of registered processes (see _(un)registerProcess).
250_processes = []
251
252
253
254#---- internal support routines
255
256def _escapeArg(arg):
257 """Escape the given command line argument for the shell."""
258 #XXX There is a probably more that we should escape here.
259 return arg.replace('"', r'\"')
260
261
262def _joinArgv(argv):
263 r"""Join an arglist to a string appropriate for running.
264
265 >>> import os
266 >>> _joinArgv(['foo', 'bar "baz'])
267 'foo "bar \\"baz"'
268 """
269 cmdstr = ""
270 for arg in argv:
271 if ' ' in arg or ';' in arg:
272 cmdstr += '"%s"' % _escapeArg(arg)
273 else:
274 cmdstr += _escapeArg(arg)
275 cmdstr += ' '
276 if cmdstr.endswith(' '): cmdstr = cmdstr[:-1] # strip trailing space
277 return cmdstr
278
279
280def _getPathFromEnv(env):
281 """Return the PATH environment variable or None.
282
283 Do the right thing for case sensitivity per platform.
284 XXX Icky. This guarantee of proper case sensitivity of environment
285 variables should be done more fundamentally in this module.
286 """
287 if sys.platform.startswith("win"):
288 for key in env.keys():
289 if key.upper() == "PATH":
290 return env[key]
291 else:
292 return None
293 else:
294 if env.has_key("PATH"):
295 return env["PATH"]
296 else:
297 return None
298
299
300def _whichFirstArg(cmd, env=None):
301 """Return the given command ensuring that the first arg (the command to
302 launch) is a full path to an existing file.
303
304 Raise a ProcessError if no such executable could be found.
305 """
306 # Parse out the first arg.
307 if cmd.startswith('"'):
308 # The .replace() is to ensure it does not mistakenly find the
309 # second '"' in, say (escaped quote):
310 # "C:\foo\"bar" arg1 arg2
311 idx = cmd.replace('\\"', 'XX').find('"', 1)
312 if idx == -1:
313 raise ProcessError("Malformed command: %r" % cmd)
314 first, rest = cmd[1:idx], cmd[idx+1:]
315 rest = rest.lstrip()
316 else:
317 if ' ' in cmd:
318 first, rest = cmd.split(' ', 1)
319 else:
320 first, rest = cmd, ""
321
322 # Ensure the first arg is a valid path to the appropriate file.
323 import which
324 if os.sep in first:
325 altpath = [os.path.dirname(first)]
326 firstbase = os.path.basename(first)
327 candidates = list(which.which(firstbase, path=altpath))
328 elif env:
329 altpath = _getPathFromEnv(env)
330 if altpath:
331 candidates = list(which.which(first, altpath.split(os.pathsep)))
332 else:
333 candidates = list(which.which(first))
334 else:
335 candidates = list(which.which(first))
336 if candidates:
337 return _joinArgv( [candidates[0]] ) + ' ' + rest
338 else:
339 raise ProcessError("Could not find an appropriate leading command "\
340 "for: %r" % cmd)
341
342
343if sys.platform.startswith("win"):
344 def _SaferCreateProcess(appName, # app name
345 cmd, # command line
346 processSA, # process security attributes
347 threadSA, # thread security attributes
348 inheritHandles, # are handles are inherited
349 creationFlags, # creation flags
350 env, # environment
351 cwd, # current working directory
352 si): # STARTUPINFO pointer
353 """If CreateProcess fails from environment type inconsistency then
354 fix that and try again.
355
356 win32process.CreateProcess requires that all environment keys and
357 values either be all ASCII or all unicode. Try to remove this burden
358 from the user of process.py.
359 """
360 isWin9x = win32api.GetVersionEx()[3] == VER_PLATFORM_WIN32_WINDOWS
361 # On Win9x all keys and values of 'env' must be ASCII (XXX
362 # Actually this is probably only true if the Unicode support
363 # libraries, which are not installed by default, are not
364 # installed). On other Windows flavours all keys and values of
365 # 'env' must all be ASCII *or* all Unicode. We will try to
366 # automatically convert to the appropriate type, issuing a
367 # warning if such an automatic conversion is necessary.
368
369 #XXX Komodo 2.0 Beta 1 hack. This requirement should be
370 # pushed out to Komodo code using process.py. Or should it?
371 if isWin9x and env:
372 aenv = {}
373 for key, value in env.items():
374 aenv[str(key)] = str(value)
375 env = aenv
376
377 log.debug("""\
378_SaferCreateProcess(appName=%r,
379 cmd=%r,
380 env=%r,
381 cwd=%r)
382 os.getcwd(): %r
383""", appName, cmd, env, cwd, os.getcwd())
384 try:
385 hProcess, hThread, processId, threadId\
386 = win32process.CreateProcess(appName, cmd, processSA,
387 threadSA, inheritHandles,
388 creationFlags, env, cwd, si)
389 except TypeError, ex:
390 if ex.args == ('All dictionary items must be strings, or all must be unicode',):
391 # Try again with an all unicode environment.
392 #XXX Would be nice if didn't have to depend on the error
393 # string to catch this.
394 #XXX Removing this warning for 2.3 release. See bug
395 # 23215. The right fix is to correct the PHPAppInfo
396 # stuff to heed the warning.
397 #import warnings
398 #warnings.warn('env: ' + str(ex), stacklevel=4)
399 if isWin9x and env:
400 aenv = {}
401 try:
402 for key, value in env.items():
403 aenv[str(key)] = str(value)
404 except UnicodeError, ex:
405 raise ProcessError(str(ex))
406 env = aenv
407 elif env:
408 uenv = {}
409 for key, val in env.items():
aca310e5
RD
410 try:
411 uenv[unicode(key)] = unicode(val) # default encoding
412 except UnicodeError:
413 try:
414 uenv[unicode(key, 'iso-8859-1')] = unicode(val, 'iso-8859-1') # backup encoding
415 except UnicodeError:
416 log.warn('Skipping environment variable "%s" in execution process: unable to convert to unicode using either the default encoding or ISO-8859-1' % (key))
1f780e48
RD
417 env = uenv
418 hProcess, hThread, processId, threadId\
419 = win32process.CreateProcess(appName, cmd, processSA,
420 threadSA, inheritHandles,
421 creationFlags, env, cwd,
422 si)
423 else:
424 raise
425 return hProcess, hThread, processId, threadId
426
427
428# Maintain references to all spawned ProcessProxy objects to avoid hangs.
429# Otherwise, if the user lets the a ProcessProxy object go out of
430# scope before the process has terminated, it is possible to get a
431# hang (at least it *used* to be so when we had the
432# win32api.CloseHandle(<stdin handle>) call in the __del__() method).
433# XXX Is this hang possible on Linux as well?
434# A reference is removed from this list when the process's .wait or
435# .kill method is called.
436# XXX Should an atexit() handler be registered to kill all curently
437# running processes? Else *could* get hangs, n'est ce pas?
438def _registerProcess(process):
439 global _processes
440 log.info("_registerprocess(process=%r)", process)
441
442 # Clean up zombie processes.
443 # If the user does not call .wait() or .kill() on processes then
444 # the ProcessProxy object will not get cleaned up until Python
445 # exits and _processes goes out of scope. Under heavy usage that
446 # is a big memory waste. Cleaning up here alleviates that.
447 for p in _processes[:]: # use copy of _process, because we may modifiy it
448 try:
449 # poll to see if is process still running
450 if sys.platform.startswith("win"):
451 timeout = 0
452 else:
453 timeout = os.WNOHANG
454 p.wait(timeout)
455 _unregisterProcess(p)
456 except ProcessError, ex:
457 if ex.errno == ProcessProxy.WAIT_TIMEOUT:
458 pass
459 else:
460 raise
461
462 _processes.append(process)
463
464def _unregisterProcess(process):
465 global _processes
466 log.info("_unregisterProcess(process=%r)", process)
467 try:
468 _processes.remove(process)
469 del process
470 except ValueError:
471 pass
472
473
474def _fixupCommand(cmd, env=None):
475 """Fixup the command string so it is launchable via CreateProcess.
476
477 One cannot just launch, say "python", via CreateProcess. A full path
478 to an executable is required. In general there are two choices:
479 1. Launch the command string via the shell. The shell will find
480 the fullpath to the appropriate executable. This shell will
481 also be able to execute special shell commands, like "dir",
482 which don't map to an actual executable.
483 2. Find the fullpath to the appropriate executable manually and
484 launch that exe.
485
486 Option (1) is preferred because you don't have to worry about not
487 exactly duplicating shell behaviour and you get the added bonus of
488 being able to launch "dir" and friends.
489
490 However, (1) is not always an option. Doing so when the shell is
491 command.com (as on all Win9x boxes) or when using WinNT's cmd.exe,
492 problems are created with .kill() because these shells seem to eat
493 up Ctrl-C's and Ctrl-Break's sent via
494 win32api.GenerateConsoleCtrlEvent(). Strangely this only happens
495 when spawn via this Python interface. For example, Ctrl-C get
496 through to hang.exe here:
497 C:\> ...\w9xpopen.exe "C:\WINDOWS\COMMAND.COM /c hang.exe"
498 ^C
499 but not here:
500 >>> p = ProcessOpen('hang.exe')
501 # This results in the same command to CreateProcess as
502 # above.
503 >>> p.kill()
504
505 Hence, for these platforms we fallback to option (2). Cons:
506 - cannot spawn shell commands like 'dir' directly
507 - cannot spawn batch files
508 """
509 if sys.platform.startswith("win"):
510 # Fixup the command string to spawn. (Lifted from
511 # posixmodule.c::_PyPopenCreateProcess() with some modifications)
512 comspec = os.environ.get("COMSPEC", None)
513 win32Version = win32api.GetVersion()
514 if comspec is None:
515 raise ProcessError("Cannot locate a COMSPEC environment "\
516 "variable to use as the shell")
517 # Explicitly check if we are using COMMAND.COM. If we
518 # are then use the w9xpopen hack.
519 elif (win32Version & 0x80000000L == 0) and\
520 (win32Version & 0x5L >= 5) and\
521 os.path.basename(comspec).lower() != "command.com":
522 # 2000/XP and not using command.com.
523 if '"' in cmd or "'" in cmd:
524 cmd = comspec + ' /c "%s"' % cmd
525 else:
526 cmd = comspec + ' /c ' + cmd
527 elif (win32Version & 0x80000000L == 0) and\
528 (win32Version & 0x5L < 5) and\
529 os.path.basename(comspec).lower() != "command.com":
530 # NT and not using command.com.
531 try:
532 cmd = _whichFirstArg(cmd, env)
533 except ProcessError:
534 raise ProcessError("Could not find a suitable executable "\
535 "to launch for '%s'. On WinNT you must manually prefix "\
536 "shell commands and batch files with 'cmd.exe /c' to "\
537 "have the shell run them." % cmd)
538 else:
539 # Oh gag, we're on Win9x and/or using COMMAND.COM. Use the
540 # workaround listed in KB: Q150956
541 w9xpopen = os.path.join(
542 os.path.dirname(win32api.GetModuleFileName(0)),
543 'w9xpopen.exe')
544 if not os.path.exists(w9xpopen):
545 # Eeek - file-not-found - possibly an embedding
546 # situation - see if we can locate it in sys.exec_prefix
547 w9xpopen = os.path.join(os.path.dirname(sys.exec_prefix),
548 'w9xpopen.exe')
549 if not os.path.exists(w9xpopen):
550 raise ProcessError(\
551 "Can not locate 'w9xpopen.exe' which is needed "\
552 "for ProcessOpen to work with your shell or "\
553 "platform.")
554 ## This would be option (1):
555 #cmd = '%s "%s /c %s"'\
556 # % (w9xpopen, comspec, cmd.replace('"', '\\"'))
557 try:
558 cmd = _whichFirstArg(cmd, env)
559 except ProcessError:
560 raise ProcessError("Could not find a suitable executable "\
561 "to launch for '%s'. On Win9x you must manually prefix "\
562 "shell commands and batch files with 'command.com /c' "\
563 "to have the shell run them." % cmd)
564 cmd = '%s "%s"' % (w9xpopen, cmd.replace('"', '\\"'))
565 return cmd
566
567class _FileWrapper:
568 """Wrap a system file object, hiding some nitpicky details.
569
570 This class provides a Python file-like interface to either a Python
571 file object (pretty easy job), a file descriptor, or an OS-specific
572 file handle (e.g. Win32 handles to file objects on Windows). Any or
573 all of these object types may be passed to this wrapper. If more
574 than one is specified this wrapper prefers to work with certain one
575 in this order:
576 - file descriptor (because usually this allows for
577 return-immediately-on-read-if-anything-available semantics and
578 also provides text mode translation on Windows)
579 - OS-specific handle (allows for the above read semantics)
580 - file object (buffering can cause difficulty for interacting
581 with spawned programs)
582
583 It also provides a place where related such objects can be kept
584 alive together to prevent premature ref-counted collection. (E.g. on
585 Windows a Python file object may be associated with a Win32 file
586 handle. If the file handle is not kept alive the Python file object
587 will cease to function.)
588 """
589 def __init__(self, file=None, descriptor=None, handle=None):
590 self._file = file
591 self._descriptor = descriptor
592 self._handle = handle
593 self._closed = 0
594 if self._descriptor is not None or self._handle is not None:
595 self._lineBuf = "" # to support .readline()
596
597 def __del__(self):
598 self.close()
599
600 def __getattr__(self, name):
601 """Forward to the underlying file object."""
602 if self._file is not None:
603 return getattr(self._file, name)
604 else:
605 raise ProcessError("no file object to pass '%s' attribute to"
606 % name)
607
608 def _win32Read(self, nBytes):
609 try:
610 log.info("[%s] _FileWrapper.read: waiting for read on pipe",
611 id(self))
612 errCode, text = win32file.ReadFile(self._handle, nBytes)
613 except pywintypes.error, ex:
614 # Ignore errors for now, like "The pipe is being closed.",
615 # etc. XXX There *may* be errors we don't want to avoid.
616 log.info("[%s] _FileWrapper.read: error reading from pipe: %s",
617 id(self), ex)
618 return ""
619 assert errCode == 0,\
620 "Why is 'errCode' from ReadFile non-zero? %r" % errCode
621 if not text:
622 # Empty text signifies that the pipe has been closed on
623 # the parent's end.
624 log.info("[%s] _FileWrapper.read: observed close of parent",
625 id(self))
626 # Signal the child so it knows to stop listening.
627 self.close()
628 return ""
629 else:
630 log.info("[%s] _FileWrapper.read: read %d bytes from pipe: %r",
631 id(self), len(text), text)
632 return text
633
634 def read(self, nBytes=-1):
635 # nBytes <= 0 means "read everything"
636 # Note that we are changing the "read everything" cue to
637 # include 0, because actually doing
638 # win32file.ReadFile(<handle>, 0) results in every subsequent
639 # read returning 0, i.e. it shuts down the pipe.
640 if self._descriptor is not None:
641 if nBytes <= 0:
642 text, self._lineBuf = self._lineBuf, ""
643 while 1:
644 t = os.read(self._descriptor, 4092)
645 if not t:
646 break
647 else:
648 text += t
649 else:
650 if len(self._lineBuf) >= nBytes:
651 text, self._lineBuf =\
652 self._lineBuf[:nBytes], self._lineBuf[nBytes:]
653 else:
654 nBytesToGo = nBytes - len(self._lineBuf)
655 text = self._lineBuf + os.read(self._descriptor,
656 nBytesToGo)
657 self._lineBuf = ""
658 return text
659 elif self._handle is not None:
660 if nBytes <= 0:
661 text, self._lineBuf = self._lineBuf, ""
662 while 1:
663 t = self._win32Read(4092)
664 if not t:
665 break
666 else:
667 text += t
668 else:
669 if len(self._lineBuf) >= nBytes:
670 text, self._lineBuf =\
671 self._lineBuf[:nBytes], self._lineBuf[nBytes:]
672 else:
673 nBytesToGo = nBytes - len(self._lineBuf)
674 text, self._lineBuf =\
675 self._lineBuf + self._win32Read(nBytesToGo), ""
676 return text
677 elif self._file is not None:
678 return self._file.read(nBytes)
679 else:
680 raise "FileHandle.read: no handle to read with"
681
682 def readline(self):
683 if self._descriptor is not None or self._handle is not None:
684 while 1:
685 #XXX This is not portable to the Mac.
686 idx = self._lineBuf.find('\n')
687 if idx != -1:
688 line, self._lineBuf =\
689 self._lineBuf[:idx+1], self._lineBuf[idx+1:]
690 break
691 else:
692 lengthBefore = len(self._lineBuf)
693 t = self.read(4092)
694 if len(t) <= lengthBefore: # no new data was read
695 line, self._lineBuf = self._lineBuf, ""
696 break
697 else:
698 self._lineBuf += t
699 return line
700 elif self._file is not None:
701 return self._file.readline()
702 else:
703 raise "FileHandle.readline: no handle to read with"
704
705 def readlines(self):
706 if self._descriptor is not None or self._handle is not None:
707 lines = []
708 while 1:
709 line = self.readline()
710 if line:
711 lines.append(line)
712 else:
713 break
714 return lines
715 elif self._file is not None:
716 return self._file.readlines()
717 else:
718 raise "FileHandle.readline: no handle to read with"
719
720 def write(self, text):
721 if self._descriptor is not None:
722 os.write(self._descriptor, text)
723 elif self._handle is not None:
724 try:
725 errCode, nBytesWritten = win32file.WriteFile(self._handle, text)
726 except pywintypes.error, ex:
727 # Ingore errors like "The pipe is being closed.", for
728 # now.
729 log.info("[%s] _FileWrapper.write: error writing to pipe, "\
730 "ignored", id(self))
731 return
732 assert errCode == 0,\
733 "Why is 'errCode' from WriteFile non-zero? %r" % errCode
734 if not nBytesWritten:
735 # No bytes written signifies that the pipe has been
736 # closed on the child's end.
737 log.info("[%s] _FileWrapper.write: observed close of pipe",
738 id(self))
739 return
740 else:
741 log.info("[%s] _FileWrapper.write: wrote %d bytes to pipe: %r",
742 id(self), len(text), text)
743 elif self._file is not None:
744 self._file.write(text)
745 else:
746 raise "FileHandle.write: nothing to write with"
747
748 def close(self):
749 """Close all associated file objects and handles."""
750 log.debug("[%s] _FileWrapper.close()", id(self))
751 if not self._closed:
752 self._closed = 1
753 if self._file is not None:
754 log.debug("[%s] _FileWrapper.close: close file", id(self))
755 self._file.close()
756 log.debug("[%s] _FileWrapper.close: done file close", id(self))
757 if self._descriptor is not None:
758 try:
759 os.close(self._descriptor)
760 except OSError, ex:
761 if ex.errno == 9:
762 # Ignore: OSError: [Errno 9] Bad file descriptor
763 # XXX *Should* we be ignoring this? It appears very
764 # *in*frequently in test_wait.py.
765 log.debug("[%s] _FileWrapper.close: closing "\
766 "descriptor raised OSError", id(self))
767 else:
768 raise
769 if self._handle is not None:
770 log.debug("[%s] _FileWrapper.close: close handle", id(self))
771 try:
772 win32api.CloseHandle(self._handle)
773 except win32api.error:
774 log.debug("[%s] _FileWrapper.close: closing handle raised",
775 id(self))
776 pass
777 log.debug("[%s] _FileWrapper.close: done closing handle",
778 id(self))
779
780 def __repr__(self):
781 return "<_FileWrapper: file:%r fd:%r os_handle:%r>"\
782 % (self._file, self._descriptor, self._handle)
783
784
785class _CountingCloser:
786 """Call .close() on the given object after own .close() is called
787 the precribed number of times.
788 """
789 def __init__(self, objectsToClose, count):
790 """
791 "objectsToClose" is a list of object on which to call .close().
792 "count" is the number of times this object's .close() method
793 must be called before .close() is called on the given objects.
794 """
795 self.objectsToClose = objectsToClose
796 self.count = count
797 if self.count <= 0:
798 raise ProcessError("illegal 'count' value: %s" % self.count)
799
800 def close(self):
801 self.count -= 1
802 log.debug("[%d] _CountingCloser.close(): count=%d", id(self),
803 self.count)
804 if self.count == 0:
805 for objectToClose in self.objectsToClose:
806 objectToClose.close()
807
808
809
810#---- public interface
811
812class Process:
813 """Create a process.
814
815 One can optionally specify the starting working directory, the
816 process environment, and std handles to have the child process
817 inherit (all defaults are the parent's current settings). 'wait' and
818 'kill' method allow for control of the child's termination.
819 """
820 # TODO:
821 # - Rename this or merge it with ProcessOpen somehow.
822 #
823 if sys.platform.startswith("win"):
824 # .wait() argument constants
825 INFINITE = win32event.INFINITE
826 # .wait() return error codes
827 WAIT_FAILED = win32event.WAIT_FAILED
828 WAIT_TIMEOUT = win32event.WAIT_TIMEOUT
829 # creation "flags" constants
830 # XXX Should drop these and just document usage of
831 # win32process.CREATE_* constants on windows.
832 CREATE_NEW_CONSOLE = win32process.CREATE_NEW_CONSOLE
833 else:
834 # .wait() argument constants
835 INFINITE = 0
836 # .wait() return error codes
837 WAIT_TIMEOUT = 258
838 WAIT_FAILED = -1
839 # creation "flags" constants
840 CREATE_NEW_CONSOLE = 0x10 # same as win32process.CREATE_NEW_CONSOLE
841
842 def __init__(self, cmd, cwd=None, env=None, flags=0):
843 """Create a child process.
844
845 "cmd" is a command string or argument vector to spawn.
846 "cwd" is a working directory in which to start the child process.
847 "env" is an environment dictionary for the child.
848 "flags" are system-specific process creation flags. On Windows
849 this can be a bitwise-OR of any of the win32process.CREATE_*
850 constants (Note: win32process.CREATE_NEW_PROCESS_GROUP is always
851 OR'd in). On Unix, this is currently ignored.
852 """
853 log.info("Process.__init__(cmd=%r, cwd=%r, env=%r, flags=%r)",
854 cmd, cwd, env, flags)
855 self._cmd = cmd
856 if not self._cmd:
857 raise ProcessError("You must specify a command.")
858 self._cwd = cwd
859 self._env = env
860 self._flags = flags
861 if sys.platform.startswith("win"):
862 self._flags |= win32process.CREATE_NEW_PROCESS_GROUP
863
864 if sys.platform.startswith("win"):
865 self._startOnWindows()
866 else:
867 self.__retvalCache = None
868 self._startOnUnix()
869
870 def _runChildOnUnix(self):
871 #XXX Errors running the child do *not* get communicated back.
872
873 #XXX Perhaps we should *always* prefix with '/bin/sh -c'? There is a
874 # disparity btwn how this works on Linux and Windows.
875 if isinstance(self._cmd, types.StringTypes):
876 # This is easier than trying to reproduce shell interpretation to
877 # separate the arguments.
878 cmd = ['/bin/sh', '-c', self._cmd]
879 else:
880 cmd = self._cmd
881
882 # Close all file descriptors (except std*) inherited from the parent.
883 MAXFD = 256 # Max number of file descriptors (os.getdtablesize()???)
884 for i in range(3, MAXFD):
885 try:
886 os.close(i)
887 except OSError:
888 pass
889
890 try:
891 if self._env:
892 os.execvpe(cmd[0], cmd, self._env)
893 else:
894 os.execvp(cmd[0], cmd)
895 finally:
896 os._exit(1) # Should never get here.
897
898 def _forkAndExecChildOnUnix(self):
899 """Fork and start the child process.
900
901 Sets self._pid as a side effect.
902 """
903 pid = os.fork()
904 if pid == 0: # child
905 self._runChildOnUnix()
906 # parent
907 self._pid = pid
908
909 def _startOnUnix(self):
910 if self._cwd:
911 oldDir = os.getcwd()
912 try:
913 os.chdir(self._cwd)
914 except OSError, ex:
915 raise ProcessError(msg=str(ex), errno=ex.errno)
916 self._forkAndExecChildOnUnix()
917
918 # parent
919 if self._cwd:
920 os.chdir(oldDir)
921
922 def _startOnWindows(self):
923 if type(self._cmd) in (types.ListType, types.TupleType):
924 # And arg vector was passed in.
925 cmd = _joinArgv(self._cmd)
926 else:
927 cmd = self._cmd
928
929 si = win32process.STARTUPINFO()
930 si.dwFlags = win32process.STARTF_USESHOWWINDOW
931 si.wShowWindow = SW_SHOWDEFAULT
932
933 if not (self._flags & self.CREATE_NEW_CONSOLE):
934 #XXX This is hacky.
935 # We cannot then use _fixupCommand because this will cause a
936 # shell to be openned as the command is launched. Therefore need
937 # to ensure be have the full path to the executable to launch.
938 try:
939 cmd = _whichFirstArg(cmd, self._env)
940 except ProcessError:
941 # Could not find the command, perhaps it is an internal
942 # shell command -- fallback to _fixupCommand
943 cmd = _fixupCommand(cmd, self._env)
944 else:
945 cmd = _fixupCommand(cmd, self._env)
946 log.debug("cmd = %r", cmd)
947
948 # Start the child process.
949 try:
950 self._hProcess, self._hThread, self._processId, self._threadId\
951 = _SaferCreateProcess(
952 None, # app name
953 cmd, # command line
954 None, # process security attributes
955 None, # primary thread security attributes
956 0, # handles are inherited
957 self._flags, # creation flags
958 self._env, # environment
959 self._cwd, # current working directory
960 si) # STARTUPINFO pointer
961 win32api.CloseHandle(self._hThread)
962 except win32api.error, ex:
963 raise ProcessError(msg="Error creating process for '%s': %s"\
964 % (cmd, ex.args[2]),
965 errno=ex.args[0])
966
967 def wait(self, timeout=None):
968 """Wait for the started process to complete.
969
970 "timeout" (on Windows) is a floating point number of seconds after
971 which to timeout. Default is win32event.INFINITE.
972 "timeout" (on Unix) is akin to the os.waitpid() "options" argument
973 (os.WNOHANG may be used to return immediately if the process has
974 not exited). Default is 0, i.e. wait forever.
975
976 If the wait time's out it will raise a ProcessError. Otherwise it
977 will return the child's exit value (on Windows) or the child's exit
978 status excoded as per os.waitpid() (on Linux):
979 "a 16-bit number, whose low byte is the signal number that killed
980 the process, and whose high byte is the exit status (if the
981 signal number is zero); the high bit of the low byte is set if a
982 core file was produced."
983 In the latter case, use the os.W*() methods to interpret the return
984 value.
985 """
986 # XXX Or should returning the exit value be move out to another
987 # function as on Win32 process control? If so, then should
988 # perhaps not make WaitForSingleObject semantic transformation.
989 if sys.platform.startswith("win"):
990 if timeout is None:
991 timeout = win32event.INFINITE
992 else:
993 timeout = timeout * 1000.0 # Win32 API's timeout is in millisecs
994
995 rc = win32event.WaitForSingleObject(self._hProcess, timeout)
996 if rc == win32event.WAIT_FAILED:
997 raise ProcessError("'WAIT_FAILED' when waiting for process to "\
998 "terminate: %r" % self._cmd, rc)
999 elif rc == win32event.WAIT_TIMEOUT:
1000 raise ProcessError("'WAIT_TIMEOUT' when waiting for process to "\
1001 "terminate: %r" % self._cmd, rc)
1002
1003 retval = win32process.GetExitCodeProcess(self._hProcess)
1004 else:
1005 # os.waitpid() will raise:
1006 # OSError: [Errno 10] No child processes
1007 # on subsequent .wait() calls. Change these semantics to have
1008 # subsequent .wait() calls return the exit status and return
1009 # immediately without raising an exception.
1010 # (XXX It would require synchronization code to handle the case
1011 # of multiple simultaneous .wait() requests, however we can punt
1012 # on that because it is moot while Linux still has the problem
1013 # for which _ThreadFixer() exists.)
1014 if self.__retvalCache is not None:
1015 retval = self.__retvalCache
1016 else:
1017 if timeout is None:
1018 timeout = 0
1019 pid, sts = os.waitpid(self._pid, timeout)
1020 if pid == self._pid:
1021 self.__retvalCache = retval = sts
1022 else:
1023 raise ProcessError("Wait for process timed out.",
1024 self.WAIT_TIMEOUT)
1025 return retval
1026
1027 def kill(self, exitCode=0, gracePeriod=1.0, sig=None):
1028 """Kill process.
1029
1030 "exitCode" [deprecated, not supported] (Windows only) is the
1031 code the terminated process should exit with.
1032 "gracePeriod" (Windows only) is a number of seconds the process is
1033 allowed to shutdown with a WM_CLOSE signal before a hard
1034 terminate is called.
1035 "sig" (Unix only) is the signal to use to kill the process. Defaults
1036 to signal.SIGKILL. See os.kill() for more information.
1037
1038 Windows:
1039 Try for an orderly shutdown via WM_CLOSE. If still running
1040 after gracePeriod (1 sec. default), terminate.
1041 """
1042 if sys.platform.startswith("win"):
1043 import win32gui
1044 # Send WM_CLOSE to windows in this process group.
1045 win32gui.EnumWindows(self._close_, 0)
1046
1047 # Send Ctrl-Break signal to all processes attached to this
1048 # console. This is supposed to trigger shutdown handlers in
1049 # each of the processes.
1050 try:
1051 win32api.GenerateConsoleCtrlEvent(CTRL_BREAK_EVENT,
1052 self._processId)
1053 except AttributeError:
1054 log.warn("The win32api module does not have "\
1055 "GenerateConsoleCtrlEvent(). This may mean that "\
1056 "parts of this process group have NOT been killed.")
1057 except win32api.error, ex:
1058 if ex.args[0] not in (6, 87):
1059 # Ignore the following:
1060 # api_error: (87, 'GenerateConsoleCtrlEvent', 'The parameter is incorrect.')
1061 # api_error: (6, 'GenerateConsoleCtrlEvent', 'The handle is invalid.')
1062 # Get error 6 if there is no console.
1063 raise
1064
1065 # Last resort: call TerminateProcess if it has not yet.
1066 retval = 0
1067 try:
1068 self.wait(gracePeriod)
1069 except ProcessError, ex:
1070 log.info("[%s] Process.kill: calling TerminateProcess", id(self))
1071 win32process.TerminateProcess(self._hProcess, -1)
1072 win32api.Sleep(100) # wait for resources to be released
1073
1074 else:
1075 if sig is None:
1076 sig = signal.SIGKILL
1077 try:
1078 os.kill(self._pid, sig)
1079 except OSError, ex:
1080 if ex.errno != 3:
1081 # Ignore: OSError: [Errno 3] No such process
1082 raise
1083
1084 def _close_(self, hwnd, dummy):
1085 """Callback used by .kill() on Windows.
1086
1087 EnumWindows callback - sends WM_CLOSE to any window owned by this
1088 process.
1089 """
1090 threadId, processId = win32process.GetWindowThreadProcessId(hwnd)
1091 if processId == self._processId:
1092 import win32gui
1093 win32gui.PostMessage(hwnd, WM_CLOSE, 0, 0)
1094
1095
1096class ProcessOpen(Process):
1097 """Create a process and setup pipes to it standard handles.
1098
1099 This is a super popen3.
1100 """
1101 # TODO:
1102 # - Share some implementation with Process and ProcessProxy.
1103 #
1104
1105 def __init__(self, cmd, mode='t', cwd=None, env=None):
1106 """Create a Process with proxy threads for each std handle.
1107
1108 "cmd" is the command string or argument vector to run.
1109 "mode" (Windows only) specifies whether the pipes used to communicate
1110 with the child are openned in text, 't', or binary, 'b', mode.
1111 This is ignored on platforms other than Windows. Default is 't'.
1112 "cwd" optionally specifies the directory in which the child process
1113 should be started. Default is None, a.k.a. inherits the cwd from
1114 the parent.
1115 "env" is optionally a mapping specifying the environment in which to
1116 start the child. Default is None, a.k.a. inherits the environment
1117 of the parent.
1118 """
1119 # Keep a reference to ensure it is around for this object's destruction.
1120 self.__log = log
1121 log.info("ProcessOpen.__init__(cmd=%r, mode=%r, cwd=%r, env=%r)",
1122 cmd, mode, cwd, env)
1123 self._cmd = cmd
1124 if not self._cmd:
1125 raise ProcessError("You must specify a command.")
1126 self._cwd = cwd
1127 self._env = env
1128 self._mode = mode
1129 if self._mode not in ('t', 'b'):
1130 raise ProcessError("'mode' must be 't' or 'b'.")
1131 self._closed = 0
1132
1133 if sys.platform.startswith("win"):
1134 self._startOnWindows()
1135 else:
1136 self.__retvalCache = None
1137 self._startOnUnix()
1138
1139 _registerProcess(self)
1140
1141 def __del__(self):
1142 #XXX Should probably not rely upon this.
1143 logres.info("[%s] ProcessOpen.__del__()", id(self))
1144 self.close()
1145 del self.__log # drop reference
1146
1147 def close(self):
1148 if not self._closed:
1149 self.__log.info("[%s] ProcessOpen.close()" % id(self))
1150
1151 # Ensure that all IOBuffer's are closed. If they are not, these
1152 # can cause hangs.
1153 try:
1154 self.__log.info("[%s] ProcessOpen: closing stdin (%r)."\
1155 % (id(self), self.stdin))
1156 self.stdin.close()
1157 except AttributeError:
1158 # May not have gotten far enough in the __init__ to set
1159 # self.stdin, etc.
1160 pass
1161 try:
1162 self.__log.info("[%s] ProcessOpen: closing stdout (%r)."\
1163 % (id(self), self.stdout))
1164 self.stdout.close()
1165 except AttributeError:
1166 # May not have gotten far enough in the __init__ to set
1167 # self.stdout, etc.
1168 pass
1169 try:
1170 self.__log.info("[%s] ProcessOpen: closing stderr (%r)."\
1171 % (id(self), self.stderr))
1172 self.stderr.close()
1173 except AttributeError:
1174 # May not have gotten far enough in the __init__ to set
1175 # self.stderr, etc.
1176 pass
1177
1178 self._closed = 1
1179
1180 def _forkAndExecChildOnUnix(self, fdChildStdinRd, fdChildStdoutWr,
1181 fdChildStderrWr):
1182 """Fork and start the child process.
1183
1184 Sets self._pid as a side effect.
1185 """
1186 pid = os.fork()
1187 if pid == 0: # child
1188 os.dup2(fdChildStdinRd, 0)
1189 os.dup2(fdChildStdoutWr, 1)
1190 os.dup2(fdChildStderrWr, 2)
1191 self._runChildOnUnix()
1192 # parent
1193 self._pid = pid
1194
1195 def _startOnUnix(self):
1196 # Create pipes for std handles.
1197 fdChildStdinRd, fdChildStdinWr = os.pipe()
1198 fdChildStdoutRd, fdChildStdoutWr = os.pipe()
1199 fdChildStderrRd, fdChildStderrWr = os.pipe()
1200
1201 if self._cwd:
1202 oldDir = os.getcwd()
1203 try:
1204 os.chdir(self._cwd)
1205 except OSError, ex:
1206 raise ProcessError(msg=str(ex), errno=ex.errno)
1207 self._forkAndExecChildOnUnix(fdChildStdinRd, fdChildStdoutWr,
1208 fdChildStderrWr)
1209 if self._cwd:
1210 os.chdir(oldDir)
1211
1212 os.close(fdChildStdinRd)
1213 os.close(fdChildStdoutWr)
1214 os.close(fdChildStderrWr)
1215
1216 self.stdin = _FileWrapper(descriptor=fdChildStdinWr)
1217 logres.info("[%s] ProcessOpen._start(): create child stdin: %r",
1218 id(self), self.stdin)
1219 self.stdout = _FileWrapper(descriptor=fdChildStdoutRd)
1220 logres.info("[%s] ProcessOpen._start(): create child stdout: %r",
1221 id(self), self.stdout)
1222 self.stderr = _FileWrapper(descriptor=fdChildStderrRd)
1223 logres.info("[%s] ProcessOpen._start(): create child stderr: %r",
1224 id(self), self.stderr)
1225
1226 def _startOnWindows(self):
1227 if type(self._cmd) in (types.ListType, types.TupleType):
1228 # An arg vector was passed in.
1229 cmd = _joinArgv(self._cmd)
1230 else:
1231 cmd = self._cmd
1232
1233 # Create pipes for std handles.
1234 # (Set the bInheritHandle flag so pipe handles are inherited.)
1235 saAttr = pywintypes.SECURITY_ATTRIBUTES()
1236 saAttr.bInheritHandle = 1
1237 #XXX Should maybe try with os.pipe. Dunno what that does for
1238 # inheritability though.
1239 hChildStdinRd, hChildStdinWr = win32pipe.CreatePipe(saAttr, 0)
1240 hChildStdoutRd, hChildStdoutWr = win32pipe.CreatePipe(saAttr, 0)
1241 hChildStderrRd, hChildStderrWr = win32pipe.CreatePipe(saAttr, 0)
1242
1243 try:
1244 # Duplicate the parent ends of the pipes so they are not
1245 # inherited.
1246 hChildStdinWrDup = win32api.DuplicateHandle(
1247 win32api.GetCurrentProcess(),
1248 hChildStdinWr,
1249 win32api.GetCurrentProcess(),
1250 0,
1251 0, # not inherited
1252 DUPLICATE_SAME_ACCESS)
1253 win32api.CloseHandle(hChildStdinWr)
1254 self._hChildStdinWr = hChildStdinWrDup
1255 hChildStdoutRdDup = win32api.DuplicateHandle(
1256 win32api.GetCurrentProcess(),
1257 hChildStdoutRd,
1258 win32api.GetCurrentProcess(),
1259 0,
1260 0, # not inherited
1261 DUPLICATE_SAME_ACCESS)
1262 win32api.CloseHandle(hChildStdoutRd)
1263 self._hChildStdoutRd = hChildStdoutRdDup
1264 hChildStderrRdDup = win32api.DuplicateHandle(
1265 win32api.GetCurrentProcess(),
1266 hChildStderrRd,
1267 win32api.GetCurrentProcess(),
1268 0,
1269 0, # not inherited
1270 DUPLICATE_SAME_ACCESS)
1271 win32api.CloseHandle(hChildStderrRd)
1272 self._hChildStderrRd = hChildStderrRdDup
1273
1274 # Set the translation mode and buffering.
1275 if self._mode == 't':
1276 flags = os.O_TEXT
1277 else:
1278 flags = 0
1279 fdChildStdinWr = msvcrt.open_osfhandle(self._hChildStdinWr, flags)
1280 fdChildStdoutRd = msvcrt.open_osfhandle(self._hChildStdoutRd, flags)
1281 fdChildStderrRd = msvcrt.open_osfhandle(self._hChildStderrRd, flags)
1282
1283 self.stdin = _FileWrapper(descriptor=fdChildStdinWr,
1284 handle=self._hChildStdinWr)
1285 logres.info("[%s] ProcessOpen._start(): create child stdin: %r",
1286 id(self), self.stdin)
1287 self.stdout = _FileWrapper(descriptor=fdChildStdoutRd,
1288 handle=self._hChildStdoutRd)
1289 logres.info("[%s] ProcessOpen._start(): create child stdout: %r",
1290 id(self), self.stdout)
1291 self.stderr = _FileWrapper(descriptor=fdChildStderrRd,
1292 handle=self._hChildStderrRd)
1293 logres.info("[%s] ProcessOpen._start(): create child stderr: %r",
1294 id(self), self.stderr)
1295
1296 # Start the child process.
1297 si = win32process.STARTUPINFO()
1298 si.dwFlags = win32process.STARTF_USESHOWWINDOW
1299 si.wShowWindow = 0 # SW_HIDE
1300 si.hStdInput = hChildStdinRd
1301 si.hStdOutput = hChildStdoutWr
1302 si.hStdError = hChildStderrWr
1303 si.dwFlags |= win32process.STARTF_USESTDHANDLES
1304
1305 cmd = _fixupCommand(cmd, self._env)
1306
1307 creationFlags = win32process.CREATE_NEW_PROCESS_GROUP
1308 try:
1309 self._hProcess, hThread, self._processId, threadId\
1310 = _SaferCreateProcess(
1311 None, # app name
1312 cmd, # command line
1313 None, # process security attributes
1314 None, # primary thread security attributes
1315 1, # handles are inherited
1316 creationFlags, # creation flags
1317 self._env, # environment
1318 self._cwd, # current working directory
1319 si) # STARTUPINFO pointer
1320 except win32api.error, ex:
1321 raise ProcessError(msg=ex.args[2], errno=ex.args[0])
1322 win32api.CloseHandle(hThread)
1323
1324 finally:
1325 # Close child ends of pipes on the parent's side (the
1326 # parent's ends of the pipe are closed in the _FileWrappers.)
1327 win32file.CloseHandle(hChildStdinRd)
1328 win32file.CloseHandle(hChildStdoutWr)
1329 win32file.CloseHandle(hChildStderrWr)
1330
1331 def wait(self, timeout=None):
1332 """Wait for the started process to complete.
1333
1334 "timeout" (on Windows) is a floating point number of seconds after
1335 which to timeout. Default is win32event.INFINITE.
1336 "timeout" (on Unix) is akin to the os.waitpid() "options" argument
1337 (os.WNOHANG may be used to return immediately if the process has
1338 not exited). Default is 0, i.e. wait forever.
1339
1340 If the wait time's out it will raise a ProcessError. Otherwise it
1341 will return the child's exit value (on Windows) or the child's exit
1342 status excoded as per os.waitpid() (on Linux):
1343 "a 16-bit number, whose low byte is the signal number that killed
1344 the process, and whose high byte is the exit status (if the
1345 signal number is zero); the high bit of the low byte is set if a
1346 core file was produced."
1347 In the latter case, use the os.W*() methods to interpret the return
1348 value.
1349 """
1350 # XXX Or should returning the exit value be move out to another
1351 # function as on Win32 process control? If so, then should
1352 # perhaps not make WaitForSingleObject semantic
1353 # transformation.
1354 # TODO:
1355 # - Need to rationalize the .wait() API for Windows vs. Unix.
1356 # It is a real pain in the current situation.
1357 if sys.platform.startswith("win"):
1358 if timeout is None:
1359 timeout = win32event.INFINITE
1360 else:
1361 timeout = timeout * 1000.0 # Win32 API's timeout is in millisecs
1362
1363 #rc = win32event.WaitForSingleObject(self._hProcess, timeout)
1364 rc = win32event.WaitForSingleObject(self._hProcess, int(timeout)) # MATT -- Making timeout an integer
1365 if rc == win32event.WAIT_FAILED:
1366 raise ProcessError("'WAIT_FAILED' when waiting for process to "\
1367 "terminate: %r" % self._cmd, rc)
1368 elif rc == win32event.WAIT_TIMEOUT:
1369 raise ProcessError("'WAIT_TIMEOUT' when waiting for process to "\
1370 "terminate: %r" % self._cmd, rc)
1371
1372 retval = win32process.GetExitCodeProcess(self._hProcess)
1373 else:
1374 # os.waitpid() will raise:
1375 # OSError: [Errno 10] No child processes
1376 # on subsequent .wait() calls. Change these semantics to have
1377 # subsequent .wait() calls return the exit status and return
1378 # immediately without raising an exception.
1379 # (XXX It would require synchronization code to handle the case
1380 # of multiple simultaneous .wait() requests, however we can punt
1381 # on that because it is moot while Linux still has the problem
1382 # for which _ThreadFixer() exists.)
1383 if self.__retvalCache is not None:
1384 retval = self.__retvalCache
1385 else:
1386 if timeout is None:
1387 timeout = 0
1388 pid, sts = os.waitpid(self._pid, timeout)
1389 if pid == self._pid:
1390 self.__retvalCache = retval = sts
1391 else:
1392 raise ProcessError("Wait for process timed out.",
1393 self.WAIT_TIMEOUT)
1394 _unregisterProcess(self)
1395 return retval
1396
1397 def kill(self, exitCode=0, gracePeriod=1.0, sig=None):
1398 """Kill process.
1399
1400 "exitCode" [deprecated, not supported] (Windows only) is the
1401 code the terminated process should exit with.
1402 "gracePeriod" (Windows only) is a number of seconds the process is
1403 allowed to shutdown with a WM_CLOSE signal before a hard
1404 terminate is called.
1405 "sig" (Unix only) is the signal to use to kill the process. Defaults
1406 to signal.SIGKILL. See os.kill() for more information.
1407
1408 Windows:
1409 Try for an orderly shutdown via WM_CLOSE. If still running
1410 after gracePeriod (1 sec. default), terminate.
1411 """
1412 if sys.platform.startswith("win"):
1413 import win32gui
1414 # Send WM_CLOSE to windows in this process group.
1415 win32gui.EnumWindows(self._close_, 0)
1416
1417 # Send Ctrl-Break signal to all processes attached to this
1418 # console. This is supposed to trigger shutdown handlers in
1419 # each of the processes.
1420 try:
1421 win32api.GenerateConsoleCtrlEvent(CTRL_BREAK_EVENT,
1422 self._processId)
1423 except AttributeError:
1424 log.warn("The win32api module does not have "\
1425 "GenerateConsoleCtrlEvent(). This may mean that "\
1426 "parts of this process group have NOT been killed.")
1427 except win32api.error, ex:
1428 if ex.args[0] not in (6, 87):
1429 # Ignore the following:
1430 # api_error: (87, 'GenerateConsoleCtrlEvent', 'The parameter is incorrect.')
1431 # api_error: (6, 'GenerateConsoleCtrlEvent', 'The handle is invalid.')
1432 # Get error 6 if there is no console.
1433 raise
1434
1435 # Last resort: call TerminateProcess if it has not yet.
1436 retval = 0
1437 try:
1438 self.wait(gracePeriod)
1439 except ProcessError, ex:
1440 log.info("[%s] Process.kill: calling TerminateProcess", id(self))
1441 win32process.TerminateProcess(self._hProcess, -1)
1442 win32api.Sleep(100) # wait for resources to be released
1443
1444 else:
1445 if sig is None:
1446 sig = signal.SIGKILL
1447 try:
1448 os.kill(self._pid, sig)
1449 except OSError, ex:
1450 if ex.errno != 3:
1451 # Ignore: OSError: [Errno 3] No such process
1452 raise
1453
1454 _unregisterProcess(self)
1455
1456 def _close_(self, hwnd, dummy):
1457 """Callback used by .kill() on Windows.
1458
1459 EnumWindows callback - sends WM_CLOSE to any window owned by this
1460 process.
1461 """
1462 threadId, processId = win32process.GetWindowThreadProcessId(hwnd)
1463 if processId == self._processId:
1464 import win32gui
1465 win32gui.PostMessage(hwnd, WM_CLOSE, 0, 0)
1466
1467
1468class ProcessProxy(Process):
1469 """Create a process and proxy communication via the standard handles.
1470 """
1471 #XXX To add to docstring:
1472 # - stdout/stderr proxy handling
1473 # - stdin proxy handling
1474 # - termination
1475 # - how to .start(), i.e. basic usage rules
1476 # - mention that pased in stdin/stdout/stderr objects have to
1477 # implement at least .write (is .write correct for stdin)?
1478 # - if you pass in stdin, stdout, and/or stderr streams it is the
1479 # user's responsibility to close them afterwards.
1480 # - 'cmd' arg can be a command string or an arg vector
1481 # - etc.
1482 #TODO:
1483 # - .suspend() and .resume()? See Win32::Process Perl module.
1484 #
1485 def __init__(self, cmd, mode='t', cwd=None, env=None,
1486 stdin=None, stdout=None, stderr=None):
1487 """Create a Process with proxy threads for each std handle.
1488
1489 "cmd" is the command string or argument vector to run.
1490 "mode" (Windows only) specifies whether the pipes used to communicate
1491 with the child are openned in text, 't', or binary, 'b', mode.
1492 This is ignored on platforms other than Windows. Default is 't'.
1493 "cwd" optionally specifies the directory in which the child process
1494 should be started. Default is None, a.k.a. inherits the cwd from
1495 the parent.
1496 "env" is optionally a mapping specifying the environment in which to
1497 start the child. Default is None, a.k.a. inherits the environment
1498 of the parent.
1499 "stdin", "stdout", "stderr" can be used to specify objects with
1500 file-like interfaces to handle read (stdout/stderr) and write
1501 (stdin) events from the child. By default a process.IOBuffer
1502 instance is assigned to each handler. IOBuffer may be
1503 sub-classed. See the IOBuffer doc string for more information.
1504 """
1505 # Keep a reference to ensure it is around for this object's destruction.
1506 self.__log = log
1507 log.info("ProcessProxy.__init__(cmd=%r, mode=%r, cwd=%r, env=%r, "\
1508 "stdin=%r, stdout=%r, stderr=%r)",
1509 cmd, mode, cwd, env, stdin, stdout, stderr)
1510 self._cmd = cmd
1511 if not self._cmd:
1512 raise ProcessError("You must specify a command.")
1513 self._mode = mode
1514 if self._mode not in ('t', 'b'):
1515 raise ProcessError("'mode' must be 't' or 'b'.")
1516 self._cwd = cwd
1517 self._env = env
1518 if stdin is None:
1519 self.stdin = IOBuffer(name='<stdin>')
1520 else:
1521 self.stdin = stdin
1522 if stdout is None:
1523 self.stdout = IOBuffer(name='<stdout>')
1524 else:
1525 self.stdout = stdout
1526 if stderr is None:
1527 self.stderr = IOBuffer(name='<stderr>')
1528 else:
1529 self.stderr = stderr
1530 self._closed = 0
1531
1532 if sys.platform.startswith("win"):
1533 self._startOnWindows()
1534 else:
1535 self.__retvalCache = None
1536 self._startOnUnix()
1537
1538 _registerProcess(self)
1539
1540 def __del__(self):
1541 #XXX Should probably not rely upon this.
1542 logres.info("[%s] ProcessProxy.__del__()", id(self))
1543 self.close()
1544 del self.__log # drop reference
1545
1546 def close(self):
1547 if not self._closed:
1548 self.__log.info("[%s] ProcessProxy.close()" % id(self))
1549
1550 # Ensure that all IOBuffer's are closed. If they are not, these
1551 # can cause hangs.
1552 self.__log.info("[%s] ProcessProxy: closing stdin (%r)."\
1553 % (id(self), self.stdin))
1554 try:
1555 self.stdin.close()
1556 self._stdinProxy.join()
1557 except AttributeError:
1558 # May not have gotten far enough in the __init__ to set
1559 # self.stdin, etc.
1560 pass
1561 self.__log.info("[%s] ProcessProxy: closing stdout (%r)."\
1562 % (id(self), self.stdout))
1563 try:
1564 self.stdout.close()
1565 if self._stdoutProxy is not threading.currentThread():
1566 self._stdoutProxy.join()
1567 except AttributeError:
1568 # May not have gotten far enough in the __init__ to set
1569 # self.stdout, etc.
1570 pass
1571 self.__log.info("[%s] ProcessProxy: closing stderr (%r)."\
1572 % (id(self), self.stderr))
1573 try:
1574 self.stderr.close()
1575 if self._stderrProxy is not threading.currentThread():
1576 self._stderrProxy.join()
1577 except AttributeError:
1578 # May not have gotten far enough in the __init__ to set
1579 # self.stderr, etc.
1580 pass
1581
1582 self._closed = 1
1583
1584 def _forkAndExecChildOnUnix(self, fdChildStdinRd, fdChildStdoutWr,
1585 fdChildStderrWr):
1586 """Fork and start the child process.
1587
1588 Sets self._pid as a side effect.
1589 """
1590 pid = os.fork()
1591 if pid == 0: # child
1592 os.dup2(fdChildStdinRd, 0)
1593 os.dup2(fdChildStdoutWr, 1)
1594 os.dup2(fdChildStderrWr, 2)
1595 self._runChildOnUnix()
1596 # parent
1597 self._pid = pid
1598
1599 def _startOnUnix(self):
1600 # Create pipes for std handles.
1601 fdChildStdinRd, fdChildStdinWr = os.pipe()
1602 fdChildStdoutRd, fdChildStdoutWr = os.pipe()
1603 fdChildStderrRd, fdChildStderrWr = os.pipe()
1604
1605 if self._cwd:
1606 oldDir = os.getcwd()
1607 try:
1608 os.chdir(self._cwd)
1609 except OSError, ex:
1610 raise ProcessError(msg=str(ex), errno=ex.errno)
1611 self._forkAndExecChildOnUnix(fdChildStdinRd, fdChildStdoutWr,
1612 fdChildStderrWr)
1613 if self._cwd:
1614 os.chdir(oldDir)
1615
1616 os.close(fdChildStdinRd)
1617 os.close(fdChildStdoutWr)
1618 os.close(fdChildStderrWr)
1619
1620 childStdin = _FileWrapper(descriptor=fdChildStdinWr)
1621 logres.info("[%s] ProcessProxy._start(): create child stdin: %r",
1622 id(self), childStdin)
1623 childStdout = _FileWrapper(descriptor=fdChildStdoutRd)
1624 logres.info("[%s] ProcessProxy._start(): create child stdout: %r",
1625 id(self), childStdout)
1626 childStderr = _FileWrapper(descriptor=fdChildStderrRd)
1627 logres.info("[%s] ProcessProxy._start(): create child stderr: %r",
1628 id(self), childStderr)
1629
1630 # Create proxy threads for the out pipes.
1631 self._stdinProxy = _InFileProxy(self.stdin, childStdin, name='<stdin>')
1632 self._stdinProxy.start()
1633 # Clean up the parent's side of <stdin> when it is observed that
1634 # the child has closed its side of <stdout> and <stderr>. (This
1635 # is one way of determining when it is appropriate to clean up
1636 # this pipe, with compromises. See the discussion at the top of
1637 # this module.)
1638 closer = _CountingCloser([self.stdin, childStdin, self], 2)
1639 self._stdoutProxy = _OutFileProxy(childStdout, self.stdout,
1640 [closer],
1641 name='<stdout>')
1642 self._stdoutProxy.start()
1643 self._stderrProxy = _OutFileProxy(childStderr, self.stderr,
1644 [closer],
1645 name='<stderr>')
1646 self._stderrProxy.start()
1647
1648 def _startOnWindows(self):
1649 if type(self._cmd) in (types.ListType, types.TupleType):
1650 # An arg vector was passed in.
1651 cmd = _joinArgv(self._cmd)
1652 else:
1653 cmd = self._cmd
1654
1655 # Create pipes for std handles.
1656 # (Set the bInheritHandle flag so pipe handles are inherited.)
1657 saAttr = pywintypes.SECURITY_ATTRIBUTES()
1658 saAttr.bInheritHandle = 1
1659 #XXX Should maybe try with os.pipe. Dunno what that does for
1660 # inheritability though.
1661 hChildStdinRd, hChildStdinWr = win32pipe.CreatePipe(saAttr, 0)
1662 hChildStdoutRd, hChildStdoutWr = win32pipe.CreatePipe(saAttr, 0)
1663 hChildStderrRd, hChildStderrWr = win32pipe.CreatePipe(saAttr, 0)
1664
1665 try:
1666 # Duplicate the parent ends of the pipes so they are not
1667 # inherited.
1668 hChildStdinWrDup = win32api.DuplicateHandle(
1669 win32api.GetCurrentProcess(),
1670 hChildStdinWr,
1671 win32api.GetCurrentProcess(),
1672 0,
1673 0, # not inherited
1674 DUPLICATE_SAME_ACCESS)
1675 win32api.CloseHandle(hChildStdinWr)
1676 self._hChildStdinWr = hChildStdinWrDup
1677 hChildStdoutRdDup = win32api.DuplicateHandle(
1678 win32api.GetCurrentProcess(),
1679 hChildStdoutRd,
1680 win32api.GetCurrentProcess(),
1681 0,
1682 0, # not inherited
1683 DUPLICATE_SAME_ACCESS)
1684 win32api.CloseHandle(hChildStdoutRd)
1685 self._hChildStdoutRd = hChildStdoutRdDup
1686 hChildStderrRdDup = win32api.DuplicateHandle(
1687 win32api.GetCurrentProcess(),
1688 hChildStderrRd,
1689 win32api.GetCurrentProcess(),
1690 0,
1691 0, # not inherited
1692 DUPLICATE_SAME_ACCESS)
1693 win32api.CloseHandle(hChildStderrRd)
1694 self._hChildStderrRd = hChildStderrRdDup
1695
1696 # Set the translation mode.
1697 if self._mode == 't':
1698 flags = os.O_TEXT
1699 mode = ''
1700 else:
1701 flags = 0
1702 mode = 'b'
1703 fdChildStdinWr = msvcrt.open_osfhandle(self._hChildStdinWr, flags)
1704 fdChildStdoutRd = msvcrt.open_osfhandle(self._hChildStdoutRd, flags)
1705 fdChildStderrRd = msvcrt.open_osfhandle(self._hChildStderrRd, flags)
1706
1707 childStdin = _FileWrapper(descriptor=fdChildStdinWr,
1708 handle=self._hChildStdinWr)
1709 logres.info("[%s] ProcessProxy._start(): create child stdin: %r",
1710 id(self), childStdin)
1711 childStdout = _FileWrapper(descriptor=fdChildStdoutRd,
1712 handle=self._hChildStdoutRd)
1713 logres.info("[%s] ProcessProxy._start(): create child stdout: %r",
1714 id(self), childStdout)
1715 childStderr = _FileWrapper(descriptor=fdChildStderrRd,
1716 handle=self._hChildStderrRd)
1717 logres.info("[%s] ProcessProxy._start(): create child stderr: %r",
1718 id(self), childStderr)
1719
1720 # Start the child process.
1721 si = win32process.STARTUPINFO()
1722 si.dwFlags = win32process.STARTF_USESHOWWINDOW
1723 si.wShowWindow = 0 # SW_HIDE
1724 si.hStdInput = hChildStdinRd
1725 si.hStdOutput = hChildStdoutWr
1726 si.hStdError = hChildStderrWr
1727 si.dwFlags |= win32process.STARTF_USESTDHANDLES
1728
1729 cmd = _fixupCommand(cmd, self._env)
1730 log.debug("cmd = %r", cmd)
1731
1732 creationFlags = win32process.CREATE_NEW_PROCESS_GROUP
1733 try:
1734 self._hProcess, hThread, self._processId, threadId\
1735 = _SaferCreateProcess(
1736 None, # app name
1737 cmd, # command line
1738 None, # process security attributes
1739 None, # primary thread security attributes
1740 1, # handles are inherited
1741 creationFlags, # creation flags
1742 self._env, # environment
1743 self._cwd, # current working directory
1744 si) # STARTUPINFO pointer
1745 except win32api.error, ex:
1746 raise ProcessError(msg=ex.args[2], errno=ex.args[0])
1747 win32api.CloseHandle(hThread)
1748
1749 finally:
1750 # Close child ends of pipes on the parent's side (the
1751 # parent's ends of the pipe are closed in the _FileWrappers.)
1752 win32file.CloseHandle(hChildStdinRd)
1753 win32file.CloseHandle(hChildStdoutWr)
1754 win32file.CloseHandle(hChildStderrWr)
1755
1756 # Create proxy threads for the pipes.
1757 self._stdinProxy = _InFileProxy(self.stdin, childStdin, name='<stdin>')
1758 self._stdinProxy.start()
1759 # Clean up the parent's side of <stdin> when it is observed that
1760 # the child has closed its side of <stdout>. (This is one way of
1761 # determining when it is appropriate to clean up this pipe, with
1762 # compromises. See the discussion at the top of this module.)
1763 self._stdoutProxy = _OutFileProxy(childStdout, self.stdout,
1764 [self.stdin, childStdin, self],
1765 name='<stdout>')
1766 self._stdoutProxy.start()
1767 self._stderrProxy = _OutFileProxy(childStderr, self.stderr,
1768 name='<stderr>')
1769 self._stderrProxy.start()
1770
1771 def wait(self, timeout=None):
1772 """Wait for the started process to complete.
1773
1774 "timeout" (on Windows) is a floating point number of seconds after
1775 which to timeout. Default is win32event.INFINITE.
1776 "timeout" (on Unix) is akin to the os.waitpid() "options" argument
1777 (os.WNOHANG may be used to return immediately if the process has
1778 not exited). Default is 0, i.e. wait forever.
1779
1780 If the wait time's out it will raise a ProcessError. Otherwise it
1781 will return the child's exit value (on Windows) or the child's exit
1782 status excoded as per os.waitpid() (on Linux):
1783 "a 16-bit number, whose low byte is the signal number that killed
1784 the process, and whose high byte is the exit status (if the
1785 signal number is zero); the high bit of the low byte is set if a
1786 core file was produced."
1787 In the latter case, use the os.W*() methods to interpret the return
1788 value.
1789 """
1790 # XXX Or should returning the exit value be move out to another
1791 # function as on Win32 process control? If so, then should
1792 # perhaps not make WaitForSingleObject semantic transformation.
1793 if sys.platform.startswith("win"):
1794 if timeout is None:
1795 timeout = win32event.INFINITE
1796 else:
1797 timeout = timeout * 1000.0 # Win32 API's timeout is in millisecs
1798
1799 rc = win32event.WaitForSingleObject(self._hProcess, timeout)
1800 if rc == win32event.WAIT_FAILED:
1801 raise ProcessError("'WAIT_FAILED' when waiting for process to "\
1802 "terminate: %r" % self._cmd, rc)
1803 elif rc == win32event.WAIT_TIMEOUT:
1804 raise ProcessError("'WAIT_TIMEOUT' when waiting for process to "\
1805 "terminate: %r" % self._cmd, rc)
1806
1807 retval = win32process.GetExitCodeProcess(self._hProcess)
1808 else:
1809 # os.waitpid() will raise:
1810 # OSError: [Errno 10] No child processes
1811 # on subsequent .wait() calls. Change these semantics to have
1812 # subsequent .wait() calls return the exit status and return
1813 # immediately without raising an exception.
1814 # (XXX It would require synchronization code to handle the case
1815 # of multiple simultaneous .wait() requests, however we can punt
1816 # on that because it is moot while Linux still has the problem
1817 # for which _ThreadFixer() exists.)
1818 if self.__retvalCache is not None:
1819 retval = self.__retvalCache
1820 else:
1821 if timeout is None:
1822 timeout = 0
1823 pid, sts = os.waitpid(self._pid, timeout)
1824 if pid == self._pid:
1825 self.__retvalCache = retval = sts
1826 else:
1827 raise ProcessError("Wait for process timed out.",
1828 self.WAIT_TIMEOUT)
1829 _unregisterProcess(self)
1830 return retval
1831
1832 def kill(self, exitCode=0, gracePeriod=1.0, sig=None):
1833 """Kill process.
1834
1835 "exitCode" [deprecated, not supported] (Windows only) is the
1836 code the terminated process should exit with.
1837 "gracePeriod" (Windows only) is a number of seconds the process is
1838 allowed to shutdown with a WM_CLOSE signal before a hard
1839 terminate is called.
1840 "sig" (Unix only) is the signal to use to kill the process. Defaults
1841 to signal.SIGKILL. See os.kill() for more information.
1842
1843 Windows:
1844 Try for an orderly shutdown via WM_CLOSE. If still running
1845 after gracePeriod (1 sec. default), terminate.
1846 """
1847 if sys.platform.startswith("win"):
1848 import win32gui
1849 # Send WM_CLOSE to windows in this process group.
1850 win32gui.EnumWindows(self._close_, 0)
1851
1852 # Send Ctrl-Break signal to all processes attached to this
1853 # console. This is supposed to trigger shutdown handlers in
1854 # each of the processes.
1855 try:
1856 win32api.GenerateConsoleCtrlEvent(CTRL_BREAK_EVENT,
1857 self._processId)
1858 except AttributeError:
1859 log.warn("The win32api module does not have "\
1860 "GenerateConsoleCtrlEvent(). This may mean that "\
1861 "parts of this process group have NOT been killed.")
1862 except win32api.error, ex:
1863 if ex.args[0] not in (6, 87):
1864 # Ignore the following:
1865 # api_error: (87, 'GenerateConsoleCtrlEvent', 'The parameter is incorrect.')
1866 # api_error: (6, 'GenerateConsoleCtrlEvent', 'The handle is invalid.')
1867 # Get error 6 if there is no console.
1868 raise
1869
1870 # Last resort: call TerminateProcess if it has not yet.
1871 retval = 0
1872 try:
1873 self.wait(gracePeriod)
1874 except ProcessError, ex:
1875 log.info("[%s] Process.kill: calling TerminateProcess", id(self))
1876 win32process.TerminateProcess(self._hProcess, -1)
1877 win32api.Sleep(100) # wait for resources to be released
1878
1879 else:
1880 if sig is None:
1881 sig = signal.SIGKILL
1882 try:
1883 os.kill(self._pid, sig)
1884 except OSError, ex:
1885 if ex.errno != 3:
1886 # Ignore: OSError: [Errno 3] No such process
1887 raise
1888
1889 _unregisterProcess(self)
1890
1891 def _close_(self, hwnd, dummy):
1892 """Callback used by .kill() on Windows.
1893
1894 EnumWindows callback - sends WM_CLOSE to any window owned by this
1895 process.
1896 """
1897 threadId, processId = win32process.GetWindowThreadProcessId(hwnd)
1898 if processId == self._processId:
1899 import win32gui
1900 win32gui.PostMessage(hwnd, WM_CLOSE, 0, 0)
1901
1902
1903class IOBuffer:
1904 """Want to be able to both read and write to this buffer from
1905 difference threads and have the same read/write semantics as for a
1906 std handler.
1907
1908 This class is subclass-able. _doRead(), _doWrite(), _doReadline(),
1909 _doClose(), _haveLine(), and _haveNumBytes() can be overridden for
1910 specific functionality. The synchronization issues (block on read
1911 until write provides the needed data, termination) are handled for
1912 free.
1913
1914 Cannot support:
1915 .seek() # Because we are managing *two* positions (one each
1916 .tell() # for reading and writing), these do not make
1917 # sense.
1918 """
1919 #TODO:
1920 # - Is performance a problem? This will likely be slower that
1921 # StringIO.StringIO().
1922 #
1923 def __init__(self, mutex=None, stateChange=None, name=None):
1924 """'name' can be set for debugging, it will be used in log messages."""
1925 if name is not None:
1926 self._name = name
1927 else:
1928 self._name = id(self)
1929 log.info("[%s] IOBuffer.__init__()" % self._name)
1930
1931 self.__buf = ''
1932 # A state change is defined as the buffer being closed or a
26ee3a06 1933 # write occuring.
1f780e48
RD
1934 if mutex is not None:
1935 self._mutex = mutex
1936 else:
1937 self._mutex = threading.Lock()
1938 if stateChange is not None:
1939 self._stateChange = stateChange
1940 else:
1941 self._stateChange = threading.Condition()
1942 self._closed = 0
1943
1944 def _doWrite(self, s):
1945 self.__buf += s # Append to buffer.
1946
1947 def write(self, s):
1948 log.info("[%s] IOBuffer.write(s=%r)", self._name, s)
1949 # Silently drop writes after the buffer has been close()'d.
1950 if self._closed:
1951 return
1952 # If empty write, close buffer (mimicking behaviour from
1953 # koprocess.cpp.)
1954 if not s:
1955 self.close()
1956 return
1957
1958 self._mutex.acquire()
1959 self._doWrite(s)
1960 self._stateChange.acquire()
1961 self._stateChange.notifyAll() # Notify of the write().
1962 self._stateChange.release()
1963 self._mutex.release()
1964
1965 def writelines(self, list):
1966 self.write(''.join(list))
1967
1968 def _doRead(self, n):
1969 """Pop 'n' bytes from the internal buffer and return them."""
1970 if n < 0:
1971 idx = len(self.__buf)
1972 else:
1973 idx = min(n, len(self.__buf))
1974 retval, self.__buf = self.__buf[:idx], self.__buf[idx:]
1975 return retval
1976
1977 def read(self, n=-1):
1978 log.info("[%s] IOBuffer.read(n=%r)" % (self._name, n))
1979 log.info("[%s] IOBuffer.read(): wait for data" % self._name)
1980 if n < 0:
1981 # Wait until the buffer is closed, i.e. no more writes will
1982 # come.
1983 while 1:
1984 if self._closed: break
1985 #log.debug("[%s] <<< IOBuffer.read: state change .wait()"\
1986 # % self._name)
1987 self._stateChange.acquire()
1988 self._stateChange.wait()
1989 self._stateChange.release()
1990 #log.debug("[%s] >>> IOBuffer.read: done change .wait()"\
1991 # % self._name)
1992 else:
1993 # Wait until there are the requested number of bytes to read
1994 # (or until the buffer is closed, i.e. no more writes will
1995 # come).
1996 # XXX WARNING: I *think* there is a race condition around
1997 # here whereby self.fparent.read() in _InFileProxy can
1998 # hang. *Sometime* test_stdin::test_stdin_buffer() will
1999 # hang. This was *before* I moved the
2000 # _stateChange.acquire() and .release() calls out side
2001 # of the 'while 1:' here. ...and now they are back
2002 # inside.
2003 while 1:
2004 if self._closed: break
2005 if self._haveNumBytes(n): break
2006 #log.debug("[%s] <<< IOBuffer.read: state change .wait()"\
2007 # % self._name)
2008 self._stateChange.acquire()
2009 self._stateChange.wait()
2010 self._stateChange.release()
2011 #log.debug("[%s] >>> IOBuffer.read: done change .wait()"\
2012 # % self._name)
2013 log.info("[%s] IOBuffer.read(): done waiting for data" % self._name)
2014
2015 self._mutex.acquire()
2016 retval = self._doRead(n)
2017 self._mutex.release()
2018 return retval
2019
2020 def _doReadline(self, n):
2021 """Pop the front line (or n bytes of it, whichever is less) from
2022 the internal buffer and return it.
2023 """
2024 idx = self.__buf.find('\n')
2025 if idx == -1:
2026 idx = len(self.__buf)
2027 else:
2028 idx += 1 # include the '\n'
2029 if n is not None:
2030 idx = min(idx, n)
2031 retval, self.__buf = self.__buf[:idx], self.__buf[idx:]
2032 return retval
2033
2034 def _haveLine(self):
2035 return self.__buf.find('\n') != -1
2036
2037 def _haveNumBytes(self, n=None):
2038 return len(self.__buf) >= n
2039
2040 def readline(self, n=None):
2041 # Wait until there is a full line (or at least 'n' bytes)
2042 # in the buffer or until the buffer is closed, i.e. no more
2043 # writes will come.
2044 log.info("[%s] IOBuffer.readline(n=%r)" % (self._name, n))
2045
2046 log.info("[%s] IOBuffer.readline(): wait for data" % self._name)
2047 while 1:
2048 if self._closed: break
2049 if self._haveLine(): break
2050 if n is not None and self._haveNumBytes(n): break
2051 self._stateChange.acquire()
2052 self._stateChange.wait()
2053 self._stateChange.release()
2054 log.info("[%s] IOBuffer.readline(): done waiting for data"\
2055 % self._name)
2056
2057 self._mutex.acquire()
2058 retval = self._doReadline(n)
2059 self._mutex.release()
2060 return retval
2061
2062 def readlines(self):
2063 lines = []
2064 while 1:
2065 line = self.readline()
2066 if line:
2067 lines.append(line)
2068 else:
2069 break
2070 return lines
2071
2072 def _doClose(self):
2073 pass
2074
2075 def close(self):
2076 if not self._closed:
2077 log.info("[%s] IOBuffer.close()" % self._name)
2078 self._doClose()
2079 self._closed = 1
2080 self._stateChange.acquire()
2081 self._stateChange.notifyAll() # Notify of the close().
2082 self._stateChange.release()
2083
2084 def flush(self):
2085 log.info("[%s] IOBuffer.flush()" % self._name)
2086 #XXX Perhaps flush() should unwedged possible waiting .read()
2087 # and .readline() calls that are waiting for more data???
2088
2089
2090class _InFileProxy(threading.Thread):
2091 """A thread to proxy stdin.write()'s from the parent to the child."""
2092 def __init__(self, fParent, fChild, name=None):
2093 """
2094 "fParent" is a Python file-like object setup for writing.
2095 "fChild" is a Win32 handle to the a child process' output pipe.
2096 "name" can be set for debugging, it will be used in log messages.
2097 """
2098 log.info("[%s, %s] _InFileProxy.__init__(fChild=%r, fParent=%r)",
2099 name, id(self), fChild, fParent)
2100 threading.Thread.__init__(self, name=name)
2101 self.fChild = fChild
2102 self.fParent = fParent
2103
2104 def run(self):
2105 log.info("[%s] _InFileProxy: start" % self.getName())
2106 try:
2107 self._proxyFromParentToChild()
2108 finally:
2109 log.info("[%s] _InFileProxy: closing parent (%r)"\
2110 % (self.getName(), self.fParent))
2111 try:
2112 self.fParent.close()
2113 except IOError:
2114 pass # Ignore: IOError: [Errno 4] Interrupted system call
2115 log.info("[%s] _InFileProxy: done" % self.getName())
2116
2117 def _proxyFromParentToChild(self):
2118 CHUNKSIZE = 4096
2119 # Read output from the child process, and (for now) just write
2120 # it out.
2121 while 1:
2122 log.info("[%s] _InFileProxy: waiting for read on parent (%r)"\
2123 % (self.getName(), self.fParent))
2124 # XXX Get hangs here (!) even with
2125 # self.stdin.close() in ProcessProxy' __del__() under this
2126 # cond:
2127 # p = ProcessProxy([...], stdin=sys.stdin)
2128 # The user must manually send '\n' via <Enter> or EOF
2129 # via <Ctrl-Z> to unlock this. How to get around that?
2130 # See cleanOnTermination note in _OutFileProxy.run()
2131 # below.
2132 #log.debug("XXX -> start read on %r" % self.fParent)
2133 try:
2134 text = self.fParent.read(CHUNKSIZE)
2135 except ValueError, ex:
2136 # ValueError is raised with trying to write to a closed
2137 # file/pipe.
2138 text = None
2139 #log.debug("XXX <- done read on %r" % self.fParent)
2140 if not text:
2141 # Empty text signifies that the pipe has been closed on
2142 # the parent's end.
2143 log.info("[%s] _InFileProxy: observed close of parent (%r)"\
2144 % (self.getName(), self.fParent))
2145 # Signal the child so it knows to stop listening.
2146 try:
2147 logres.info("[%s] _InFileProxy: closing child after "\
2148 "observing parent's close: %r", self.getName(),
2149 self.fChild)
2150 try:
2151 self.fChild.close()
2152 except IOError:
2153 pass # Ignore: IOError: [Errno 4] Interrupted system call
2154 except IOError, ex:
2155 # Ignore: IOError: [Errno 9] Bad file descriptor
2156 # XXX Do we *know* we want to do that?
2157 pass
2158 break
2159 else:
2160 log.info("[%s] _InFileProxy: read %d bytes from parent: %r"\
2161 % (self.getName(), len(text), text))
2162
2163 log.info("[%s, %s] _InFileProxy: writing %r to child (%r)",
2164 self.getName(), id(self), text, self.fChild)
2165 try:
2166 self.fChild.write(text)
2167 except (OSError, IOError), ex:
2168 # Ignore errors for now. For example:
2169 # - Get this on Win9x when writing multiple lines to "dir":
2170 # OSError: [Errno 32] Broken pipe
2171 #XXX There *may* be errors we don't want to avoid.
2172 #XXX Should maybe just ignore EnvironmentError (base class).
2173 log.info("[%s] _InFileProxy: error writing to child (%r), "\
2174 "closing: %s" % (self.getName(), self.fParent, ex))
2175 break
2176 log.info("[%s] _InFileProxy: wrote %d bytes to child: %r"\
2177 % (self.getName(), len(text), text))
2178
2179
2180class _OutFileProxy(threading.Thread):
2181 """A thread to watch an "out" file from the spawned child process
2182 and pass on write's to the parent.
2183 """
2184 def __init__(self, fChild, fParent, toClose=[], name=None):
2185 """
2186 "fChild" is a Win32 handle to the a child process' output pipe.
2187 "fParent" is a Python file-like object setup for writing.
2188 "toClose" is a list of objects on which to call .close when this
2189 proxy is terminating.
2190 "name" can be set for debugging, it will be used in log messages.
2191 """
2192 log.info("[%s] _OutFileProxy.__init__(fChild=%r, fParent=%r, "\
2193 "toClose=%r)", name, fChild, fParent, toClose)
2194 threading.Thread.__init__(self, name=name)
2195 self.fChild = fChild
2196 self.fParent = fParent
2197 self.toClose = toClose
2198
2199 def run(self):
2200 log.info("[%s] _OutFileProxy: start" % self.getName())
2201 try:
2202 self._proxyFromChildToParent()
2203 finally:
2204 logres.info("[%s] _OutFileProxy: terminating, close child (%r)",
2205 self.getName(), self.fChild)
2206 try:
2207 self.fChild.close()
2208 except IOError:
2209 pass # Ignore: IOError: [Errno 4] Interrupted system call
2210 log.info("[%s] _OutFileProxy: closing parent (%r)",
2211 self.getName(), self.fParent)
2212 try:
2213 self.fParent.close()
2214 except IOError:
2215 pass # Ignore: IOError: [Errno 4] Interrupted system call
2216 while self.toClose:
2217 logres.info("[%s] _OutFileProxy: closing %r after "\
2218 "closing parent", self.getName(), self.toClose[0])
2219 try:
2220 self.toClose[0].close()
2221 except IOError:
2222 pass # Ignore: IOError: [Errno 4] Interrupted system call
2223 del self.toClose[0]
2224 log.info("[%s] _OutFileProxy: done" % self.getName())
2225
2226 def _proxyFromChildToParent(self):
2227 CHUNKSIZE = 4096
2228 # Read output from the child process, and (for now) just write
2229 # it out.
2230 while 1:
2231 text = None
2232 try:
2233 log.info("[%s] _OutFileProxy: waiting for read on child (%r)"\
2234 % (self.getName(), self.fChild))
2235 text = self.fChild.read(CHUNKSIZE)
2236 except IOError, ex:
2237 # Ignore: IOError: [Errno 9] Bad file descriptor
2238 # XXX Do we *know* we want to do that?
2239 log.info("[%s] _OutFileProxy: error reading from child (%r), "\
2240 "shutting down: %s", self.getName(), self.fChild, ex)
2241 break
2242 if not text:
2243 # Empty text signifies that the pipe has been closed on
2244 # the child's end.
2245 log.info("[%s] _OutFileProxy: observed close of child (%r)"\
2246 % (self.getName(), self.fChild))
2247 break
2248
2249 log.info("[%s] _OutFileProxy: text(len=%d): %r",
2250 self.getName(), len(text), text)
2251 self.fParent.write(text)
2252
2253
2254
2255if sys.platform.startswith("linux"):
2256 class _ThreadFixer:
2257 """Mixin class for various classes in the Process hierarchy to
2258 work around the known LinuxThreads bug where one cannot .wait()
2259 on a created process from a subthread of the thread that created
2260 the process.
2261
2262 Usage:
2263 class ProcessXXX(_ThreadFixer, BrokenProcessXXX):
2264 _pclass = BrokenProcessXXX
2265
2266 Details:
2267 Because we must do all real os.wait() calls on the child
2268 process from the thread that spawned it, we use a proxy
2269 thread whose only responsibility is just that. The proxy
2270 thread just starts the child and then immediately wait's for
2271 the child to terminate. On termination is stores the exit
2272 status (for use by the main thread) and notifies any thread
2273 waiting for this termination (possibly the main thread). The
2274 overriden .wait() uses this stored exit status and the
2275 termination notification to simulate the .wait().
2276 """
2277 def __init__(self, *args, **kwargs):
2278 # Keep a reference to 'log' ensure it is around for this object's
2279 # destruction.
2280 self.__log = log
2281 self.__waiter = None
2282 self.__hasTerminated = threading.Condition()
2283 self.__terminationResult = None
2284 self.__childStarted = threading.Condition()
2285 self._pclass.__init__(self, *args, **kwargs)
2286
2287 def _forkAndExecChildOnUnix(self, *args, **kwargs):
2288 """Fork and start the child process do it in a special subthread
2289 that will negotiate subsequent .wait()'s.
2290
2291 Sets self._pid as a side effect.
2292 """
2293 self.__waiter = threading.Thread(
2294 target=self.__launchAndWait, args=args, kwargs=kwargs)
2295
2296 # Start subthread that will launch child and wait until it
2297 # *has* started.
2298 self.__childStarted.acquire()
2299 self.__waiter.start()
2300 self.__childStarted.wait()
2301 self.__childStarted.release()
2302
2303 def __launchAndWait(self, *args, **kwargs):
2304 """Launch the given command and wait for it to terminate.
2305
2306 When the process has terminated then store its exit value
2307 and finish.
2308 """
2309 logfix.info("start child in thread %s",
2310 threading.currentThread().getName())
2311
2312 # Spawn the child process and notify the main thread of
2313 # this.
2314 self.__childStarted.acquire()
2315 self._pclass._forkAndExecChildOnUnix(self, *args, **kwargs)
2316 self.__childStarted.notifyAll()
2317 self.__childStarted.release()
2318
2319 # Wait on the thread and store appropriate results when
2320 # finished.
2321 try:
2322 waitResult = self._pclass.wait(self)
2323 except ProcessError, ex:
2324 waitResult = ex
2325 self.__hasTerminated.acquire()
2326 self.__terminationResult = waitResult
2327 self.__hasTerminated.notifyAll()
2328 self.__hasTerminated.release()
2329
2330 self.__waiter = None # drop ref that would keep instance alive
2331
2332 def wait(self, timeout=None):
2333 # If the process __hasTerminated then return the exit
2334 # status. Otherwise simulate the wait as appropriate.
2335 # Note:
2336 # - This class is only used on linux so 'timeout' has the
2337 # Unix 'timeout' semantics.
2338 self.__hasTerminated.acquire()
2339 if self.__terminationResult is None:
2340 if timeout == os.WNOHANG: # Poll.
2341 self.__hasTerminated.wait(0)
2342 else: # Block until process finishes.
2343 self.__hasTerminated.wait()
2344 terminationResult = self.__terminationResult
2345 self.__hasTerminated.release()
2346
2347 if terminationResult is None:
2348 # process has not finished yet
2349 raise ProcessError("Wait for process timed out.",
2350 self.WAIT_TIMEOUT)
2351 elif isinstance(terminationResult, Exception):
2352 # some error waiting for process termination
2353 raise terminationResult
2354 else:
2355 # the process terminated
2356 return terminationResult
2357
2358 _ThreadBrokenProcess = Process
2359 class Process(_ThreadFixer, _ThreadBrokenProcess):
2360 _pclass = _ThreadBrokenProcess
2361
2362 _ThreadBrokenProcessOpen = ProcessOpen
2363 class ProcessOpen(_ThreadFixer, _ThreadBrokenProcessOpen):
2364 _pclass = _ThreadBrokenProcessOpen
2365
2366 _ThreadBrokenProcessProxy = ProcessProxy
2367 class ProcessProxy(_ThreadFixer, _ThreadBrokenProcessProxy):
2368 _pclass = _ThreadBrokenProcessProxy
2369
2370