use serde::{Deserialize, Serialize};
#[cfg(test)]
const NAV_ENDPOINT: &str = "https://api.bilibili.com/x/web-interface/nav";
#[derive(Default, Debug, Clone, Serialize, Deserialize)]
#[serde(default)]
pub struct NavData {
#[serde(rename = "isLogin")]
pub is_login: bool,
pub wbi_img: WbiImg,
pub email_verified: i32,
pub face: String,
pub face_nft: i32,
pub level_info: LevelInfo,
pub mid: u64,
pub mobile_verified: i32,
pub money: f64,
pub moral: i32,
pub official: Official,
#[serde(rename = "officialVerify")]
pub official_verify: OfficialVerify,
pub pendant: Pendant,
pub scores: i32,
pub uname: String,
#[serde(rename = "vipDueDate")]
pub vip_due_date: u64,
#[serde(rename = "vipStatus")]
pub vip_status: i32,
#[serde(rename = "vipType")]
pub vip_type: i32,
pub vip_pay_type: i32,
pub vip_theme_type: i32,
pub vip_label: VipLabel,
pub vip_avatar_subscript: i32,
pub vip_nickname_color: String,
pub vip: Vip,
pub wallet: Wallet,
pub has_shop: bool,
pub shop_url: String,
pub is_senior_member: i32,
pub is_jury: bool,
pub name_render: Option<serde_json::Value>,
}
#[derive(Default, Debug, Clone, Serialize, Deserialize)]
pub struct Wallet {
pub mid: u64,
pub bcoin_balance: i64,
pub coupon_balance: i64,
pub coupon_due_time: i64,
}
#[derive(Default, Debug, Clone, Serialize, Deserialize)]
pub struct WbiImg {
pub img_url: String,
pub sub_url: String,
}
use crate::models::{LevelInfo, Official, OfficialVerify, Pendant, Vip, VipLabel};
#[cfg(test)]
mod tests {
use super::*;
use tracing::info;
use crate::probe::contract::HttpMethod;
use crate::probe::endpoint_contract::EndpointContract;
use crate::{ApiEnvelope, BpiClient, BpiError, BpiResult};
fn contract() -> BpiResult<EndpointContract> {
EndpointContract::from_slice(include_bytes!(
"../../../tests/contracts/login/nav/contract.json"
))
}
fn live_login_tests_enabled() -> bool {
std::env::var("BPI_LIVE_TEST").ok().as_deref() == Some("1")
}
fn live_client() -> Result<BpiClient, BpiError> {
match std::env::var("BPI_COOKIE") {
Ok(cookie) if !cookie.trim().is_empty() => BpiClient::builder().cookie(cookie).build(),
_ => BpiClient::new(),
}
}
#[ignore = "legacy live API test; requires explicit BPI_LIVE_TEST review"]
#[tokio::test]
async fn test_bilibili_uinfo() -> Result<(), BpiError> {
if !live_login_tests_enabled() {
return Ok(());
}
let bpi = live_client()?;
let data = bpi.login().nav().await?;
if data.is_login {
info!(
"登录成功!UID={:?} 昵称={:?} ",
data.mid.as_ref().map(|mid| mid.get()),
data.uname
);
}
Ok(())
}
#[ignore = "legacy live API test; requires explicit BPI_LIVE_TEST review"]
#[tokio::test]
async fn test_user_info() -> Result<(), BpiError> {
if !live_login_tests_enabled() {
return Ok(());
}
let bpi = live_client()?;
let user_info = bpi.login().nav().await?;
info!("用户信息:{:?}", user_info);
Ok(())
}
#[test]
fn legacy_login_info_nav_contract_matches_endpoint_request() -> BpiResult<()> {
let contract = contract()?;
assert_eq!(contract.name, "login.nav");
assert_eq!(contract.request.method, HttpMethod::Get);
assert_eq!(contract.request.url.as_str(), NAV_ENDPOINT);
assert!(contract.request.query.is_empty());
assert_eq!(contract.cases.len(), 3);
assert_eq!(contract.cases[0].response.api_code, Some(-101));
assert_eq!(contract.cases[1].response.api_code, Some(0));
assert_eq!(contract.cases[2].response.api_code, Some(0));
Ok(())
}
#[test]
fn legacy_login_info_nav_fixtures_parse_promoted_contract_models() -> BpiResult<()> {
for (bytes, expected_mid) in [
(
include_bytes!("../../../tests/contracts/login/nav/responses/normal.success.json")
.as_slice(),
1_000_001,
),
(
include_bytes!("../../../tests/contracts/login/nav/responses/vip.success.json")
.as_slice(),
1_000_002,
),
] {
let payload = ApiEnvelope::<NavData>::from_slice(bytes)?.into_payload()?;
assert!(payload.is_login);
assert_eq!(payload.mid, expected_mid);
assert!(!payload.uname.trim().is_empty());
}
let err = ApiEnvelope::<serde_json::Value>::from_slice(include_bytes!(
"../../../tests/contracts/login/nav/responses/anonymous.error.json"
))?
.ensure_success()
.unwrap_err();
assert!(err.requires_login());
Ok(())
}
}