1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
use serde::Serialize;
use snowchains_core::web::{
Atcoder, AtcoderLoginCredentials, Codeforces, CodeforcesLoginCredentials, CookieStorage, Login,
PlatformKind,
};
use std::{
cell::RefCell,
io::{BufRead, Write},
};
use structopt::StructOpt;
use strum::VariantNames as _;
use termcolor::WriteColor;
#[derive(StructOpt, Debug)]
pub struct OptLogin {
#[structopt(long)]
pub json: bool,
#[structopt(
long,
possible_values(crate::ColorChoice::VARIANTS),
default_value("auto")
)]
pub color: crate::ColorChoice,
#[structopt(possible_values(&["atcoder", "codeforces"]))]
pub service: PlatformKind,
}
#[derive(Clone, Copy, Debug, Serialize)]
struct Outcome {
kind: snowchains_core::web::LoginOutcome,
}
impl Outcome {
fn to_json(self) -> String {
serde_json::to_string(&self).expect("should not fail")
}
}
pub(crate) fn run(
opt: OptLogin,
ctx: crate::Context<impl BufRead, impl Write, impl WriteColor>,
) -> anyhow::Result<()> {
let OptLogin {
json,
color: _,
service,
} = opt;
let crate::Context { cwd: _, mut shell } = ctx;
let cookie_storage = CookieStorage::with_jsonl(crate::web::credentials::cookie_store_path()?)?;
let timeout = Some(crate::web::SESSION_TIMEOUT);
let outcome = match service {
PlatformKind::Atcoder => {
let shell = RefCell::new(&mut shell);
let credentials = AtcoderLoginCredentials {
username_and_password: &mut crate::web::credentials::atcoder_username_and_password(
&shell,
),
};
Atcoder::exec(Login {
credentials,
cookie_storage,
timeout,
shell: &shell,
})
}
PlatformKind::Codeforces => {
let shell = RefCell::new(&mut shell);
let credentials = CodeforcesLoginCredentials {
username_and_password:
&mut crate::web::credentials::codeforces_username_and_password(&shell),
};
Codeforces::exec(Login {
credentials,
cookie_storage,
timeout,
shell: &shell,
})
}
PlatformKind::Yukicoder => unreachable!("should be filtered by `possible_values`"),
}?;
let message = if json {
Outcome { kind: outcome }.to_json()
} else {
match outcome {
snowchains_core::web::LoginOutcome::Success => "Successfully logged in.",
snowchains_core::web::LoginOutcome::AlreadyLoggedIn => "Already logged in.",
}
.to_owned()
};
writeln!(shell.stdout, "{}", message)?;
shell.stdout.flush().map_err(Into::into)
}