| 
									
										
										
										
											2016-03-07 21:19:35 +00:00
										 |  |  | use std::io::{Read, Write};
 | 
					
						
							| 
									
										
										
										
											2016-03-08 07:51:34 +00:00
										 |  |  | use {IntoRawMode, TerminalError};
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-09 08:39:22 +00:00
										 |  |  | #[cfg(feature = "nightly")]
 | 
					
						
							|  |  |  | use std::io::Chars;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// A key.
 | 
					
						
							|  |  |  | #[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
 | 
					
						
							|  |  |  | pub enum Key {
 | 
					
						
							|  |  |  |     /// Backspace.
 | 
					
						
							|  |  |  |     Backspace,
 | 
					
						
							|  |  |  |     /// Left arrow.
 | 
					
						
							|  |  |  |     Left,
 | 
					
						
							|  |  |  |     /// Right arrow.
 | 
					
						
							|  |  |  |     Right,
 | 
					
						
							|  |  |  |     /// Up arrow.
 | 
					
						
							|  |  |  |     Up,
 | 
					
						
							|  |  |  |     /// Down arrow.
 | 
					
						
							|  |  |  |     Down,
 | 
					
						
							|  |  |  |     /// Alt modified character.
 | 
					
						
							|  |  |  |     Alt(char),
 | 
					
						
							|  |  |  |     /// Normal character.
 | 
					
						
							|  |  |  |     Char(char),
 | 
					
						
							|  |  |  |     /// Invalid character code.
 | 
					
						
							|  |  |  |     Invalid,
 | 
					
						
							|  |  |  |     // TODO handle errors better?
 | 
					
						
							|  |  |  |     /// IO error.
 | 
					
						
							|  |  |  |     Error,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[cfg(feature = "nightly")]
 | 
					
						
							|  |  |  | /// An iterator over input keys.
 | 
					
						
							|  |  |  | pub struct Keys<R> {
 | 
					
						
							|  |  |  |     chars: Chars<R>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[cfg(feature = "nightly")]
 | 
					
						
							|  |  |  | impl<R: Read> Iterator for Keys<R> {
 | 
					
						
							|  |  |  |     type Item = Key;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn next(&mut self) -> Option<Key> {
 | 
					
						
							|  |  |  |         match self.chars.next() {
 | 
					
						
							|  |  |  |             Some(Ok('\x1B')) => Some(match self.chars.next() {
 | 
					
						
							|  |  |  |                 Some(Ok('[')) => match self.chars.next() {
 | 
					
						
							|  |  |  |                     Some(Ok('D')) => Key::Left,
 | 
					
						
							|  |  |  |                     Some(Ok('C')) => Key::Right,
 | 
					
						
							|  |  |  |                     Some(Ok('A')) => Key::Up,
 | 
					
						
							|  |  |  |                     Some(Ok('B')) => Key::Down,
 | 
					
						
							|  |  |  |                     _ => Key::Invalid,
 | 
					
						
							|  |  |  |                 },
 | 
					
						
							|  |  |  |                 Some(Ok(c)) => Key::Alt(c),
 | 
					
						
							|  |  |  |                 Some(Err(_)) | None => Key::Invalid,
 | 
					
						
							|  |  |  |             }),
 | 
					
						
							|  |  |  |             Some(Ok('\x7F')) => Some(Key::Backspace),
 | 
					
						
							|  |  |  |             Some(Ok(c)) => Some(Key::Char(c)),
 | 
					
						
							|  |  |  |             None => None,
 | 
					
						
							|  |  |  |             Some(Err(_)) => Some(Key::Error),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  | /// Extension to `Read` trait.
 | 
					
						
							| 
									
										
										
										
											2016-03-08 20:39:24 +00:00
										 |  |  | pub trait TermRead {
 | 
					
						
							| 
									
										
										
										
											2016-03-09 08:39:22 +00:00
										 |  |  |     /// An iterator over key inputs.
 | 
					
						
							|  |  |  |     #[cfg(feature = "nightly")]
 | 
					
						
							|  |  |  |     fn keys(self) -> Keys<Self> where Self: Sized;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  |     /// Read a password.
 | 
					
						
							| 
									
										
										
										
											2016-03-08 09:08:50 +00:00
										 |  |  |     ///
 | 
					
						
							|  |  |  |     /// EOT and ETX will abort the prompt, returning `None`. Newline or carriage return will
 | 
					
						
							|  |  |  |     /// complete the password input.
 | 
					
						
							| 
									
										
										
										
											2016-03-08 07:51:34 +00:00
										 |  |  |     fn read_passwd<W: Write>(&mut self, writer: &mut W) -> Result<Option<String>, TerminalError>;
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 20:39:24 +00:00
										 |  |  | impl<R: Read> TermRead for R {
 | 
					
						
							| 
									
										
										
										
											2016-03-09 08:39:22 +00:00
										 |  |  |     #[cfg(feature = "nightly")]
 | 
					
						
							|  |  |  |     fn keys(self) -> Keys<R> {
 | 
					
						
							|  |  |  |         Keys {
 | 
					
						
							|  |  |  |             chars: self.chars(),
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 07:51:34 +00:00
										 |  |  |     fn read_passwd<W: Write>(&mut self, writer: &mut W) -> Result<Option<String>, TerminalError> {
 | 
					
						
							|  |  |  |         let _raw = try!(writer.into_raw_mode());
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:29:16 +00:00
										 |  |  |         let mut passbuf = Vec::with_capacity(30);
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:04:09 +00:00
										 |  |  |         for c in self.bytes() {
 | 
					
						
							| 
									
										
										
										
											2016-03-08 07:51:34 +00:00
										 |  |  |             match c {
 | 
					
						
							|  |  |  |                 Err(_) => return Err(TerminalError::StdinError),
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:29:22 +00:00
										 |  |  |                 Ok(0) | Ok(3) | Ok(4) => return Ok(None),
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:29:16 +00:00
										 |  |  |                 Ok(b'\n') | Ok(b'\r') => break,
 | 
					
						
							|  |  |  |                 Ok(c) => passbuf.push(c),
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:58:48 +00:00
										 |  |  |         let passwd = try!(String::from_utf8(passbuf).map_err(|_| TerminalError::UnicodeError));
 | 
					
						
							| 
									
										
										
										
											2016-03-08 10:29:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Ok(Some(passwd))
 | 
					
						
							| 
									
										
										
										
											2016-03-07 17:42:11 +00:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 |