steamguard-cli/steamguard/src/lib.rs

155 lines
4.2 KiB
Rust
Raw Normal View History

use crate::protobufs::service_twofactor::CTwoFactor_RemoveAuthenticator_Request;
use crate::steamapi::EResult;
use crate::{
steamapi::twofactor::TwoFactorClient, token::TwoFactorSecret, transport::WebApiTransport,
};
pub use accountlinker::{AccountLinkError, AccountLinker, FinalizeLinkError};
pub use confirmation::*;
pub use qrapprover::{QrApprover, QrApproverError};
2022-06-19 20:44:18 +02:00
pub use secrecy::{ExposeSecret, SecretString};
2021-08-01 14:43:18 +02:00
use serde::{Deserialize, Serialize};
use std::io::Read;
use token::Tokens;
use transport::TransportError;
pub use userlogin::{DeviceDetails, LoginError, UserLogin};
#[macro_use]
extern crate lazy_static;
#[macro_use]
extern crate anyhow;
extern crate maplit;
pub mod accountlinker;
2022-12-06 16:02:07 +01:00
mod api_responses;
mod confirmation;
pub mod phonelinker;
pub mod protobufs;
mod qrapprover;
pub mod refresher;
2022-06-19 20:44:18 +02:00
mod secret_string;
2021-08-01 14:43:18 +02:00
pub mod steamapi;
pub mod token;
pub mod transport;
pub mod userlogin;
2021-03-22 02:21:29 +01:00
extern crate base64;
2021-03-30 21:51:26 +02:00
extern crate cookie;
2021-08-01 14:43:18 +02:00
extern crate hmacsha1;
2021-03-22 02:21:29 +01:00
2021-03-27 17:14:34 +01:00
#[derive(Debug, Clone, Serialize, Deserialize)]
2021-03-22 02:21:29 +01:00
pub struct SteamGuardAccount {
2021-08-08 18:54:46 +02:00
pub account_name: String,
pub steam_id: u64,
2021-08-08 18:54:46 +02:00
pub serial_number: String,
#[serde(with = "secret_string")]
pub revocation_code: SecretString,
pub shared_secret: TwoFactorSecret,
2021-08-08 18:54:46 +02:00
pub token_gid: String,
#[serde(with = "secret_string")]
pub identity_secret: SecretString,
#[serde(with = "secret_string")]
pub uri: SecretString,
2021-08-08 18:54:46 +02:00
pub device_id: String,
#[serde(with = "secret_string")]
pub secret_1: SecretString,
pub tokens: Option<Tokens>,
2021-03-22 02:21:29 +01:00
}
impl Default for SteamGuardAccount {
fn default() -> Self {
Self {
2021-08-08 18:54:46 +02:00
account_name: String::from(""),
steam_id: 0,
2021-08-08 18:54:46 +02:00
serial_number: String::from(""),
revocation_code: String::from("").into(),
shared_secret: TwoFactorSecret::new(),
2021-08-08 18:54:46 +02:00
token_gid: String::from(""),
identity_secret: String::from("").into(),
uri: String::from("").into(),
2021-08-08 18:54:46 +02:00
device_id: String::from(""),
secret_1: String::from("").into(),
tokens: None,
}
}
}
impl SteamGuardAccount {
pub fn new() -> Self {
Self::default()
2021-08-08 18:54:46 +02:00
}
pub fn from_reader<T>(r: T) -> anyhow::Result<Self>
where
T: Read,
{
Ok(serde_json::from_reader(r)?)
}
pub fn set_tokens(&mut self, tokens: Tokens) {
self.tokens = Some(tokens);
}
pub fn is_logged_in(&self) -> bool {
self.tokens.is_some()
}
pub fn generate_code(&self, time: u64) -> String {
self.shared_secret.generate_code(time)
2021-08-08 18:54:46 +02:00
}
2021-08-12 01:39:29 +02:00
/// Removes the mobile authenticator from the steam account. If this operation succeeds, this object can no longer be considered valid.
/// Returns whether or not the operation was successful.
pub fn remove_authenticator(
&self,
revocation_code: Option<&String>,
) -> Result<(), RemoveAuthenticatorError> {
if !matches!(revocation_code, Some(_)) && self.revocation_code.expose_secret().is_empty() {
return Err(RemoveAuthenticatorError::MissingRevocationCode);
}
let Some(tokens) = &self.tokens else {
return Err(RemoveAuthenticatorError::TransportError(TransportError::Unauthorized));
2021-08-08 18:54:46 +02:00
};
let mut client = TwoFactorClient::new(WebApiTransport::default());
let mut req = CTwoFactor_RemoveAuthenticator_Request::new();
req.set_revocation_code(
revocation_code
.unwrap_or(self.revocation_code.expose_secret())
.to_owned(),
);
let resp = client.remove_authenticator(req, tokens.access_token())?;
// returns EResult::TwoFactorCodeMismatch if the revocation code is incorrect
if resp.result != EResult::OK && resp.result != EResult::TwoFactorCodeMismatch {
return Err(resp.result.into());
}
let resp = resp.into_response_data();
if !resp.success() {
return Err(RemoveAuthenticatorError::IncorrectRevocationCode {
attempts_remaining: resp.revocation_attempts_remaining(),
});
}
Ok(())
}
}
#[derive(Debug, thiserror::Error)]
pub enum RemoveAuthenticatorError {
#[error("Missing revocation code")]
MissingRevocationCode,
#[error("Incorrect revocation code, {attempts_remaining} attempts remaining")]
IncorrectRevocationCode { attempts_remaining: u32 },
#[error("Transport error: {0}")]
TransportError(#[from] TransportError),
#[error("Steam returned an enexpected result: {0:?}")]
UnknownEResult(EResult),
#[error("Unexpected error: {0}")]
Unknown(#[from] anyhow::Error),
}
impl From<EResult> for RemoveAuthenticatorError {
fn from(e: EResult) -> Self {
Self::UnknownEResult(e)
2021-08-08 18:54:46 +02:00
}
}