2021-03-24 22:49:09 +01:00
|
|
|
extern crate rpassword;
|
2021-03-27 17:14:34 +01:00
|
|
|
use borrow::BorrowMut;
|
2021-07-30 01:42:45 +02:00
|
|
|
use collections::HashSet;
|
|
|
|
use io::{Write, stdout};
|
2021-07-27 22:24:56 +02:00
|
|
|
use steamapi::Session;
|
2021-03-22 02:21:29 +01:00
|
|
|
use steamguard_cli::*;
|
2021-07-30 01:42:45 +02:00
|
|
|
use termion::{color::Color, raw::IntoRawMode, screen::AlternateScreen};
|
2021-03-24 22:49:09 +01:00
|
|
|
use ::std::*;
|
|
|
|
use text_io::read;
|
2021-07-30 01:42:45 +02:00
|
|
|
use std::{convert::TryInto, io::stdin, path::Path, sync::Arc};
|
2021-03-27 14:31:38 +01:00
|
|
|
use clap::{App, Arg, crate_version};
|
2021-03-27 15:35:52 +01:00
|
|
|
use log::*;
|
2021-04-04 16:40:16 +02:00
|
|
|
use regex::Regex;
|
2021-07-30 01:42:45 +02:00
|
|
|
use termion::event::{Key, Event};
|
|
|
|
use termion::input::{TermRead};
|
2021-03-24 22:49:09 +01:00
|
|
|
|
2021-04-04 16:40:16 +02:00
|
|
|
#[macro_use]
|
|
|
|
extern crate lazy_static;
|
2021-03-26 00:47:44 +01:00
|
|
|
mod accountmanager;
|
2021-07-27 22:24:56 +02:00
|
|
|
mod accountlinker;
|
2021-03-22 02:21:29 +01:00
|
|
|
|
2021-04-04 16:40:16 +02:00
|
|
|
lazy_static! {
|
|
|
|
static ref CAPTCHA_VALID_CHARS: Regex = Regex::new("^([A-H]|[J-N]|[P-R]|[T-Z]|[2-4]|[7-9]|[@%&])+$").unwrap();
|
|
|
|
}
|
|
|
|
|
2021-03-22 02:21:29 +01:00
|
|
|
fn main() {
|
2021-03-27 14:31:38 +01:00
|
|
|
let matches = App::new("steamguard-cli")
|
|
|
|
.version(crate_version!())
|
|
|
|
.bin_name("steamguard")
|
|
|
|
.author("dyc3 (Carson McManus)")
|
|
|
|
.about("Generate Steam 2FA codes and confirm Steam trades from the command line.")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("username")
|
|
|
|
.long("username")
|
|
|
|
.short("u")
|
|
|
|
.help("Select the account you want by steam username. By default, the first account in the manifest is selected.")
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("all")
|
|
|
|
.long("all")
|
|
|
|
.short("a")
|
2021-03-27 17:14:34 +01:00
|
|
|
.takes_value(false)
|
2021-03-27 14:31:38 +01:00
|
|
|
.help("Select all accounts in the manifest.")
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("mafiles-path")
|
|
|
|
.long("mafiles-path")
|
|
|
|
.short("m")
|
|
|
|
.default_value("~/maFiles")
|
|
|
|
.help("Specify which folder your maFiles are in.")
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("passkey")
|
|
|
|
.long("passkey")
|
|
|
|
.short("p")
|
|
|
|
.help("Specify your encryption passkey.")
|
|
|
|
)
|
2021-03-27 15:35:52 +01:00
|
|
|
.arg(
|
|
|
|
Arg::with_name("verbosity")
|
|
|
|
.short("v")
|
|
|
|
.help("Log what is going on verbosely.")
|
|
|
|
.takes_value(false)
|
|
|
|
.multiple(true)
|
|
|
|
)
|
2021-03-27 14:31:38 +01:00
|
|
|
.subcommand(
|
|
|
|
App::new("trade")
|
|
|
|
.about("Interactive interface for trade confirmations")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("accept-all")
|
|
|
|
.short("a")
|
|
|
|
.long("accept-all")
|
2021-03-27 17:14:34 +01:00
|
|
|
.takes_value(false)
|
2021-03-27 14:31:38 +01:00
|
|
|
.help("Accept all open trade confirmations. Does not open interactive interface.")
|
|
|
|
)
|
|
|
|
)
|
2021-07-27 22:24:56 +02:00
|
|
|
.subcommand(
|
|
|
|
App::new("setup")
|
|
|
|
.about("Set up a new account with steamguard-cli")
|
|
|
|
)
|
2021-07-30 01:42:45 +02:00
|
|
|
.subcommand(
|
|
|
|
App::new("debug")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("demo-conf-menu")
|
|
|
|
.help("Show an example confirmation menu using dummy data.")
|
|
|
|
.takes_value(false)
|
|
|
|
)
|
|
|
|
)
|
2021-03-27 14:31:38 +01:00
|
|
|
.get_matches();
|
|
|
|
|
2021-03-22 02:21:29 +01:00
|
|
|
|
2021-03-27 15:35:52 +01:00
|
|
|
let verbosity = matches.occurrences_of("verbosity") as usize + 2;
|
|
|
|
stderrlog::new()
|
|
|
|
.verbosity(verbosity)
|
|
|
|
.module(module_path!()).init().unwrap();
|
2021-03-24 22:49:09 +01:00
|
|
|
|
2021-07-30 01:42:45 +02:00
|
|
|
if let Some(demo_matches) = matches.subcommand_matches("debug") {
|
|
|
|
if demo_matches.is_present("demo-conf-menu") {
|
|
|
|
demo_confirmation_menu();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-27 15:35:52 +01:00
|
|
|
let path = Path::new(matches.value_of("mafiles-path").unwrap()).join("manifest.json");
|
|
|
|
let mut manifest: accountmanager::Manifest;
|
|
|
|
match accountmanager::Manifest::load(path.as_path()) {
|
|
|
|
Ok(m) => {
|
|
|
|
manifest = m;
|
2021-03-27 13:39:26 +01:00
|
|
|
}
|
|
|
|
Err(e) => {
|
2021-03-27 15:35:52 +01:00
|
|
|
error!("Could not load manifest: {}", e);
|
|
|
|
return;
|
2021-03-27 13:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
2021-03-27 15:35:52 +01:00
|
|
|
|
|
|
|
manifest.load_accounts();
|
2021-07-27 22:24:56 +02:00
|
|
|
|
|
|
|
if matches.is_present("setup") {
|
|
|
|
info!("setup");
|
|
|
|
let mut linker = accountlinker::AccountLinker::new();
|
|
|
|
do_login(&mut linker.account);
|
|
|
|
// linker.link(linker.account.session.expect("no login session"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-27 17:14:34 +01:00
|
|
|
let mut selected_accounts: Vec<SteamGuardAccount> = vec![];
|
|
|
|
if matches.is_present("all") {
|
|
|
|
// manifest.accounts.iter().map(|a| selected_accounts.push(a.b));
|
|
|
|
for account in manifest.accounts {
|
|
|
|
selected_accounts.push(account.clone());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for account in manifest.accounts {
|
|
|
|
if !matches.is_present("username") {
|
|
|
|
selected_accounts.push(account.clone());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if matches.value_of("username").unwrap() == account.account_name {
|
|
|
|
selected_accounts.push(account.clone());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
debug!("selected accounts: {:?}", selected_accounts.iter().map(|a| a.account_name.clone()).collect::<Vec<String>>());
|
|
|
|
|
2021-07-28 21:48:06 +02:00
|
|
|
if let Some(trade_matches) = matches.subcommand_matches("trade") {
|
2021-03-30 21:51:26 +02:00
|
|
|
info!("trade");
|
2021-07-27 22:24:56 +02:00
|
|
|
for a in selected_accounts.iter_mut() {
|
|
|
|
let mut account = a; // why is this necessary?
|
2021-04-04 20:07:06 +02:00
|
|
|
|
|
|
|
info!("Checking for trade confirmations");
|
2021-07-29 15:08:06 +02:00
|
|
|
let confirmations: Vec<Confirmation>;
|
2021-07-28 01:50:39 +02:00
|
|
|
loop {
|
|
|
|
match account.get_trade_confirmations() {
|
|
|
|
Ok(confs) => {
|
2021-07-29 15:08:06 +02:00
|
|
|
confirmations = confs;
|
2021-07-28 01:50:39 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
info!("failed to get trade confirmations, asking user to log in");
|
|
|
|
do_login(&mut account);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-29 15:08:06 +02:00
|
|
|
|
|
|
|
if trade_matches.is_present("accept-all") {
|
|
|
|
info!("accepting all confirmations");
|
|
|
|
for conf in &confirmations {
|
|
|
|
let result = account.accept_confirmation(conf);
|
|
|
|
debug!("accept confirmation result: {:?}", result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2021-07-30 01:42:45 +02:00
|
|
|
if termion::is_tty(&stdout()) {
|
|
|
|
let (accept, deny) = prompt_confirmation_menu(confirmations);
|
|
|
|
for conf in &accept {
|
|
|
|
let result = account.accept_confirmation(conf);
|
|
|
|
debug!("accept confirmation result: {:?}", result);
|
|
|
|
}
|
|
|
|
for conf in &deny {
|
|
|
|
let result = account.deny_confirmation(conf);
|
|
|
|
debug!("deny confirmation result: {:?}", result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
warn!("not a tty, not showing menu");
|
|
|
|
for conf in &confirmations {
|
|
|
|
println!("{}", conf.description());
|
|
|
|
}
|
|
|
|
}
|
2021-07-29 15:08:06 +02:00
|
|
|
}
|
2021-03-30 21:51:26 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let server_time = steamapi::get_server_time();
|
|
|
|
for account in selected_accounts {
|
|
|
|
trace!("{:?}", account);
|
|
|
|
let code = account.generate_code(server_time);
|
|
|
|
println!("{}", code);
|
|
|
|
}
|
2021-03-27 15:35:52 +01:00
|
|
|
}
|
2021-03-22 02:21:29 +01:00
|
|
|
}
|
2021-04-04 16:40:16 +02:00
|
|
|
|
|
|
|
fn validate_captcha_text(text: &String) -> bool {
|
|
|
|
return CAPTCHA_VALID_CHARS.is_match(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_validate_captcha_text() {
|
|
|
|
assert!(validate_captcha_text(&String::from("2WWUA@")));
|
|
|
|
assert!(validate_captcha_text(&String::from("3G8HT2")));
|
|
|
|
assert!(validate_captcha_text(&String::from("3J%@X3")));
|
|
|
|
assert!(validate_captcha_text(&String::from("2GCZ4A")));
|
|
|
|
assert!(validate_captcha_text(&String::from("3G8HT2")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("asd823")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("!PQ4RD")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("1GQ4XZ")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("8GO4XZ")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("IPQ4RD")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("0PT4RD")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("APTSRD")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("AP5TRD")));
|
|
|
|
assert!(!validate_captcha_text(&String::from("AP6TRD")));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Prompt the user for text input.
|
|
|
|
fn prompt() -> String {
|
|
|
|
let mut text = String::new();
|
|
|
|
let _ = std::io::stdout().flush();
|
|
|
|
stdin().read_line(&mut text).expect("Did not enter a correct string");
|
|
|
|
return String::from(text.strip_suffix('\n').unwrap());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn prompt_captcha_text(captcha_gid: &String) -> String {
|
|
|
|
println!("Captcha required. Open this link in your web browser: https://steamcommunity.com/public/captcha.php?gid={}", captcha_gid);
|
|
|
|
let mut captcha_text;
|
|
|
|
loop {
|
|
|
|
print!("Enter captcha text: ");
|
|
|
|
captcha_text = prompt();
|
|
|
|
if captcha_text.len() > 0 && validate_captcha_text(&captcha_text) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
warn!("Invalid chars for captcha text found in user's input. Prompting again...");
|
|
|
|
}
|
|
|
|
return captcha_text;
|
|
|
|
}
|
2021-07-27 22:24:56 +02:00
|
|
|
|
2021-07-30 01:42:45 +02:00
|
|
|
/// Returns a tuple of (accepted, denied). Ignored confirmations are not included.
|
|
|
|
fn prompt_confirmation_menu(confirmations: Vec<Confirmation>) -> (Vec<Confirmation>, Vec<Confirmation>) {
|
|
|
|
println!("press a key other than enter to show the menu.");
|
|
|
|
let mut to_accept_idx: HashSet<usize> = HashSet::new();
|
|
|
|
let mut to_deny_idx: HashSet<usize> = HashSet::new();
|
|
|
|
|
|
|
|
let mut screen = AlternateScreen::from(stdout().into_raw_mode().unwrap());
|
|
|
|
let stdin = stdin();
|
|
|
|
|
|
|
|
let mut selected_idx = 0;
|
|
|
|
|
|
|
|
for c in stdin.events() {
|
|
|
|
match c.expect("could not get events") {
|
|
|
|
Event::Key(Key::Char('a')) => {
|
|
|
|
to_accept_idx.insert(selected_idx);
|
|
|
|
to_deny_idx.remove(&selected_idx);
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('d')) => {
|
|
|
|
to_accept_idx.remove(&selected_idx);
|
|
|
|
to_deny_idx.insert(selected_idx);
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('i')) => {
|
|
|
|
to_accept_idx.remove(&selected_idx);
|
|
|
|
to_deny_idx.remove(&selected_idx);
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('A')) => {
|
|
|
|
(0..confirmations.len()).for_each(|i| { to_accept_idx.insert(i); to_deny_idx.remove(&i); });
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('D')) => {
|
|
|
|
(0..confirmations.len()).for_each(|i| { to_accept_idx.remove(&i); to_deny_idx.insert(i); });
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('I')) => {
|
|
|
|
(0..confirmations.len()).for_each(|i| { to_accept_idx.remove(&i); to_deny_idx.remove(&i); });
|
|
|
|
}
|
|
|
|
Event::Key(Key::Up) if selected_idx > 0 => {
|
|
|
|
selected_idx -= 1;
|
|
|
|
}
|
|
|
|
Event::Key(Key::Down) if selected_idx < confirmations.len() - 1 => {
|
|
|
|
selected_idx += 1;
|
|
|
|
}
|
|
|
|
Event::Key(Key::Char('\n')) => {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Event::Key(Key::Esc) | Event::Key(Key::Ctrl('c')) => {
|
|
|
|
return (vec![], vec![]);
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
|
|
|
write!(screen, "{}{}{}arrow keys to select, [a]ccept, [d]eny, [i]gnore, [enter] confirm choices\n\n", termion::clear::All, termion::cursor::Goto(1, 1), termion::color::Fg(termion::color::White)).unwrap();
|
|
|
|
for i in 0..confirmations.len() {
|
|
|
|
if selected_idx == i {
|
|
|
|
write!(screen, "\r{} >", termion::color::Fg(termion::color::LightYellow)).unwrap();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
write!(screen, "\r{} ", termion::color::Fg(termion::color::White)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
if to_accept_idx.contains(&i) {
|
|
|
|
write!(screen, "{}[a]", termion::color::Fg(termion::color::LightGreen)).unwrap();
|
|
|
|
}
|
|
|
|
else if to_deny_idx.contains(&i) {
|
|
|
|
write!(screen, "{}[d]", termion::color::Fg(termion::color::LightRed)).unwrap();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
write!(screen, "[ ]").unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
if selected_idx == i {
|
|
|
|
write!(screen, "{}", termion::color::Fg(termion::color::LightYellow)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
write!(screen, " {}\n", confirmations[i].description()).unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
to_accept_idx.iter().map(|i| confirmations[*i]).collect(),
|
|
|
|
to_deny_idx.iter().map(|i| confirmations[*i]).collect(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-07-27 22:24:56 +02:00
|
|
|
fn do_login(account: &mut SteamGuardAccount) {
|
|
|
|
if account.account_name.len() > 0 {
|
|
|
|
println!("Username: {}", account.account_name);
|
|
|
|
} else {
|
|
|
|
print!("Username: ");
|
|
|
|
account.account_name = prompt();
|
|
|
|
}
|
|
|
|
let _ = std::io::stdout().flush();
|
|
|
|
let password = rpassword::prompt_password_stdout("Password: ").unwrap();
|
|
|
|
if password.len() > 0 {
|
|
|
|
debug!("password is present");
|
|
|
|
} else {
|
|
|
|
debug!("password is empty");
|
|
|
|
}
|
|
|
|
// TODO: reprompt if password is empty
|
|
|
|
let mut login = steamapi::UserLogin::new(account.account_name.clone(), password);
|
|
|
|
let mut loops = 0;
|
|
|
|
loop {
|
|
|
|
match login.login() {
|
|
|
|
steamapi::LoginResult::Ok(s) => {
|
|
|
|
account.session = Option::Some(s);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
steamapi::LoginResult::Need2FA => {
|
|
|
|
let server_time = steamapi::get_server_time();
|
|
|
|
login.twofactor_code = account.generate_code(server_time);
|
|
|
|
}
|
|
|
|
steamapi::LoginResult::NeedCaptcha{ captcha_gid } => {
|
|
|
|
login.captcha_text = prompt_captcha_text(&captcha_gid);
|
|
|
|
}
|
|
|
|
steamapi::LoginResult::NeedEmail => {
|
|
|
|
println!("You should have received an email with a code.");
|
|
|
|
print!("Enter code");
|
|
|
|
login.email_code = prompt();
|
|
|
|
}
|
|
|
|
r => {
|
|
|
|
error!("Fatal login result: {:?}", r);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
loops += 1;
|
|
|
|
if loops > 2 {
|
|
|
|
error!("Too many loops. Aborting login process, to avoid getting rate limited.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-30 01:42:45 +02:00
|
|
|
|
|
|
|
fn demo_confirmation_menu() {
|
|
|
|
info!("showing demo menu");
|
|
|
|
let (accept, deny) = prompt_confirmation_menu(vec![
|
|
|
|
Confirmation {
|
|
|
|
id: 1234,
|
|
|
|
key: 12345,
|
|
|
|
conf_type: ConfirmationType::Trade,
|
|
|
|
int_type: 0,
|
|
|
|
creator: 09870987,
|
|
|
|
},
|
|
|
|
Confirmation {
|
|
|
|
id: 1234,
|
|
|
|
key: 12345,
|
|
|
|
conf_type: ConfirmationType::MarketSell,
|
|
|
|
int_type: 0,
|
|
|
|
creator: 09870987,
|
|
|
|
},
|
|
|
|
Confirmation {
|
|
|
|
id: 1234,
|
|
|
|
key: 12345,
|
|
|
|
conf_type: ConfirmationType::AccountRecovery,
|
|
|
|
int_type: 0,
|
|
|
|
creator: 09870987,
|
|
|
|
},
|
|
|
|
Confirmation {
|
|
|
|
id: 1234,
|
|
|
|
key: 12345,
|
|
|
|
conf_type: ConfirmationType::Trade,
|
|
|
|
int_type: 0,
|
|
|
|
creator: 09870987,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
println!("accept: {}, deny: {}", accept.len(), deny.len());
|
|
|
|
}
|