Moved VirtualMailManager/Exceptions to VirtualMailManager/errors.
Renamed VMM*Exception classes to *Error.
No longer add the attribute 'message' to VMMError if it doesn't exist, like in
Python 2.4. It has been deprecated as of Python 2.6.
Also removed the methods code() and msg(), the values are now accessible via
the attributes 'code' and 'msg'.
# -*- coding: UTF-8 -*-
# Copyright (c) 2010, Pascal Volk
# See COPYING for distribution information.
"""
VirtualMailManager.cli
VirtualMailManager's command line interface.
"""
import os
from cStringIO import StringIO
from getpass import getpass
from textwrap import TextWrapper
from VirtualMailManager import ENCODING
__all__ = ('get_winsize', 'read_pass', 'string_io', 'w_err', 'w_std')
_std_write = os.sys.stdout.write
_err_write = os.sys.stderr.write
def w_std(*args):
"""Writes each arg of `args`, encoded in the current ENCODING, to stdout
and appends a newline."""
for arg in args:
_std_write(arg.encode(ENCODING, 'replace'))
_std_write('\n')
def w_err(code, *args):
"""Writes each arg of `args`, encoded in the current ENCODING, to stderr
and appends a newline.
This function additional interrupts the program execution and uses
`code` system exit status."""
for arg in args:
_err_write(arg.encode(ENCODING, 'replace'))
_err_write('\n')
os.sys.exit(code)
def get_winsize():
"""Returns a tuple of integers ``(ws_row, ws_col)`` with the height and
width of the terminal."""
fd = None
for dev in (os.sys.stdout, os.sys.stderr, os.sys.stdin):
if hasattr(dev, 'fileno') and os.isatty(dev.fileno()):
fd = dev.fileno()
break
if fd is None:# everything seems to be redirected
# fall back to environment or assume some common defaults
ws_row, ws_col = 24, 80
try:
ws_col = int(os.environ.get('COLUMNS', 80))
ws_row = int(os.environ.get('LINES', 24))
except ValueError:
pass
return ws_row, ws_col
from array import array
from fcntl import ioctl
from termios import TIOCGWINSZ
#"struct winsize" with the ``unsigned short int``s ws_{row,col,{x,y}pixel}
ws = array('H', (0, 0, 0, 0))
ioctl(fd, TIOCGWINSZ, ws, True)
ws_row, ws_col = ws[:2]
return ws_row, ws_col
def read_pass():
"""Interactive 'password chat', returns the password in plain format.
Throws a VMMException after the third failure.
"""
# TP: Please preserve the trailing space.
readp_msg0 = _(u'Enter new password: ').encode(ENCODING, 'replace')
# TP: Please preserve the trailing space.
readp_msg1 = _(u'Retype new password: ').encode(ENCODING, 'replace')
mismatched = True
failures = 0
while mismatched:
if failures > 2:
raise VMMException(_(u'Too many failures - try again later.'),
ERR.VMM_TOO_MANY_FAILURES)
clear0 = getpass(prompt=readp_msg0)
clear1 = getpass(prompt=readp_msg1)
if clear0 != clear1:
failures += 1
w_std(_(u'Sorry, passwords do not match'))
continue
if not clear0:
failures += 1
w_std(_(u'Sorry, empty passwords are not permitted'))
continue
mismatched = False
return clear0
def string_io():
"""Returns a new `cStringIO.StringIO` instance."""
return StringIO()