demo.py
178 lines
| 6.5 KiB
| text/x-python
|
PythonLexer
/ IPython / demo.py
fperez
|
r22 | """Module for interactive demos using IPython. | ||
fperez
|
r23 | |||
Sorry, but this uses Python 2.3 features, so it won't work in 2.2 environments. | ||||
fperez
|
r22 | """ | ||
#***************************************************************************** | ||||
# Copyright (C) 2005 Fernando Perez. <Fernando.Perez@colorado.edu> | ||||
# | ||||
# Distributed under the terms of the BSD License. The full license is in | ||||
# the file COPYING, distributed as part of this software. | ||||
# | ||||
#***************************************************************************** | ||||
fperez
|
r24 | import sys | ||
fperez
|
r22 | import exceptions | ||
fperez
|
r23 | import re | ||
fperez
|
r22 | |||
from IPython.PyColorize import Parser | ||||
fperez
|
r24 | from IPython.genutils import marquee, shlex_split | ||
fperez
|
r22 | |||
class DemoError(exceptions.Exception): pass | ||||
class Demo: | ||||
fperez
|
r24 | def __init__(self,fname,arg_str='',mark_pause='# pause', | ||
mark_silent='# silent',auto=False): | ||||
"""Make a new demo object. To run the demo, simply call the object. | ||||
Inputs: | ||||
- fname = filename. | ||||
Optional inputs: | ||||
- arg_str(''): a string of arguments, internally converted to a list | ||||
just like sys.argv, so the demo script can see a similar | ||||
environment. | ||||
- mark_pause ('# pause'), mark_silent('# silent'): marks for pausing | ||||
(block boundaries) and to tag blocks as silent. The marks are | ||||
turned into regexps which match them as standalone in a line, with | ||||
all leading/trailing whitespace ignored. | ||||
- auto(False): flag to run each block automatically without | ||||
confirmation. Note that silent blocks are always automatically | ||||
executed. This flag is an attribute of the object, and can be | ||||
changed at runtime simply by reassigning it. | ||||
""" | ||||
fperez
|
r22 | |||
self.fname = fname | ||||
fperez
|
r23 | self.mark_pause = mark_pause | ||
self.re_pause = re.compile(r'^\s*%s\s*$' % mark_pause,re.MULTILINE) | ||||
self.mark_silent = mark_silent | ||||
self.re_silent = re.compile(r'^\s*%s\s*$' % mark_silent,re.MULTILINE) | ||||
fperez
|
r22 | self.auto = auto | ||
fperez
|
r25 | self.sys_argv = [fname]+shlex_split(arg_str) | ||
fperez
|
r22 | |||
# get a few things from ipython. While it's a bit ugly design-wise, | ||||
# it ensures that things like color scheme and the like are always in | ||||
# sync with the ipython mode being used. This class is only meant to | ||||
# be used inside ipython anyways, so it's OK. | ||||
self.ip_showtraceback = __IPYTHON__.showtraceback | ||||
self.ip_ns = __IPYTHON__.user_ns | ||||
self.ip_colors = __IPYTHON__.rc['colors'] | ||||
# read data and parse into blocks | ||||
fobj = file(fname,'r') | ||||
self.src = fobj.read() | ||||
fobj.close() | ||||
fperez
|
r23 | self.src_blocks = [b.strip() for b in self.re_pause.split(self.src) if b] | ||
self.silent = [bool(self.re_silent.findall(b)) for b in self.src_blocks] | ||||
fperez
|
r22 | self.nblocks = len(self.src_blocks) | ||
# try to colorize blocks | ||||
colorize = Parser().format | ||||
col_scheme = self.ip_colors | ||||
self.src_blocks_colored = [colorize(s_blk,'str',col_scheme) | ||||
for s_blk in self.src_blocks] | ||||
# finish initialization | ||||
self.reset() | ||||
def reset(self): | ||||
fperez
|
r23 | """Reset the namespace and seek pointer to restart the demo""" | ||
fperez
|
r22 | self.user_ns = {} | ||
self.finished = False | ||||
self.block_index = 0 | ||||
def again(self): | ||||
fperez
|
r23 | """Repeat the last block""" | ||
fperez
|
r22 | self.block_index -= 1 | ||
fperez
|
r26 | self.finished = False | ||
fperez
|
r22 | self() | ||
def _validate_index(self,index): | ||||
if index<0 or index>=self.nblocks: | ||||
raise ValueError('invalid block index %s' % index) | ||||
def seek(self,index): | ||||
fperez
|
r23 | """Move the current seek pointer to the given block""" | ||
fperez
|
r22 | self._validate_index(index) | ||
self.block_index = index-1 | ||||
self.finished = False | ||||
fperez
|
r23 | def show_block(self,index=None): | ||
"""Show a single block on screen""" | ||||
fperez
|
r22 | if index is None: | ||
fperez
|
r23 | if self.finished: | ||
print 'Demo finished. Use reset() if you want to rerun it.' | ||||
return | ||||
fperez
|
r22 | index = self.block_index | ||
else: | ||||
self._validate_index(index) | ||||
fperez
|
r26 | print marquee('<%s> block # %s (%s remaining)' % | ||
(self.fname,index,self.nblocks-index-1)) | ||||
fperez
|
r22 | print self.src_blocks_colored[index], | ||
fperez
|
r23 | |||
def show(self): | ||||
"""Show entire demo on screen, block by block""" | ||||
fname = self.fname | ||||
nblocks = self.nblocks | ||||
silent = self.silent | ||||
for index,block in enumerate(self.src_blocks_colored): | ||||
if silent[index]: | ||||
fperez
|
r26 | print marquee('<%s> SILENT block # %s (%s remaining)' % | ||
(fname,index,nblocks-index-1)) | ||||
fperez
|
r23 | else: | ||
fperez
|
r26 | print marquee('<%s> block # %s (%s remaining)' % | ||
(fname,index,nblocks-index-1)) | ||||
fperez
|
r23 | print block, | ||
fperez
|
r22 | def __call__(self,index=None): | ||
"""run a block of the demo. | ||||
If index is given, it should be an integer >=1 and <= nblocks. This | ||||
means that the calling convention is one off from typical Python | ||||
lists. The reason for the inconsistency is that the demo always | ||||
prints 'Block n/N, and N is the total, so it would be very odd to use | ||||
zero-indexing here.""" | ||||
if index is None and self.finished: | ||||
print 'Demo finished. Use reset() if you want to rerun it.' | ||||
return | ||||
if index is None: | ||||
index = self.block_index | ||||
self._validate_index(index) | ||||
try: | ||||
next_block = self.src_blocks[index] | ||||
self.block_index += 1 | ||||
fperez
|
r23 | if self.silent[index]: | ||
fperez
|
r26 | print marquee('Executing silent block # %s (%s remaining)' % | ||
(index,self.nblocks-index-1)) | ||||
fperez
|
r23 | else: | ||
self.show_block(index) | ||||
fperez
|
r26 | if self.auto: | ||
print marquee('output') | ||||
else: | ||||
fperez
|
r23 | print marquee('Press <q> to quit, <Enter> to execute...'), | ||
ans = raw_input().strip() | ||||
if ans: | ||||
print marquee('Block NOT executed') | ||||
return | ||||
fperez
|
r24 | try: | ||
save_argv = sys.argv | ||||
sys.argv = self.sys_argv | ||||
exec next_block in self.user_ns | ||||
finally: | ||||
sys.argv = save_argv | ||||
fperez
|
r22 | |||
except: | ||||
self.ip_showtraceback(filename=self.fname) | ||||
else: | ||||
self.ip_ns.update(self.user_ns) | ||||
if self.block_index == self.nblocks: | ||||
print marquee(' END OF DEMO ') | ||||
print marquee('Use reset() if you want to rerun it.') | ||||
self.finished = True | ||||