blob: 9a1273cbc82b50a192c874e4654949a6ad4b30f7 [file] [log] [blame]
Guido van Rossumb5903ac1998-04-09 20:37:16 +00001"""Utilities to get a password and/or the current user name.
2
Gregory P. Smith19b44112008-04-22 08:08:41 +00003getpass(prompt[, stream]) - Prompt for a password, with echo turned off.
4getuser() - Get the user name from the environment or password database.
5
6GetPassWarning - This UserWarning is issued when getpass() cannot prevent
7 echoing of the password contents while reading.
Guido van Rossumb5903ac1998-04-09 20:37:16 +00008
Jeremy Hylton88d23301999-10-18 22:25:22 +00009On Windows, the msvcrt module will be used.
10On the Mac EasyDialogs.AskPassword is used, if available.
11
Guido van Rossumb5903ac1998-04-09 20:37:16 +000012"""
13
Guido van Rossum98d9fd32000-02-28 15:12:25 +000014# Authors: Piers Lauder (original)
15# Guido van Rossum (Windows support and cleanup)
Gregory P. Smith19b44112008-04-22 08:08:41 +000016# Gregory P. Smith (tty support & GetPassWarning)
Guido van Rossum98d9fd32000-02-28 15:12:25 +000017
Gregory P. Smith19b44112008-04-22 08:08:41 +000018import os, sys, warnings
Guido van Rossumb5903ac1998-04-09 20:37:16 +000019
Gregory P. Smith19b44112008-04-22 08:08:41 +000020__all__ = ["getpass","getuser","GetPassWarning"]
21
22
23class GetPassWarning(UserWarning): pass
24
Skip Montanaroeccd02a2001-01-20 23:34:12 +000025
Georg Brandl338ef7d2006-03-31 18:42:16 +000026def unix_getpass(prompt='Password: ', stream=None):
Tim Peters07e99cb2001-01-14 23:47:14 +000027 """Prompt for a password, with echo turned off.
Guido van Rossumb5903ac1998-04-09 20:37:16 +000028
Gregory P. Smith19b44112008-04-22 08:08:41 +000029 Args:
30 prompt: Written on stream to ask for the input. Default: 'Password: '
31 stream: A writable file object to display the prompt. Defaults to
32 the tty. If no tty is available defaults to sys.stderr.
33 Returns:
34 The seKr3t input.
35 Raises:
36 EOFError: If our input tty or stdin was closed.
37 GetPassWarning: When we were unable to turn echo off on the input.
38
39 Always restores terminal settings before returning.
Tim Peters07e99cb2001-01-14 23:47:14 +000040 """
Gregory P. Smith19b44112008-04-22 08:08:41 +000041 fd = None
42 tty = None
Tim Peters07e99cb2001-01-14 23:47:14 +000043 try:
Gregory P. Smith19b44112008-04-22 08:08:41 +000044 # Always try reading and writing directly on the tty first.
45 fd = os.open('/dev/tty', os.O_RDWR|os.O_NOCTTY)
46 tty = os.fdopen(fd, 'w+', 1)
47 input = tty
48 if not stream:
49 stream = tty
50 except EnvironmentError, e:
51 # If that fails, see if stdin can be controlled.
52 try:
53 fd = sys.stdin.fileno()
Benjamin Peterson4d714cb2009-09-17 02:46:54 +000054 except (AttributeError, ValueError):
Gregory P. Smith19b44112008-04-22 08:08:41 +000055 passwd = fallback_getpass(prompt, stream)
56 input = sys.stdin
57 if not stream:
58 stream = sys.stderr
Guido van Rossumb5903ac1998-04-09 20:37:16 +000059
Gregory P. Smith19b44112008-04-22 08:08:41 +000060 if fd is not None:
61 passwd = None
62 try:
63 old = termios.tcgetattr(fd) # a copy to save
64 new = old[:]
65 new[3] &= ~termios.ECHO # 3 == 'lflags'
66 try:
67 termios.tcsetattr(fd, termios.TCSADRAIN, new)
68 passwd = _raw_input(prompt, stream, input=input)
69 finally:
70 termios.tcsetattr(fd, termios.TCSADRAIN, old)
71 except termios.error, e:
72 if passwd is not None:
73 # _raw_input succeeded. The final tcsetattr failed. Reraise
74 # instead of leaving the terminal in an unknown state.
75 raise
76 # We can't control the tty or stdin. Give up and use normal IO.
77 # fallback_getpass() raises an appropriate warning.
78 del input, tty # clean up unused file objects before blocking
79 passwd = fallback_getpass(prompt, stream)
Guido van Rossumb5903ac1998-04-09 20:37:16 +000080
Georg Brandl338ef7d2006-03-31 18:42:16 +000081 stream.write('\n')
Tim Peters07e99cb2001-01-14 23:47:14 +000082 return passwd
Guido van Rossumb5903ac1998-04-09 20:37:16 +000083
84
Georg Brandl338ef7d2006-03-31 18:42:16 +000085def win_getpass(prompt='Password: ', stream=None):
Tim Peters07e99cb2001-01-14 23:47:14 +000086 """Prompt for password with echo off, using Windows getch()."""
Guido van Rossum60250e22001-08-30 15:07:44 +000087 if sys.stdin is not sys.__stdin__:
Gregory P. Smith19b44112008-04-22 08:08:41 +000088 return fallback_getpass(prompt, stream)
Tim Peters07e99cb2001-01-14 23:47:14 +000089 import msvcrt
90 for c in prompt:
91 msvcrt.putch(c)
92 pw = ""
93 while 1:
94 c = msvcrt.getch()
95 if c == '\r' or c == '\n':
96 break
97 if c == '\003':
98 raise KeyboardInterrupt
99 if c == '\b':
100 pw = pw[:-1]
101 else:
102 pw = pw + c
103 msvcrt.putch('\r')
104 msvcrt.putch('\n')
105 return pw
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000106
107
Gregory P. Smith19b44112008-04-22 08:08:41 +0000108def fallback_getpass(prompt='Password: ', stream=None):
109 warnings.warn("Can not control echo on the terminal.", GetPassWarning,
110 stacklevel=2)
111 if not stream:
112 stream = sys.stderr
113 print >>stream, "Warning: Password input may be echoed."
Georg Brandl338ef7d2006-03-31 18:42:16 +0000114 return _raw_input(prompt, stream)
Guido van Rossum1a7bab01998-07-28 19:28:43 +0000115
116
Gregory P. Smith19b44112008-04-22 08:08:41 +0000117def _raw_input(prompt="", stream=None, input=None):
Tim Peters07e99cb2001-01-14 23:47:14 +0000118 # A raw_input() replacement that doesn't save the string in the
119 # GNU readline history.
Gregory P. Smith19b44112008-04-22 08:08:41 +0000120 if not stream:
121 stream = sys.stderr
122 if not input:
123 input = sys.stdin
Tim Peters07e99cb2001-01-14 23:47:14 +0000124 prompt = str(prompt)
125 if prompt:
Georg Brandl338ef7d2006-03-31 18:42:16 +0000126 stream.write(prompt)
Gregory P. Smith19b44112008-04-22 08:08:41 +0000127 stream.flush()
128 line = input.readline()
Tim Peters07e99cb2001-01-14 23:47:14 +0000129 if not line:
130 raise EOFError
131 if line[-1] == '\n':
132 line = line[:-1]
133 return line
Guido van Rossumfb9b7fd1998-04-13 20:22:21 +0000134
135
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000136def getuser():
Tim Peters07e99cb2001-01-14 23:47:14 +0000137 """Get the username from the environment or password database.
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000138
Tim Peters07e99cb2001-01-14 23:47:14 +0000139 First try various environment variables, then the password
140 database. This works on Windows as long as USERNAME is set.
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000141
Tim Peters07e99cb2001-01-14 23:47:14 +0000142 """
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000143
Tim Peters07e99cb2001-01-14 23:47:14 +0000144 import os
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000145
Tim Peters07e99cb2001-01-14 23:47:14 +0000146 for name in ('LOGNAME', 'USER', 'LNAME', 'USERNAME'):
147 user = os.environ.get(name)
148 if user:
149 return user
Guido van Rossumb5903ac1998-04-09 20:37:16 +0000150
Tim Peters07e99cb2001-01-14 23:47:14 +0000151 # If this fails, the exception will "explain" why
152 import pwd
153 return pwd.getpwuid(os.getuid())[0]
Jeremy Hylton88d23301999-10-18 22:25:22 +0000154
155# Bind the name getpass to the appropriate function
156try:
Fred Drake1191d012001-02-27 21:23:31 +0000157 import termios
Neal Norwitz201626e2002-11-20 23:15:54 +0000158 # it's possible there is an incompatible termios from the
159 # McMillan Installer, make sure we have a UNIX-compatible termios
160 termios.tcgetattr, termios.tcsetattr
161except (ImportError, AttributeError):
Tim Peters07e99cb2001-01-14 23:47:14 +0000162 try:
163 import msvcrt
164 except ImportError:
165 try:
166 from EasyDialogs import AskPassword
167 except ImportError:
Gregory P. Smith19b44112008-04-22 08:08:41 +0000168 getpass = fallback_getpass
Tim Peters07e99cb2001-01-14 23:47:14 +0000169 else:
170 getpass = AskPassword
171 else:
172 getpass = win_getpass
Jeremy Hylton88d23301999-10-18 22:25:22 +0000173else:
Tim Peters07e99cb2001-01-14 23:47:14 +0000174 getpass = unix_getpass