use serde::Deserialize;
use crate::client::AkShareClient;
use crate::error::Result;
use crate::types::MacroDataPoint;
#[derive(Debug, Deserialize)]
struct EmDatacenterResp {
result: Option<EmResult>,
}
#[derive(Debug, Deserialize)]
struct EmResult {
#[serde(default)]
data: Vec<serde_json::Value>,
}
impl AkShareClient {
pub async fn movie_boxoffice_cinema_daily(&self, date: &str) -> Result<Vec<MacroDataPoint>> {
let date_fmt = format!("{}-{}-{}", &date[..4], &date[4..6], &date[6..]);
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_CINEMA_DAILY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(TRADE_DATE='{}')", date_fmt).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Cinema Daily"))
}
pub async fn movie_boxoffice_cinema_weekly(&self, date: &str) -> Result<Vec<MacroDataPoint>> {
let date_fmt = format!("{}-{}-{}", &date[..4], &date[4..6], &date[6..]);
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_CINEMA_WEEKLY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(TRADE_DATE='{}')", date_fmt).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Cinema Weekly"))
}
pub async fn movie_boxoffice_daily(&self, date: &str) -> Result<Vec<MacroDataPoint>> {
let date_fmt = format!("{}-{}-{}", &date[..4], &date[4..6], &date[6..]);
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_DAILY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(TRADE_DATE='{}')", date_fmt).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Daily"))
}
pub async fn movie_boxoffice_monthly(&self, date: &str) -> Result<Vec<MacroDataPoint>> {
let year = &date[..4];
let month = &date[4..];
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_MONTHLY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
(
"filter",
format!("(YEAR='{}')(MONTH='{}')", year, month).as_str(),
),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Monthly"))
}
pub async fn movie_boxoffice_realtime(&self) -> Result<Vec<MacroDataPoint>> {
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_REALTIME"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Realtime"))
}
pub async fn movie_boxoffice_weekly(&self, date: &str) -> Result<Vec<MacroDataPoint>> {
let date_fmt = format!("{}-{}-{}", &date[..4], &date[4..6], &date[6..]);
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_WEEKLY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(TRADE_DATE='{}')", date_fmt).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Weekly"))
}
pub async fn movie_boxoffice_yearly(&self, year: &str) -> Result<Vec<MacroDataPoint>> {
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_YEARLY"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(YEAR='{}')", year).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Yearly"))
}
pub async fn movie_boxoffice_yearly_first_week(
&self,
year: &str,
) -> Result<Vec<MacroDataPoint>> {
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_YEARLY_FIRST_WEEK"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "BOX_OFFICE"),
("source", "WEB"),
("client", "WEB"),
("filter", format!("(YEAR='{}')", year).as_str()),
])
.send()
.await?
.json()
.await?;
Ok(parse_movie_response(resp, "Yearly First Week"))
}
pub async fn economy_box_office(&self) -> Result<Vec<MacroDataPoint>> {
let url = "https://datacenter-web.eastmoney.com/api/data/v1/get";
let resp: EmDatacenterResp = self
.get(url)
.query(&[
("reportName", "RPT_MOVIE_BOXOFFICE"),
("columns", "ALL"),
("pageNumber", "1"),
("pageSize", "500"),
("sortTypes", "-1"),
("sortColumns", "REPORT_DATE"),
("source", "WEB"),
("client", "WEB"),
])
.send()
.await?
.json()
.await?;
let data = resp.result.map(|r| r.data).unwrap_or_default();
let mut items = Vec::with_capacity(data.len());
for v in &data {
let date = v
.get("REPORT_DATE")
.or_else(|| v.get("DATE"))
.or_else(|| v.get("TRADE_DATE"))
.and_then(|x| x.as_str())
.unwrap_or("")
.to_string();
if date.is_empty() {
continue;
}
let value = v
.get("BOX_OFFICE")
.or_else(|| v.get("INDICATOR_VALUE"))
.or_else(|| v.get("VALUE"))
.or_else(|| v.get("TOTAL_BOX_OFFICE"))
.and_then(|x| x.as_f64())
.unwrap_or(0.0);
items.push(MacroDataPoint {
date: date.get(..10).unwrap_or(&date).to_string(),
value,
name: "Box Office".to_string(),
});
}
Ok(items)
}
}
fn parse_movie_response(resp: EmDatacenterResp, label: &str) -> Vec<MacroDataPoint> {
let data = resp.result.map(|r| r.data).unwrap_or_default();
let mut items = Vec::with_capacity(data.len());
for v in &data {
let date = v
.get("TRADE_DATE")
.or_else(|| v.get("REPORT_DATE"))
.or_else(|| v.get("DATE"))
.and_then(|x| x.as_str())
.unwrap_or("")
.to_string();
let value = v
.get("BOX_OFFICE")
.or_else(|| v.get("INDICATOR_VALUE"))
.or_else(|| v.get("VALUE"))
.and_then(|x| x.as_f64())
.unwrap_or(0.0);
let name = v
.get("MOVIE_NAME")
.or_else(|| v.get("NAME"))
.and_then(|x| x.as_str())
.unwrap_or(label)
.to_string();
items.push(MacroDataPoint {
date: date.get(..10).unwrap_or(&date).to_string(),
value,
name,
});
}
items
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_economy_box_office_response_structure() {
let json = r#"{
"result": {
"data": [
{"REPORT_DATE": "2024-04-01T00:00:00", "BOX_OFFICE": 12580.5},
{"REPORT_DATE": "2024-03-31T00:00:00", "BOX_OFFICE": 9876.3}
]
}
}"#;
let resp: EmDatacenterResp = serde_json::from_str(json).unwrap();
let data = resp.result.unwrap().data;
assert_eq!(data.len(), 2);
assert_eq!(
data[0].get("BOX_OFFICE").and_then(|v| v.as_f64()),
Some(12580.5)
);
}
#[test]
fn test_economy_box_office_empty_response() {
let json = r#"{"result": {"data": []}}"#;
let resp: EmDatacenterResp = serde_json::from_str(json).unwrap();
let data = resp.result.unwrap().data;
assert!(data.is_empty());
}
#[test]
fn test_economy_box_office_null_result() {
let json = r#"{"result": null}"#;
let resp: EmDatacenterResp = serde_json::from_str(json).unwrap();
let data = resp.result.map(|r| r.data).unwrap_or_default();
assert!(data.is_empty());
}
}