feat: finish p31
This commit is contained in:
1678
Cargo.lock
generated
1678
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -16,3 +16,7 @@ num-traits = "0.2"
|
|||||||
once_cell = "1.21"
|
once_cell = "1.21"
|
||||||
rayon = "1.11.0"
|
rayon = "1.11.0"
|
||||||
crypto-bigint = "0.6"
|
crypto-bigint = "0.6"
|
||||||
|
axum = "0.8"
|
||||||
|
tokio = { version = "1", features = ["full"] }
|
||||||
|
serde = { version = "1", features = ["derive"] }
|
||||||
|
reqwest = "0.12"
|
||||||
|
|||||||
14
problems/p31/Cargo.toml
Normal file
14
problems/p31/Cargo.toml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
[package]
|
||||||
|
name = "p31"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2024"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
reqwest = { workspace = true }
|
||||||
|
tokio = { workspace = true }
|
||||||
|
axum = { workspace = true }
|
||||||
|
serde = { workspace = true }
|
||||||
|
common = { path = "../../common/" }
|
||||||
|
hex = { workspace = true }
|
||||||
|
rand = { workspace = true }
|
||||||
|
anyhow = { workspace = true }
|
||||||
134
problems/p31/src/main.rs
Normal file
134
problems/p31/src/main.rs
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
|
use common::sha1;
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
use axum::{Router, extract::Query, http::StatusCode, routing::get};
|
||||||
|
use rand::Rng;
|
||||||
|
use serde::Deserialize;
|
||||||
|
use std::sync::LazyLock;
|
||||||
|
use tokio::time::sleep;
|
||||||
|
|
||||||
|
static HMAC_KEY: LazyLock<[u8; 20]> = LazyLock::new(|| {
|
||||||
|
// 生成随机密钥
|
||||||
|
let mut rng = rand::rng();
|
||||||
|
let key: [u8; 20] = rng.random();
|
||||||
|
key
|
||||||
|
});
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct Params {
|
||||||
|
file: String,
|
||||||
|
signature: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn verify_handler(Query(params): Query<Params>) -> StatusCode {
|
||||||
|
let my_signature = hmac_sha1(&*HMAC_KEY, params.file.as_bytes());
|
||||||
|
|
||||||
|
let provided_signature = match hex::decode(¶ms.signature) {
|
||||||
|
Ok(bytes) => bytes,
|
||||||
|
Err(e) => {
|
||||||
|
eprintln!("Hex decode error: {:?}", e);
|
||||||
|
return StatusCode::BAD_REQUEST;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
match insecure_compare(&my_signature, &provided_signature).await {
|
||||||
|
true => StatusCode::OK,
|
||||||
|
false => StatusCode::INTERNAL_SERVER_ERROR,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn insecure_compare(a: &[u8], b: &[u8]) -> bool {
|
||||||
|
if a.len() != b.len() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for i in 0..a.len() {
|
||||||
|
if a[i] != b[i] {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
sleep(Duration::from_millis(50)).await;
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
fn hmac_sha1(key: &[u8], message: &[u8]) -> [u8; 20] {
|
||||||
|
let mut key = match key.len() {
|
||||||
|
0..=64 => key.to_vec(),
|
||||||
|
_ => sha1(key).to_vec(),
|
||||||
|
};
|
||||||
|
key.extend(vec![0; 64 - key.len()]);
|
||||||
|
let mut part_1 = key.iter().map(|x| x ^ 0x5c).collect::<Vec<u8>>();
|
||||||
|
let mut part_2_input = key.iter().map(|x| x ^ 0x36).collect::<Vec<u8>>();
|
||||||
|
part_2_input.extend(message);
|
||||||
|
let part_2 = sha1(&part_2_input);
|
||||||
|
part_1.extend(part_2);
|
||||||
|
sha1(&part_1)
|
||||||
|
}
|
||||||
|
#[test]
|
||||||
|
fn test_hmac_sha1() {
|
||||||
|
let output = hmac_sha1("123".as_bytes(), "helloworld".as_bytes());
|
||||||
|
let output = hex::encode(output);
|
||||||
|
let correct = "8849fb5557760d35f19e2100aa250bfb97152bd9";
|
||||||
|
assert_eq!(output, correct);
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn start_attack() -> Result<()> {
|
||||||
|
let mut signature = vec![0u8; 20]; // 用可变字节数组
|
||||||
|
let mut pos = 0; // 当前爆破位置
|
||||||
|
|
||||||
|
while pos < 20 {
|
||||||
|
let mut max_time = Duration::from_millis(0);
|
||||||
|
let mut best_byte = 0u8;
|
||||||
|
|
||||||
|
// 尝试当前位置的所有可能字节值
|
||||||
|
for byte in 0x00..=0xff {
|
||||||
|
signature[pos] = byte;
|
||||||
|
let url = format!(
|
||||||
|
"http://localhost:3000/verify?file={}&signature={}",
|
||||||
|
"helloworld",
|
||||||
|
hex::encode(&signature)
|
||||||
|
);
|
||||||
|
|
||||||
|
let start = Instant::now();
|
||||||
|
let status = reqwest::get(&url).await?.status();
|
||||||
|
let elapsed = start.elapsed();
|
||||||
|
|
||||||
|
if status == StatusCode::OK {
|
||||||
|
println!("✓ 找到完整签名: {}", hex::encode(&signature));
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录耗时最长的字节
|
||||||
|
if elapsed > max_time {
|
||||||
|
max_time = elapsed;
|
||||||
|
best_byte = byte;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
signature[pos] = best_byte;
|
||||||
|
println!(
|
||||||
|
"位置 {} 确定为: {:02x}, 耗时: {:?}",
|
||||||
|
pos, best_byte, max_time
|
||||||
|
);
|
||||||
|
pos += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() -> Result<()> {
|
||||||
|
let app = Router::new().route("/verify", get(verify_handler));
|
||||||
|
|
||||||
|
let listener = tokio::net::TcpListener::bind("127.0.0.1:3000").await?;
|
||||||
|
|
||||||
|
dbg!(hex::encode(*HMAC_KEY));
|
||||||
|
|
||||||
|
// 后台启动服务器
|
||||||
|
tokio::spawn(async move {
|
||||||
|
axum::serve(listener, app).await.unwrap();
|
||||||
|
});
|
||||||
|
start_attack().await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user