use crate::models::Vip;
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct OfficialVerify {
#[serde(rename = "type")]
pub verify_type: i8,
pub desc: String,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct RelationListItem {
pub mid: u64,
pub attribute: u8,
pub mtime: Option<u64>,
pub tag: Option<Vec<u64>>,
pub special: u8,
pub contract_info: Option<serde_json::Value>,
pub uname: String,
pub face: String,
pub sign: String,
pub face_nft: u8,
pub official_verify: OfficialVerify,
pub vip: Vip,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct FansListResponseData {
pub list: Vec<RelationListItem>,
pub offset: String,
pub re_version: u32,
pub total: u64,
}
#[cfg(test)]
mod tests {
use super::*;
use crate::ids::Mid;
use crate::probe::contract::HttpMethod;
use crate::probe::endpoint_contract::EndpointContract;
use crate::user::params::UserFollowersParams;
use crate::{ApiEnvelope, BpiClient, BpiError, BpiResult};
use tracing::info;
const TEST_VMID: u64 = 4279370;
fn contract() -> BpiResult<EndpointContract> {
EndpointContract::from_slice(include_bytes!(
"../../../tests/contracts/user/relation-read/followers/contract.json"
))
}
#[ignore = "legacy live API test; requires explicit BPI_LIVE_TEST review"]
#[tokio::test]
async fn test_user_followers() -> Result<(), BpiError> {
if std::env::var_os("BPI_LIVE_TEST").is_none() {
return Ok(());
}
let bpi = BpiClient::new().expect("client should build");
let data = bpi
.user()
.followers(
UserFollowersParams::new(Mid::new(TEST_VMID)?)
.with_page_size(50)
.with_page(1),
)
.await?;
info!("用户粉丝明细: {:?}", data);
assert!(!data.list.is_empty());
assert_eq!(data.list.len(), 50);
assert!(!data.offset.is_empty());
assert!(data.total > 0);
Ok(())
}
#[test]
fn legacy_user_followers_contract_matches_endpoint_request() -> BpiResult<()> {
let contract = contract()?;
assert_eq!(contract.name, "user.followers");
assert_eq!(contract.request.method, HttpMethod::Get);
assert_eq!(
contract.request.url.as_str(),
"https://api.bilibili.com/x/relation/fans"
);
assert_eq!(
contract.request.query.get("vmid").map(String::as_str),
Some("2")
);
assert_eq!(
contract.request.query.get("ps").map(String::as_str),
Some("20")
);
assert_eq!(
contract.request.query.get("pn").map(String::as_str),
Some("1")
);
assert_eq!(
contract.cases[0].response.error.as_deref(),
Some("wbi_risk_control")
);
Ok(())
}
#[test]
fn legacy_user_followers_fixtures_parse_promoted_contract_models() -> BpiResult<()> {
let err = ApiEnvelope::<serde_json::Value>::from_slice(include_bytes!(
"../../../tests/contracts/user/relation-read/followers/responses/anonymous.error.json"
))?
.ensure_success()
.unwrap_err();
assert_eq!(err.code(), Some(-352));
let followers = ApiEnvelope::<FansListResponseData>::from_slice(include_bytes!(
"../../../tests/contracts/user/relation-read/followers/responses/success.json"
))?
.into_payload()?;
assert_eq!(followers.list.len(), 1);
assert_eq!(followers.total, 1);
Ok(())
}
}