##// END OF EJS Templates
codeutil into zmq, to prevent IPython.kernel import
codeutil into zmq, to prevent IPython.kernel import

File last commit:

r3166:147b245d
r3557:f2c5960c
Show More
page.py
327 lines | 11.2 KiB | text/x-python | PythonLexer
Brian Granger
Continuing a massive refactor of everything.
r2205 #!/usr/bin/env python
# encoding: utf-8
"""
Paging capabilities for IPython.core
Authors:
* Brian Granger
* Fernando Perez
Notes
-----
For now this uses ipapi, so it can't be in IPython.utils. If we can get
rid of that dependency, we could move it there.
-----
"""
#-----------------------------------------------------------------------------
# Copyright (C) 2008-2009 The IPython Development Team
#
# Distributed under the terms of the BSD License. The full license is in
# the file COPYING, distributed as part of this software.
#-----------------------------------------------------------------------------
#-----------------------------------------------------------------------------
# Imports
#-----------------------------------------------------------------------------
import os
import re
import sys
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 import tempfile
Brian Granger
Continuing a massive refactor of everything.
r2205
from IPython.core import ipapi
from IPython.core.error import TryNext
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 from IPython.utils.cursesimport import use_curses
from IPython.utils.data import chop
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 import IPython.utils.io
Fernando Perez
Fully refactored subprocess handling on all platforms....
r2908 from IPython.utils.process import system
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 from IPython.utils.terminal import get_terminal_size
Brian Granger
Continuing a massive refactor of everything.
r2205
#-----------------------------------------------------------------------------
# Classes and functions
#-----------------------------------------------------------------------------
esc_re = re.compile(r"(\x1b[^m]+m)")
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 def page_dumb(strng, start=0, screen_lines=25):
Brian Granger
Continuing a massive refactor of everything.
r2205 """Very dumb 'pager' in Python, for when nothing else works.
Only moves forward, same interface as page(), except for pager_cmd and
mode."""
out_ln = strng.splitlines()[start:]
screens = chop(out_ln,screen_lines-1)
if len(screens) == 1:
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 print >>IPython.utils.io.Term.cout, os.linesep.join(screens[0])
Brian Granger
Continuing a massive refactor of everything.
r2205 else:
last_escape = ""
for scr in screens[0:-1]:
hunk = os.linesep.join(scr)
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 print >>IPython.utils.io.Term.cout, last_escape + hunk
Brian Granger
Continuing a massive refactor of everything.
r2205 if not page_more():
return
esc_list = esc_re.findall(hunk)
if len(esc_list) > 0:
last_escape = esc_list[-1]
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 print >>IPython.utils.io.Term.cout, last_escape + os.linesep.join(screens[-1])
Brian Granger
Continuing a massive refactor of everything.
r2205
Brian Granger
Work to address the review comments on Fernando's branch....
r2498
def page(strng, start=0, screen_lines=0, pager_cmd=None):
Brian Granger
Continuing a massive refactor of everything.
r2205 """Print a string, piping through a pager after a certain length.
The screen_lines parameter specifies the number of *usable* lines of your
terminal screen (total lines minus lines you need to reserve to show other
information).
If you set screen_lines to a number <=0, page() will try to auto-determine
your screen size and will only use up to (screen_size+screen_lines) for
printing, paging after that. That is, if you want auto-detection but need
to reserve the bottom 3 lines of the screen, use screen_lines = -3, and for
auto-detection without any lines reserved simply use screen_lines = 0.
If a string won't fit in the allowed lines, it is sent through the
specified pager command. If none given, look for PAGER in the environment,
and ultimately default to less.
If no system pager works, the string is sent through a 'dumb pager'
written in python, very simplistic.
"""
# Some routines may auto-compute start offsets incorrectly and pass a
# negative value. Offset to 0 for robustness.
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 start = max(0, start)
Brian Granger
Continuing a massive refactor of everything.
r2205
# first, try the hook
ip = ipapi.get()
if ip:
try:
ip.hooks.show_in_pager(strng)
return
except TryNext:
pass
# Ugly kludge, but calling curses.initscr() flat out crashes in emacs
TERM = os.environ.get('TERM','dumb')
if TERM in ['dumb','emacs'] and os.name != 'nt':
print strng
return
# chop off the topmost part of the string we don't want to see
str_lines = strng.split(os.linesep)[start:]
str_toprint = os.linesep.join(str_lines)
num_newlines = len(str_lines)
len_str = len(str_toprint)
# Dumb heuristics to guesstimate number of on-screen lines the string
# takes. Very basic, but good enough for docstrings in reasonable
# terminals. If someone later feels like refining it, it's not hard.
numlines = max(num_newlines,int(len_str/80)+1)
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 screen_lines_def = get_terminal_size()[1]
Brian Granger
Continuing a massive refactor of everything.
r2205
# auto-determine screen size
if screen_lines <= 0:
Fernando Perez
Avoid lockups with ? or ?? in SunOS, due to a bug in termios....
r2584 if (TERM=='xterm' or TERM=='xterm-color') and sys.platform != 'sunos5':
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 local_use_curses = use_curses
Brian Granger
Continuing a massive refactor of everything.
r2205 else:
Fernando Perez
Avoid lockups with ? or ?? in SunOS, due to a bug in termios....
r2584 # curses causes problems on many terminals other than xterm, and
# some termios calls lock up on Sun OS5.
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 local_use_curses = False
if local_use_curses:
Brian Granger
Work on startup related things....
r2252 import termios
import curses
Brian Granger
Continuing a massive refactor of everything.
r2205 # There is a bug in curses, where *sometimes* it fails to properly
# initialize, and then after the endwin() call is made, the
# terminal is left in an unusable state. Rather than trying to
# check everytime for this (by requesting and comparing termios
# flags each time), we just save the initial terminal state and
# unconditionally reset it every time. It's cheaper than making
# the checks.
term_flags = termios.tcgetattr(sys.stdout)
Fernando Perez
Work around issue induced by curses with print buffering....
r3166
# Curses modifies the stdout buffer size by default, which messes
# up Python's normal stdout buffering. This would manifest itself
# to IPython users as delayed printing on stdout after having used
# the pager.
#
# We can prevent this by manually setting the NCURSES_NO_SETBUF
# environment variable. For more details, see:
# http://bugs.python.org/issue10144
NCURSES_NO_SETBUF = os.environ.get('NCURSES_NO_SETBUF', None)
os.environ['NCURSES_NO_SETBUF'] = ''
# Proceed with curses initialization
Brian Granger
Continuing a massive refactor of everything.
r2205 scr = curses.initscr()
screen_lines_real,screen_cols = scr.getmaxyx()
curses.endwin()
Fernando Perez
Work around issue induced by curses with print buffering....
r3166
# Restore environment
if NCURSES_NO_SETBUF is None:
del os.environ['NCURSES_NO_SETBUF']
else:
os.environ['NCURSES_NO_SETBUF'] = NCURSES_NO_SETBUF
Brian Granger
Continuing a massive refactor of everything.
r2205 # Restore terminal state in case endwin() didn't.
termios.tcsetattr(sys.stdout,termios.TCSANOW,term_flags)
# Now we have what we needed: the screen size in rows/columns
screen_lines += screen_lines_real
#print '***Screen size:',screen_lines_real,'lines x',\
#screen_cols,'columns.' # dbg
else:
screen_lines += screen_lines_def
#print 'numlines',numlines,'screenlines',screen_lines # dbg
if numlines <= screen_lines :
#print '*** normal print' # dbg
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 print >>IPython.utils.io.Term.cout, str_toprint
Brian Granger
Continuing a massive refactor of everything.
r2205 else:
# Try to open pager and default to internal one if that fails.
# All failure modes are tagged as 'retval=1', to match the return
# value of a failed system command. If any intermediate attempt
# sets retval to 1, at the end we resort to our own page_dumb() pager.
pager_cmd = get_pager_cmd(pager_cmd)
pager_cmd += ' ' + get_pager_start(pager_cmd,start)
if os.name == 'nt':
if pager_cmd.startswith('type'):
# The default WinXP 'type' command is failing on complex strings.
retval = 1
else:
tmpname = tempfile.mktemp('.txt')
tmpfile = file(tmpname,'wt')
tmpfile.write(strng)
tmpfile.close()
cmd = "%s < %s" % (pager_cmd,tmpname)
if os.system(cmd):
retval = 1
else:
retval = None
os.remove(tmpname)
else:
try:
retval = None
# if I use popen4, things hang. No idea why.
#pager,shell_out = os.popen4(pager_cmd)
pager = os.popen(pager_cmd,'w')
pager.write(strng)
pager.close()
retval = pager.close() # success returns None
except IOError,msg: # broken pipe when user quits
if msg.args == (32,'Broken pipe'):
retval = None
else:
retval = 1
except OSError:
# Other strange problems, sometimes seen in Win2k/cygwin
retval = 1
if retval is not None:
page_dumb(strng,screen_lines=screen_lines)
Brian Granger
Work to address the review comments on Fernando's branch....
r2498
def page_file(fname, start=0, pager_cmd=None):
Brian Granger
Continuing a massive refactor of everything.
r2205 """Page a file, using an optional pager command and starting line.
"""
pager_cmd = get_pager_cmd(pager_cmd)
pager_cmd += ' ' + get_pager_start(pager_cmd,start)
try:
if os.environ['TERM'] in ['emacs','dumb']:
raise EnvironmentError
Fernando Perez
Fully refactored subprocess handling on all platforms....
r2908 system(pager_cmd + ' ' + fname)
Brian Granger
Continuing a massive refactor of everything.
r2205 except:
try:
if start > 0:
start -= 1
page(open(fname).read(),start)
except:
print 'Unable to show file',`fname`
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 def get_pager_cmd(pager_cmd=None):
"""Return a pager command.
Brian Granger
Continuing a massive refactor of everything.
r2205
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 Makes some attempts at finding an OS-correct one.
"""
Brian Granger
Continuing a massive refactor of everything.
r2205 if os.name == 'posix':
default_pager_cmd = 'less -r' # -r for color control sequences
elif os.name in ['nt','dos']:
default_pager_cmd = 'type'
if pager_cmd is None:
try:
pager_cmd = os.environ['PAGER']
except:
pager_cmd = default_pager_cmd
return pager_cmd
Brian Granger
Work to address the review comments on Fernando's branch....
r2498
def get_pager_start(pager, start):
Brian Granger
Continuing a massive refactor of everything.
r2205 """Return the string for paging files with an offset.
This is the '+N' argument which less and more (under Unix) accept.
"""
if pager in ['less','more']:
if start:
start_string = '+' + str(start)
else:
start_string = ''
else:
start_string = ''
return start_string
Brian Granger
Work to address the review comments on Fernando's branch....
r2498
# (X)emacs on win32 doesn't like to be bypassed with msvcrt.getch()
Brian Granger
Continuing a massive refactor of everything.
r2205 if os.name == 'nt' and os.environ.get('TERM','dumb') != 'emacs':
import msvcrt
def page_more():
""" Smart pausing between pages
@return: True if need print more lines, False if quit
"""
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 IPython.utils.io.Term.cout.write('---Return to continue, q to quit--- ')
Brian Granger
Continuing a massive refactor of everything.
r2205 ans = msvcrt.getch()
if ans in ("q", "Q"):
result = False
else:
result = True
Brian Granger
Changing how IPython.utils.io.Term is handled....
r2775 IPython.utils.io.Term.cout.write("\b"*37 + " "*37 + "\b"*37)
Brian Granger
Continuing a massive refactor of everything.
r2205 return result
else:
def page_more():
ans = raw_input('---Return to continue, q to quit--- ')
if ans.lower().startswith('q'):
return False
else:
return True
Brian Granger
Work to address the review comments on Fernando's branch....
r2498
Brian Granger
Continuing a massive refactor of everything.
r2205 def snip_print(str,width = 75,print_full = 0,header = ''):
"""Print a string snipping the midsection to fit in width.
print_full: mode control:
- 0: only snip long strings
- 1: send to page() directly.
- 2: snip long strings and ask for full length viewing with page()
Return 1 if snipping was necessary, 0 otherwise."""
if print_full == 1:
page(header+str)
return 0
print header,
if len(str) < width:
print str
snip = 0
else:
whalf = int((width -5)/2)
print str[:whalf] + ' <...> ' + str[-whalf:]
snip = 1
if snip and print_full == 2:
if raw_input(header+' Snipped. View (y/n)? [N]').lower() == 'y':
page(str)
Brian Granger
Work to address the review comments on Fernando's branch....
r2498 return snip