##// END OF EJS Templates
pyflakes to avoid missing imports
pyflakes to avoid missing imports

File last commit:

r26208:778dac10
r26208:778dac10
Show More
script.py
316 lines | 10.1 KiB | text/x-python | PythonLexer
MinRK
add script magics...
r7299 """Magic functions for running cells in various scripts."""
Min RK
adopt traitlets 4.2 API...
r22340 # Copyright (c) IPython Development Team.
# Distributed under the terms of the Modified BSD License.
MinRK
add script magics...
r7299
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
Ethan Madden
Line-buffered output for %%magic scripts
r26199 import asyncio
Takafumi Arakaki
Use atexit.register to cleanup in script magic
r7622 import atexit
MinRK
add script magics...
r7299
Matthias Bussonnier
Forgot import in #12639
r26207 from subprocess import CalledProcessError
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
MinRK
skip find_cmd when setting up script magics...
r8469 from IPython.utils.process import arg_split
Min RK
adopt traitlets 4.2 API...
r22340 from traitlets import List, Dict, default
MinRK
add script magics...
r7299
Matthias Bussonnier
Forgot import in #12639
r26207
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*,
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 instead of the stderr text itself and will not be auto closed.
MinRK
update ScriptMagics per review...
r7398 """
),
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*,
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 instead of the stderr text itself and will not be autoclosed.
MinRK
update ScriptMagics per review...
r7398 """
),
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.
"""
),
M Pacer
create new --raise-error argument to raise an error if nonzero exitcode
r24410 magic_arguments.argument(
Matthias Bussonnier
Switch scripts cell magic default to raise if error happens....
r24519 '--no-raise-error', action="store_false", dest='raise_error',
M Pacer
create new --raise-error argument to raise an error if nonzero exitcode
r24410 help="""Whether you should raise an error message in addition to
a stream on stderr if you get a nonzero exit code.
"""
)
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.
"""
Min RK
adopt traitlets 4.2 API...
r22340 script_magics = List(
MinRK
add script magics...
r7299 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
""",
Min RK
adopt traitlets 4.2 API...
r22340 ).tag(config=True)
@default('script_magics')
MinRK
add script magics...
r7299 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
Min RK
adopt traitlets 4.2 API...
r22340 script_paths = Dict(
MinRK
add script magics...
r7299 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.
"""
Min RK
adopt traitlets 4.2 API...
r22340 ).tag(config=True)
MinRK
add script magics...
r7299
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:
Matthias Bussonnier
pyflakes to avoid missing imports
r26208 line = "%s %s" % (script, line)
MinRK
add script magics...
r7299 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
"""
Ethan Madden
Line-buffered output for %%magic scripts
r26199
async def _handle_stream(stream, stream_arg, file_object):
while True:
line = (await stream.readline()).decode("utf8")
if not line:
break
if stream_arg:
self.shell.user_ns[stream_arg] = line
else:
file_object.write(line)
file_object.flush()
async def _stream_communicate(process, cell):
process.stdin.write(cell)
process.stdin.close()
stdout_task = asyncio.create_task(
_handle_stream(process.stdout, args.out, sys.stdout)
)
stderr_task = asyncio.create_task(
_handle_stream(process.stderr, args.err, sys.stderr)
)
await asyncio.wait([stdout_task, stderr_task])
if sys.platform.startswith("win"):
asyncio.set_event_loop_policy(asyncio.WindowsProactorEventLoopPolicy())
loop = asyncio.get_event_loop()
Ethan Madden
Linting fixes
r26200 argv = arg_split(line, posix=not sys.platform.startswith("win"))
MinRK
update ScriptMagics per review...
r7398 args, cmd = self.shebang.parser.parse_known_args(argv)
MinRK
better message on script magic not found...
r8470 try:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 p = loop.run_until_complete(
asyncio.create_subprocess_exec(
*cmd,
stdout=asyncio.subprocess.PIPE,
stderr=asyncio.subprocess.PIPE,
stdin=asyncio.subprocess.PIPE
)
)
MinRK
better message on script magic not found...
r8470 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()
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 to_close = []
MinRK
update ScriptMagics per review...
r7398 if args.out:
self.shell.user_ns[args.out] = p.stdout
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 else:
to_close.append(p.stdout)
MinRK
update ScriptMagics per review...
r7398 if args.err:
MinRK
add basic tests for %%file, %%script
r7407 self.shell.user_ns[args.err] = p.stderr
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 else:
to_close.append(p.stderr)
self.job_manager.new(self._run_script, p, cell, to_close, 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:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 loop.run_until_complete(_stream_communicate(p, cell))
Takafumi Arakaki
Treat interruption in the script magic (blocking case)...
r7561 except KeyboardInterrupt:
Takafumi Arakaki
Avoid error while killing subprocess in %%script
r7562 try:
p.send_signal(signal.SIGINT)
time.sleep(0.1)
Ethan Madden
Line-buffered output for %%magic scripts
r26199 if p.returncode 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)
Ethan Madden
Line-buffered output for %%magic scripts
r26199 if p.returncode 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:
Matthias Bussonnier
pyflakes to avoid missing imports
r26208 print("Error while terminating subprocess (pid=%i): %s" % (p.pid, e))
Takafumi Arakaki
Treat interruption in the script magic (blocking case)...
r7561 return
M Pacer
create new --raise-error argument to raise an error if nonzero exitcode
r24410 if args.raise_error and p.returncode!=0:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 raise CalledProcessError(p.returncode, cell)
MinRK
add script magics...
r7299
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 def _run_script(self, p, cell, to_close):
MinRK
update ScriptMagics per review...
r7398 """callback for running the script in the background"""
p.stdin.write(cell)
p.stdin.close()
Matthias Bussonnier
Autoclose stdin/err for bg scripts, when not assigned to user_ns...
r24375 for s in to_close:
s.close()
MinRK
update ScriptMagics per review...
r7398 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."""
Leo Singer
Speed up shutdown by eliminating unnecessary sleeping...
r23508 if not self.bg_processes:
return
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 for p in self.bg_processes:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 if p.returncode is None:
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 try:
p.send_signal(signal.SIGINT)
except:
pass
time.sleep(0.1)
Leo Singer
Speed up shutdown by eliminating unnecessary sleeping...
r23508 self._gc_bg_processes()
if not self.bg_processes:
return
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 for p in self.bg_processes:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 if p.returncode is None:
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 try:
p.terminate()
except:
pass
time.sleep(0.1)
Leo Singer
Speed up shutdown by eliminating unnecessary sleeping...
r23508 self._gc_bg_processes()
if not self.bg_processes:
return
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 for p in self.bg_processes:
Ethan Madden
Line-buffered output for %%magic scripts
r26199 if p.returncode is None:
Takafumi Arakaki
Add kill_bg_processes for %%script magic
r7618 try:
p.kill()
except:
pass
Takafumi Arakaki
Garbage collect finished processes
r7630 self._gc_bg_processes()
def _gc_bg_processes(self):
Ethan Madden
Line-buffered output for %%magic scripts
r26199 self.bg_processes = [p for p in self.bg_processes if p.returncode is None]