Skip to content

Commit 04a5ccd

Browse files
authored
Merge pull request #12 from YDX-2147483647:itc
follow rules by ITC to obtain the `ac_id`
2 parents 2d5ee29 + 6e964e8 commit 04a5ccd

File tree

2 files changed

+23
-6
lines changed

2 files changed

+23
-6
lines changed

Cargo.toml

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
[package]
22
name = "bitsrun"
33
description = "A headless login and logout CLI for 10.0.0.55 at BIT"
4-
version = "0.3.1"
4+
version = "0.4.0"
55
edition = "2021"
66
license = "MIT"
77
homepage = "https://github.com/spencerwooo/bitsrun-rs"
@@ -42,6 +42,6 @@ pretty_env_logger = "0.5.0"
4242
strip = "symbols"
4343

4444
[package.metadata.deb]
45-
copyright = "2023 Spencer Woo"
45+
copyright = "2024 Spencer Woo"
4646
maintainer-scripts = "debian/"
4747
systemd-units = { enable = true, start = false }

src/client.rs

+21-4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ pub const SRUN_PORTAL: &str = "http://10.0.0.55";
2323
pub const SRUN_TYPE: &str = "1";
2424
pub const SRUN_N: &str = "200";
2525

26+
/// An arbitrary HTTP URL for srun to redirect
27+
pub const CAPTIVE_PORTAL_TEST: &str = "http://www.bit.edu.cn";
28+
2629
/// The response from the `/rad_user_info` endpoint
2730
///
2831
/// This response is used to determine if the device is logged in or not, and if it is logged in,
@@ -128,12 +131,12 @@ pub async fn get_login_state(client: &Client, verbose: bool) -> Result<SrunLogin
128131
Ok(parsed_json)
129132
}
130133

131-
/// Get the ac_id of the current device
132-
async fn get_acid(client: &Client) -> Result<String> {
133-
let resp = client.get(SRUN_PORTAL).send().await.with_context(|| {
134+
/// Get the ac_id of the current device by visiting a URL
135+
async fn get_acid_by_url(client: &Client, url: &str) -> Result<String> {
136+
let resp = client.get(url).send().await.with_context(|| {
134137
format!(
135138
"failed to get ac_id from `{}`",
136-
SRUN_PORTAL.if_supports_color(Stdout, |t| t.underline())
139+
url.if_supports_color(Stdout, |t| t.underline())
137140
)
138141
})?;
139142
let redirect_url = resp.url().to_string();
@@ -154,6 +157,20 @@ async fn get_acid(client: &Client) -> Result<String> {
154157
Ok(ac_id.1)
155158
}
156159

160+
/// Get the ac_id of the current device
161+
async fn get_acid(client: &Client) -> Result<String> {
162+
// Try to visit `CAPTIVE_PORTAL_TEST`.
163+
// If not logged in, it will be redirected to `SRUN_PORTAL` with ac_id.
164+
// Otherwise, we fall back to visit `SRUN_PORTAL` directly.
165+
// https://en.wikipedia.org/wiki/Captive_portal#Detection
166+
//
167+
// Because of ITC's double authentication mechanism, visiting `SRUN_PORTAL` directly is not preferred.
168+
// https://itc.bit.edu.cn/fwzn/zxbl/f2c0c8e939ce4e9cace880d5403fe4b5.htm
169+
get_acid_by_url(client, CAPTIVE_PORTAL_TEST)
170+
.await
171+
.or(get_acid_by_url(client, SRUN_PORTAL).await)
172+
}
173+
157174
/// SRUN portal response type when calling login/logout
158175
///
159176
/// Note that fields that are not used are omitted

0 commit comments

Comments
 (0)