|
|
""" Preliminary "job control" extensions for IPython
|
|
|
|
|
|
requires python 2.4 (or separate 'subprocess' module
|
|
|
|
|
|
This provides 2 features, launching background jobs and killing foreground jobs from another IPython instance.
|
|
|
|
|
|
Launching background jobs:
|
|
|
|
|
|
Usage:
|
|
|
|
|
|
[ipython]|2> import jobctrl
|
|
|
[ipython]|3> &ls
|
|
|
<3> <jobctrl.IpyPopen object at 0x00D87FD0>
|
|
|
[ipython]|4> _3.go
|
|
|
-----------> _3.go()
|
|
|
ChangeLog
|
|
|
IPython
|
|
|
MANIFEST.in
|
|
|
README
|
|
|
README_Windows.txt
|
|
|
|
|
|
...
|
|
|
|
|
|
Killing foreground tasks:
|
|
|
|
|
|
Launch IPython instance, run a blocking command:
|
|
|
|
|
|
[Q:/ipython]|1> import jobctrl
|
|
|
[Q:/ipython]|2> cat
|
|
|
|
|
|
Now launch a new IPython prompt and kill the process:
|
|
|
|
|
|
IPython 0.8.3.svn.r2919 [on Py 2.5]
|
|
|
[Q:/ipython]|1> import jobctrl
|
|
|
[Q:/ipython]|2> %tasks
|
|
|
6020: 'cat ' (Q:\ipython)
|
|
|
[Q:/ipython]|3> %kill
|
|
|
SUCCESS: The process with PID 6020 has been terminated.
|
|
|
[Q:/ipython]|4>
|
|
|
|
|
|
(you don't need to specify PID for %kill if only one task is running)
|
|
|
"""
|
|
|
|
|
|
from subprocess import Popen,PIPE
|
|
|
import os,shlex,sys,time
|
|
|
|
|
|
from IPython import genutils
|
|
|
|
|
|
import IPython.ipapi
|
|
|
|
|
|
if os.name == 'nt':
|
|
|
def kill_process(pid):
|
|
|
os.system('taskkill /F /PID %d' % pid)
|
|
|
else:
|
|
|
def kill_process(pid):
|
|
|
os.system('kill -9 %d' % pid)
|
|
|
|
|
|
|
|
|
|
|
|
class IpyPopen(Popen):
|
|
|
def go(self):
|
|
|
print self.communicate()[0]
|
|
|
def __repr__(self):
|
|
|
return '<IPython job "%s" PID=%d>' % (self.line, self.pid)
|
|
|
|
|
|
def kill(self):
|
|
|
kill_process(self.pid)
|
|
|
|
|
|
def startjob(job):
|
|
|
p = IpyPopen(shlex.split(job), stdout=PIPE, shell = False)
|
|
|
p.line = job
|
|
|
return p
|
|
|
|
|
|
def jobctrl_prefilter_f(self,line):
|
|
|
if line.startswith('&'):
|
|
|
pre,fn,rest = self.split_user_input(line[1:])
|
|
|
|
|
|
line = ip.IP.expand_aliases(fn,rest)
|
|
|
return '_ip.startjob(%s)' % genutils.make_quoted_expr(line)
|
|
|
|
|
|
raise IPython.ipapi.TryNext
|
|
|
|
|
|
|
|
|
def job_list(ip):
|
|
|
keys = ip.db.keys('tasks/*')
|
|
|
ents = [ip.db[k] for k in keys]
|
|
|
return ents
|
|
|
|
|
|
def magic_tasks(self,line):
|
|
|
""" Show a list of tasks.
|
|
|
|
|
|
A 'task' is a process that has been started in IPython when 'jobctrl' extension is enabled.
|
|
|
Tasks can be killed with %kill.
|
|
|
"""
|
|
|
ip = self.getapi()
|
|
|
ents = job_list(ip)
|
|
|
if not ents:
|
|
|
print "No tasks running"
|
|
|
for pid,cmd,cwd,t in ents:
|
|
|
dur = int(time.time()-t)
|
|
|
print "%d: '%s' (%s) %d:%02d" % (pid,cmd,cwd, dur / 60,dur%60)
|
|
|
|
|
|
def magic_kill(self,line):
|
|
|
""" Kill a task
|
|
|
|
|
|
Without args, either kill one task (if only one running) or show list (if many)
|
|
|
With arg, assume it's the process id.
|
|
|
|
|
|
%kill is typically (much) more powerful than trying to terminate a process with ctrl+C.
|
|
|
"""
|
|
|
ip = self.getapi()
|
|
|
jobs = job_list(ip)
|
|
|
|
|
|
if not line.strip():
|
|
|
if len(jobs) == 1:
|
|
|
kill_process(jobs[0][0])
|
|
|
else:
|
|
|
magic_tasks(self,line)
|
|
|
return
|
|
|
|
|
|
try:
|
|
|
pid = int(line)
|
|
|
kill_process(pid)
|
|
|
except ValueError:
|
|
|
magic_tasks(self,line)
|
|
|
|
|
|
if sys.platform == 'win32':
|
|
|
shell_internal_commands = 'break chcp cls copy ctty date del erase dir md mkdir path prompt rd rmdir time type ver vol'.split()
|
|
|
else:
|
|
|
# todo linux commands
|
|
|
shell_internal_commands = []
|
|
|
|
|
|
|
|
|
def jobctrl_shellcmd(ip,cmd):
|
|
|
""" os.system replacement that stores process info to db['tasks/t1234'] """
|
|
|
cmdname = cmd.split(None,1)[0]
|
|
|
if cmdname in shell_internal_commands:
|
|
|
use_shell = True
|
|
|
else:
|
|
|
use_shell = False
|
|
|
|
|
|
p = Popen(cmd,shell = use_shell)
|
|
|
jobentry = 'tasks/t' + str(p.pid)
|
|
|
|
|
|
try:
|
|
|
ip.db[jobentry] = (p.pid,cmd,os.getcwd(),time.time())
|
|
|
p.communicate()
|
|
|
finally:
|
|
|
del ip.db[jobentry]
|
|
|
|
|
|
|
|
|
def install():
|
|
|
global ip
|
|
|
ip = IPython.ipapi.get()
|
|
|
# needed to make startjob visible as _ip.startjob('blah')
|
|
|
ip.startjob = startjob
|
|
|
ip.set_hook('input_prefilter', jobctrl_prefilter_f)
|
|
|
ip.set_hook('shell_hook', jobctrl_shellcmd)
|
|
|
ip.expose_magic('kill',magic_kill)
|
|
|
ip.expose_magic('tasks',magic_tasks)
|
|
|
|
|
|
install()
|
|
|
|