##// END OF EJS Templates
Simplify comparison
Simplify comparison

File last commit:

r21253:ff3b995a
r21780:e3b7b169
Show More
script.py
282 lines | 8.9 KiB | text/x-python | PythonLexer
MinRK
add script magics...
r7299 """Magic functions for running cells in various scripts."""
Thomas Kluyver
Convert print statements to print function calls...
r13348 from __future__ import print_function
MinRK
add script magics...
r7299 #-----------------------------------------------------------------------------
# Copyright (c) 2012 The IPython Development Team.
#
# Distributed under the terms of the Modified BSD License.
#
# The full license is in the file COPYING.txt, distributed with this software.
#-----------------------------------------------------------------------------
#-----------------------------------------------------------------------------
# Imports
#-----------------------------------------------------------------------------
# Stdlib
MinRK
better message on script magic not found...
r8470 import errno
MinRK
add script magics...
r7299 import os
import sys
Takafumi Arakaki
Treat interruption in the script magic (blocking case)...
r7561 import signal
import time
MinRK
add script magics...
r7299 from subprocess import Popen, PIPE
Takafumi Arakaki
Use atexit.register to cleanup in script magic
r7622 import atexit
MinRK
add script magics...
r7299
# Our own packages
Min RK
update dependency imports...
r21253 from traitlets.config.configurable import Configurable
MinRK
update ScriptMagics per review...
r7398 from IPython.core import magic_arguments
MinRK
add script magics...
r7299 from IPython.core.magic import (
Magics, magics_class, line_magic, cell_magic
)
MinRK
update ScriptMagics per review...
r7398 from IPython.lib.backgroundjobs import BackgroundJobManager
from IPython.utils import py3compat
MinRK
skip find_cmd when setting up script magics...
r8469 from IPython.utils.process import arg_split
Min RK
update dependency imports...
r21253 from traitlets import List, Dict
MinRK
add script magics...
r7299
#-----------------------------------------------------------------------------
# Magic implementation classes
#-----------------------------------------------------------------------------
MinRK
update ScriptMagics per review...
r7398 def script_args(f):
"""single decorator for adding script args"""
args = [
magic_arguments.argument(
'--out', type=str,
help="""The variable in which to store stdout from the script.
If the script is backgrounded, this will be the stdout *pipe*,
instead of the stderr text itself.
"""
),
magic_arguments.argument(
'--err', type=str,
help="""The variable in which to store stderr from the script.
If the script is backgrounded, this will be the stderr *pipe*,
instead of the stderr text itself.
"""
),
magic_arguments.argument(
'--bg', action="store_true",
help="""Whether to run the script in the background.
If given, the only way to see the output of the command is
with --out/err.
"""
),
Takafumi Arakaki
Add --proc option to %%script...
r7549 magic_arguments.argument(
'--proc', type=str,
help="""The variable in which to store Popen instance.
This is used only when --bg option is given.
"""
),
MinRK
update ScriptMagics per review...
r7398 ]
for arg in args:
f = arg(f)
return f
MinRK
add script magics...
r7299 @magics_class
Matthias BUSSONNIER
By default, Magics inherit from Configurable
r13237 class ScriptMagics(Magics):
MinRK
add script magics...
r7299 """Magics for talking to scripts
This defines a base `%%script` cell magic for running a cell
with a program in a subprocess, and registers a few top-level
magics that call %%script with common interpreters.
"""
script_magics = List(config=True,
help="""Extra script cell magics to define
This generates simple wrappers of `%%script foo` as `%%foo`.
If you want to add script magics that aren't on your path,
specify them in script_paths
""",
)
def _script_magics_default(self):
MinRK
skip find_cmd when setting up script magics...
r8469 """default to a common list of programs"""
MinRK
add script magics...
r7299
MinRK
skip find_cmd when setting up script magics...
r8469 defaults = [
MinRK
add script magics...
r7299 'sh',
'bash',
'perl',
'ruby',
MinRK
skip find_cmd when setting up script magics...
r8469 'python',
Thomas Kluyver
Add %%python2 cell magic....
r16020 'python2',
MinRK
add script magics...
r7299 'python3',
'pypy',
MinRK
skip find_cmd when setting up script magics...
r8469 ]
if os.name == 'nt':
defaults.extend([
'cmd',
])
MinRK
add script magics...
r7299
return defaults
script_paths = Dict(config=True,
help="""Dict mapping short 'ruby' names to full paths, such as '/opt/secret/bin/ruby'
Only necessary for items in script_magics where the default path will not
find the right interpreter.
"""
)
def __init__(self, shell=None):
Matthias BUSSONNIER
By default, Magics inherit from Configurable
r13237 super(ScriptMagics, self).__init__(shell=shell)
MinRK
add script magics...
r7299 self._generate_script_magics()
MinRK
update ScriptMagics per review...
r7398 self.job_manager = BackgroundJobManager()
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 self.bg_processes = []
Takafumi Arakaki
Use atexit.register to cleanup in script magic
r7622 atexit.register(self.kill_bg_processes)
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618
def __del__(self):
self.kill_bg_processes()
MinRK
add script magics...
r7299
def _generate_script_magics(self):
cell_magics = self.magics['cell']
for name in self.script_magics:
cell_magics[name] = self._make_script_magic(name)
def _make_script_magic(self, name):
"""make a named magic, that calls %%script with a particular program"""
# expand to explicit path if necessary:
script = self.script_paths.get(name, name)
MinRK
update ScriptMagics per review...
r7398 @magic_arguments.magic_arguments()
@script_args
MinRK
add script magics...
r7299 def named_script_magic(line, cell):
# if line, add it as cl-flags
if line:
line = "%s %s" % (script, line)
else:
line = script
return self.shebang(line, cell)
# write a basic docstring:
named_script_magic.__doc__ = \
"""%%{name} script magic
Run cells with {script} in a subprocess.
This is a shortcut for `%%script {script}`
""".format(**locals())
return named_script_magic
MinRK
update ScriptMagics per review...
r7398
@magic_arguments.magic_arguments()
@script_args
MinRK
add script magics...
r7299 @cell_magic("script")
def shebang(self, line, cell):
"""Run a cell via a shell command
The `%%script` line is like the #! line of script,
specifying a program (bash, perl, ruby, etc.) with which to run.
The rest of the cell is run by that program.
Examples
--------
::
In [1]: %%script bash
...: for i in 1 2 3; do
...: echo $i
...: done
1
2
3
"""
MinRK
update ScriptMagics per review...
r7398 argv = arg_split(line, posix = not sys.platform.startswith('win'))
args, cmd = self.shebang.parser.parse_known_args(argv)
MinRK
better message on script magic not found...
r8470
try:
p = Popen(cmd, stdout=PIPE, stderr=PIPE, stdin=PIPE)
except OSError as e:
if e.errno == errno.ENOENT:
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("Couldn't find program: %r" % cmd[0])
MinRK
better message on script magic not found...
r8470 return
else:
raise
MinRK
py3compat fixes for %%script and tests
r7440
Eric Galloway
Add newline if missing to end of script magic cell
r17171 if not cell.endswith('\n'):
cell += '\n'
MinRK
py3compat fixes for %%script and tests
r7440 cell = cell.encode('utf8', 'replace')
MinRK
update ScriptMagics per review...
r7398 if args.bg:
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 self.bg_processes.append(p)
Takafumi Arakaki
Garbage collect finished processes
r7630 self._gc_bg_processes()
MinRK
update ScriptMagics per review...
r7398 if args.out:
self.shell.user_ns[args.out] = p.stdout
if args.err:
MinRK
add basic tests for %%file, %%script
r7407 self.shell.user_ns[args.err] = p.stderr
Takafumi Arakaki
Start threads for %%script as daemon thread
r7621 self.job_manager.new(self._run_script, p, cell, daemon=True)
Takafumi Arakaki
Add --proc option to %%script...
r7549 if args.proc:
self.shell.user_ns[args.proc] = p
MinRK
update ScriptMagics per review...
r7398 return
Takafumi Arakaki
Treat interruption in the script magic (blocking case)...
r7561 try:
out, err = p.communicate(cell)
except KeyboardInterrupt:
Takafumi Arakaki
Avoid error while killing subprocess in %%script
r7562 try:
p.send_signal(signal.SIGINT)
time.sleep(0.1)
if p.poll() is not None:
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("Process is interrupted.")
Takafumi Arakaki
Avoid error while killing subprocess in %%script
r7562 return
p.terminate()
time.sleep(0.1)
if p.poll() is not None:
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("Process is terminated.")
Takafumi Arakaki
Avoid error while killing subprocess in %%script
r7562 return
p.kill()
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("Process is killed.")
Takafumi Arakaki
Improve error handling in blocking %%script
r7565 except OSError:
pass
except Exception as e:
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("Error while terminating subprocess (pid=%i): %s" \
% (p.pid, e))
Takafumi Arakaki
Treat interruption in the script magic (blocking case)...
r7561 return
MinRK
update ScriptMagics per review...
r7398 out = py3compat.bytes_to_str(out)
err = py3compat.bytes_to_str(err)
if args.out:
self.shell.user_ns[args.out] = out
else:
sys.stdout.write(out)
sys.stdout.flush()
if args.err:
self.shell.user_ns[args.err] = err
else:
sys.stderr.write(err)
sys.stderr.flush()
MinRK
add script magics...
r7299
MinRK
update ScriptMagics per review...
r7398 def _run_script(self, p, cell):
"""callback for running the script in the background"""
p.stdin.write(cell)
p.stdin.close()
p.wait()
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618
Takafumi Arakaki
Make kill_bg_processes as a magic
r7619 @line_magic("killbgscripts")
Takafumi Arakaki
Make %killbgscripts bit more verbose
r7629 def killbgscripts(self, _nouse_=''):
"""Kill all BG processes started by %%script and its family."""
self.kill_bg_processes()
Thomas Kluyver
Convert print statements to print function calls...
r13348 print("All background processes were killed.")
Takafumi Arakaki
Make %killbgscripts bit more verbose
r7629
def kill_bg_processes(self):
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 """Kill all BG processes which are still running."""
for p in self.bg_processes:
if p.poll() is None:
try:
p.send_signal(signal.SIGINT)
except:
pass
time.sleep(0.1)
for p in self.bg_processes:
if p.poll() is None:
try:
p.terminate()
except:
pass
time.sleep(0.1)
for p in self.bg_processes:
if p.poll() is None:
try:
p.kill()
except:
pass
Takafumi Arakaki
Garbage collect finished processes
r7630 self._gc_bg_processes()
def _gc_bg_processes(self):
self.bg_processes = [p for p in self.bg_processes if p.poll() is None]