use crate::BilibiliRequest;
use crate::BpiResult;
use crate::live::LiveClient;
use reqwest::multipart::Form;
use serde::{Deserialize, Serialize};
use serde_json::Value;
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct CreateRoomData {
#[serde(rename = "roomID")]
pub room_id: Option<String>,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct UpdateRoomData {
pub sub_session_key: String,
pub audit_info: Option<AuditInfo>,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct AuditInfo {
pub audit_title_reason: String,
pub audit_title_status: u8,
pub audit_title: Option<String>,
pub update_title: Option<String>,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct RtmpInfo {
pub addr: String,
pub code: String,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct StartLiveData {
pub change: u8,
pub status: String,
pub rtmp: RtmpInfo,
pub live_key: String,
pub sub_session_key: String,
pub need_face_auth: bool,
pub room_type: Value,
pub protocols: Value,
pub notice: Value,
pub qr: Value,
pub service_source: String,
pub rtmp_backup: Value,
pub up_stream_extra: Value,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct StopLiveData {
pub change: u8,
pub status: String,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct UpdatePreLiveInfoData {
pub audit_info: Option<AuditInfo>,
}
#[derive(Debug, Clone, Deserialize, Serialize)]
pub struct PcLiveVersionData {
pub curr_version: String,
pub build: u64,
pub instruction: String,
pub file_size: String,
pub file_md5: String,
pub content: String,
pub download_url: String,
pub hdiffpatch_switch: u8,
}
impl<'a> LiveClient<'a> {
pub async fn live_create_room(&self) -> BpiResult<CreateRoomData> {
let csrf = self.client.csrf()?;
let form = Form::new()
.text("platform", "web")
.text("visit_id", "")
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
self.client
.post("https://api.live.bilibili.com/xlive/app-blink/v1/preLive/CreateRoom")
.multipart(form)
.send_bpi_payload("live.room.create")
.await
}
pub async fn live_update_room_info(
&self,
room_id: u64,
title: Option<&str>,
area_id: Option<u64>,
add_tag: Option<&str>,
del_tag: Option<&str>,
) -> BpiResult<UpdateRoomData> {
let csrf = self.client.csrf()?;
let mut form = Form::new()
.text("room_id", room_id.to_string())
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
if let Some(t) = title {
form = form.text("title", t.to_string());
}
if let Some(a) = area_id {
form = form.text("area_id", a.to_string());
}
if let Some(a_tag) = add_tag {
form = form.text("add_tag", a_tag.to_string());
}
if let Some(d_tag) = del_tag {
form = form.text("del_tag", d_tag.to_string());
}
self.client
.post("https://api.live.bilibili.com/room/v1/Room/update")
.multipart(form)
.send_bpi_payload("live.room.update")
.await
}
#[allow(dead_code)]
async fn live_start(
&self,
room_id: u64,
area_v2: u64,
platform: &str,
) -> BpiResult<StartLiveData> {
let csrf = self.client.csrf()?;
let form = Form::new()
.text("room_id", room_id.to_string())
.text("area_v2", area_v2.to_string())
.text("platform", platform.to_string())
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
self.client
.post("https://api.live.bilibili.com/room/v1/Room/startLive")
.multipart(form)
.send_bpi_payload("live.start")
.await
}
pub async fn live_stop(&self, room_id: u64, platform: &str) -> BpiResult<StopLiveData> {
let csrf = self.client.csrf()?;
let form = Form::new()
.text("platform", platform.to_string())
.text("room_id", room_id.to_string())
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
self.client
.post("https://api.live.bilibili.com/room/v1/Room/stopLive")
.multipart(form)
.send_bpi_payload("live.stop")
.await
}
pub async fn live_update_pre_live_info(
&self,
title: Option<&str>,
cover: Option<&str>,
) -> BpiResult<UpdatePreLiveInfoData> {
let csrf = self.client.csrf()?;
let mut form = Form::new()
.text("platform", "web")
.text("mobi_app", "web")
.text("build", "1")
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
if let Some(t) = title {
form = form.text("title", t.to_string());
}
if let Some(c) = cover {
form = form.text("cover", c.to_string());
}
self.client
.post("https://api.live.bilibili.com/xlive/app-blink/v1/preLive/UpdatePreLiveInfo")
.multipart(form)
.send_bpi_payload("live.pre_live_info.update")
.await
}
pub async fn live_update_room_news(
&self,
room_id: u64,
uid: u64,
content: &str,
) -> BpiResult<Value> {
let csrf = self.client.csrf()?;
let form = Form::new()
.text("room_id", room_id.to_string())
.text("uid", uid.to_string())
.text("content", content.to_string())
.text("csrf", csrf.clone())
.text("csrf_token", csrf);
self.client
.post("https://api.live.bilibili.com/xlive/app-blink/v1/index/updateRoomNews")
.multipart(form)
.send_bpi_payload("live.room_news.update")
.await
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::probe::contract::HttpMethod;
use crate::probe::endpoint_contract::EndpointContract;
use crate::{ApiEnvelope, BpiResult};
fn version_contract() -> BpiResult<EndpointContract> {
EndpointContract::from_slice(include_bytes!(
"../../tests/contracts/live/public-core/version/contract.json"
))
}
#[test]
fn live_version_contract_matches_endpoint_request() -> BpiResult<()> {
let contract = version_contract()?;
assert_eq!(contract.name, "live.version");
assert_eq!(contract.request.method, HttpMethod::Get);
assert_eq!(
contract.request.url.as_str(),
"https://api.live.bilibili.com/xlive/app-blink/v1/liveVersionInfo/getHomePageLiveVersion"
);
assert_eq!(
contract
.request
.query
.get("system_version")
.map(String::as_str),
Some("2")
);
assert_eq!(contract.cases.len(), 3);
assert_eq!(
contract.cases[0].response.rust_model.as_deref(),
Some("PcLiveVersionData")
);
Ok(())
}
#[test]
fn live_version_response_fixture_parses_declared_model() -> BpiResult<()> {
let payload = ApiEnvelope::<PcLiveVersionData>::from_slice(include_bytes!(
"../../tests/contracts/live/public-core/version/responses/success.json"
))?
.into_payload()?;
assert_eq!(payload.curr_version, "7.61.0.10694");
Ok(())
}
fn local_probe_body(profile: &str) -> Option<serde_json::Value> {
let path =
format!("target/bpi-probe-runs/live/public-core/version/{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 live_version_model_matches_local_probe_outputs_when_available() -> BpiResult<()> {
for profile in ["anonymous", "normal", "vip"] {
if let Some(body) = local_probe_body(profile) {
let payload = serde_json::from_value::<ApiEnvelope<PcLiveVersionData>>(body)?
.into_payload()?;
assert!(!payload.curr_version.is_empty());
}
}
Ok(())
}
}