use super::models::{ArticleAuthor, ArticleCategory, ArticleMedia, ArticleStats};
use serde::{Deserialize, Serialize};
pub type CardData = std::collections::HashMap<String, CardItem>;
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum CardItem {
Video(Box<VideoCard>),
Article(Box<ArticleCard>),
Live(Box<LiveCard>),
Unknown(serde_json::Value),
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VideoCard {
pub aid: i64,
pub bvid: String,
pub cid: i64,
pub copyright: i32,
pub pic: String,
pub ctime: i64,
pub desc: String,
pub dimension: VideoDimension,
pub duration: i64,
pub dynamic: String,
pub owner: VideoOwner,
pub pubdate: i64,
pub rights: VideoRights,
pub short_link_v2: String,
pub stat: VideoStat,
pub state: i32,
pub tid: i32,
pub title: String,
pub tname: String,
pub videos: i32,
pub vt_switch: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VideoDimension {
pub height: i32,
pub rotate: i32,
pub width: i32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VideoOwner {
pub face: String,
pub mid: i64,
pub name: String,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VideoRights {
pub arc_pay: i32,
pub autoplay: i32,
pub bp: i32,
pub download: i32,
pub elec: i32,
pub hd5: i32,
pub is_cooperation: i32,
pub movie: i32,
pub no_background: i32,
pub no_reprint: i32,
pub pay: i32,
pub pay_free_watch: i32,
pub ugc_pay: i32,
pub ugc_pay_preview: i32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VideoStat {
pub aid: i64,
pub coin: i64,
pub danmaku: i64,
pub dislike: i64,
pub favorite: i64,
pub his_rank: i32,
pub like: i64,
pub now_rank: i32,
pub reply: i64,
pub share: i64,
pub view: i64,
pub vt: i32,
pub vv: i32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ArticleCard {
pub act_id: i64,
pub apply_time: String,
pub attributes: i32,
#[serde(rename = "authenMark")]
pub authen_mark: Option<serde_json::Value>,
pub author: ArticleAuthor,
pub banner_url: String,
pub categories: Vec<ArticleCategory>,
pub category: ArticleCategory,
pub check_state: i32,
pub check_time: String,
pub content_pic_list: Option<serde_json::Value>,
pub cover_avid: i64,
pub ctime: i64,
pub dispute: Option<serde_json::Value>,
pub dynamic: String,
pub id: i64,
pub image_urls: Vec<String>,
pub is_like: bool,
pub list: Option<ArticleList>,
pub media: ArticleMedia,
pub mtime: i64,
pub origin_image_urls: Vec<String>,
pub origin_template_id: i32,
pub original: i32,
pub private_pub: i32,
pub publish_time: i64,
pub reprint: i32,
pub state: i32,
pub stats: ArticleStats,
pub summary: String,
pub template_id: i32,
pub title: String,
pub top_video_info: Option<serde_json::Value>,
pub r#type: i32,
pub words: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct AuthorVip {
pub avatar_subscript: i32,
pub due_date: i64,
pub label: VipLabel,
pub nickname_color: String,
pub status: i32,
pub theme_type: i32,
pub r#type: i32,
pub vip_pay_type: i32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VipLabel {
pub label_theme: String,
pub path: String,
pub text: String,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ArticleList {
pub apply_time: String,
pub articles_count: i32,
pub check_time: String,
pub ctime: i64,
pub id: i64,
pub image_url: String,
pub mid: i64,
pub name: String,
pub publish_time: i64,
pub read: i64,
pub reason: String,
pub state: i32,
pub summary: String,
pub update_time: i64,
pub words: i64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct LiveCard {
pub area_v2_name: String,
pub cover: String,
pub face: String,
pub live_status: i32,
pub online: i64,
pub pendent_ru: String,
pub pendent_ru_color: String,
pub pendent_ru_pic: String,
pub role: i32,
pub room_id: i64,
pub title: String,
pub uid: i64,
pub uname: String,
}
#[cfg(test)]
mod tests {
use super::*;
use crate::article::params::ArticleCardsParams;
use crate::probe::contract::HttpMethod;
use crate::probe::endpoint_contract::EndpointContract;
use crate::{ApiEnvelope, BpiClient, BpiError, BpiResult};
use std::mem;
fn contract() -> BpiResult<EndpointContract> {
EndpointContract::from_slice(include_bytes!(
"../../tests/contracts/article/cards/contract.json"
))
}
#[ignore = "legacy live API test; requires explicit BPI_LIVE_TEST review"]
#[tokio::test]
async fn test_get_article_cards() -> Result<(), Box<BpiError>> {
let bpi = BpiClient::new().expect("client should build");
let params = ArticleCardsParams::new("av2,cv1,cv2")?;
let data = bpi.article().cards(params).await?;
tracing::info!("{:#?}", data);
Ok(())
}
#[test]
fn card_item_keeps_large_payloads_boxed() {
assert!(mem::size_of::<CardItem>() <= 64);
}
#[test]
fn article_cards_contract_matches_endpoint_request() -> BpiResult<()> {
let contract = contract()?;
let params = ArticleCardsParams::new("av2,cv1,cv2")?;
assert_eq!(contract.name, "article.cards");
assert_eq!(contract.request.method, HttpMethod::Get);
assert_eq!(
contract.request.url.as_str(),
"https://api.bilibili.com/x/article/cards"
);
assert_eq!(
contract.request.query.get("ids").map(String::as_str),
Some("av2,cv1,cv2")
);
assert_eq!(
contract
.request
.query
.get("web_location")
.map(String::as_str),
Some("333.1305")
);
assert_eq!(
params.query_pairs(),
vec![
("ids", "av2,cv1,cv2".to_string()),
("web_location", "333.1305".to_string()),
]
);
assert_eq!(contract.cases.len(), 3);
assert_eq!(
contract.cases[0].response.error.as_deref(),
Some("wbi_risk_control")
);
assert_eq!(
contract.cases[1].response.rust_model.as_deref(),
Some("CardData")
);
Ok(())
}
#[test]
fn article_cards_response_fixtures_parse_declared_model() -> BpiResult<()> {
for bytes in [
include_bytes!("../../tests/contracts/article/cards/responses/normal.success.json")
.as_slice(),
include_bytes!("../../tests/contracts/article/cards/responses/vip.success.json")
.as_slice(),
] {
let payload = ApiEnvelope::<CardData>::from_slice(bytes)?.into_payload()?;
assert!(payload.contains_key("av2"));
assert!(payload.contains_key("cv1"));
}
Ok(())
}
#[test]
fn article_cards_anonymous_fixture_records_wbi_error() -> BpiResult<()> {
let err = ApiEnvelope::<serde_json::Value>::from_slice(include_bytes!(
"../../tests/contracts/article/cards/responses/anonymous.error.json"
))?
.ensure_success()
.unwrap_err();
assert_eq!(err.code(), Some(-352));
Ok(())
}
fn local_probe_body(profile: &str) -> Option<serde_json::Value> {
let path = format!("target/bpi-probe-runs/article/read/cards/{profile}.response.json");
let bytes = std::fs::read(path).ok()?;
let value: serde_json::Value = serde_json::from_slice(&bytes).ok()?;
value
.get("response")
.and_then(|response| response.get("body"))
.cloned()
}
#[test]
fn article_cards_model_matches_local_probe_outputs_when_available() -> BpiResult<()> {
for profile in ["normal", "vip"] {
let Some(body) = local_probe_body(profile) else {
continue;
};
let payload = serde_json::from_value::<ApiEnvelope<CardData>>(body)?.into_payload()?;
assert!(payload.contains_key("cv1"));
}
Ok(())
}
}