| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116 |
- use std::str::FromStr;
- use std::sync::atomic::{AtomicI64, AtomicU64, Ordering};
- use anyhow::{anyhow, bail, Result};
- use rust_decimal::Decimal;
- use serde_json::Value;
- use tracing::{info, warn};
- use crate::utils::response::Response;
- pub struct DataManager {
- pub best_ask: Decimal,
- pub best_bid: Decimal,
- pub delay_total: AtomicI64,
- pub delay_count: AtomicU64,
- }
- impl DataManager {
- pub fn new() -> Self {
- DataManager {
- best_ask: Default::default(),
- best_bid: Default::default(),
-
- delay_total: AtomicI64::new(0),
- delay_count: AtomicU64::new(0),
- }
- }
- pub fn record_latency(&self, received_at: i64, origin_timestamp: i64) {
- if let Some(delay) = received_at.checked_sub(origin_timestamp) {
- self.delay_total.fetch_add(delay, Ordering::Relaxed); // 原子加
- self.delay_count.fetch_add(1, Ordering::Relaxed); // 原子加
- } else {
- warn!("时间戳计算出现问题: received_at={}, origin_timestamp={}", received_at, origin_timestamp);
- }
- }
- // 获取当前的统计数据
- pub fn get_delay_stats(&self) -> (i64, u64) {
- let total = self.delay_total.load(Ordering::Relaxed);
- let count = self.delay_count.load(Ordering::Relaxed);
- (total, count)
- }
- // 重置统计数据 -> 这个是关键!
- pub fn reset_delay_stats(&self) {
- self.delay_total.store(0, Ordering::Relaxed); // 原子写
- self.delay_count.store(0, Ordering::Relaxed); // 原子写
- }
-
- pub async fn dispatch_message(&mut self, response: &Response) -> Result<()> {
- // 1. 预解析为通用的 Value
- let v = response.data.clone();
-
- // info!("准备分发的消息:{}, {}", serde_json::to_string_pretty(&v)?, response.label);
- // 2. 获取 topic_info 字段用于路由消息,在该策略中extended可以用label
- let topic_info = &response.label;
-
- // 3. 根据 topic_info 的内容进行分发 (match)
- if topic_info.contains("ExtendedBestPrices") {
- self.process_best_prices(&v).await?;
- } else if topic_info.contains("spot@public.aggre.depth.v3.api.pb") {
-
- } else {
- // 如果是未知的 topic,返回一个错误
- bail!("Received a message with an unknown topic_info: {}", topic_info);
- }
- Ok(())
- }
-
- pub async fn process_best_prices(&mut self, value: &Value) -> Result<()> {
- // 预先捕获整个 Value 的字符串表示,用于错误报告
- let value_str = serde_json::to_string(&value).unwrap_or_else(|_| "无法序列化 JSON Value".to_string());
- // 尝试获取 data 字段
- let data = value.get("data")
- .ok_or_else(|| anyhow!("获取 'data' 字段失败,原始 JSON: {}", value_str))?;
- // 尝试从 data 中获取 "a" (asks) 数组
- let asks_array = data.get("a")
- .and_then(|v| v.as_array()) // and_then 链式调用,确保只有当 v 存在且是数组时才继续
- .ok_or_else(|| anyhow!("获取 'data.a' 数组失败,原始 JSON: {}", value_str))?;
- // 尝试从 data 中获取 "b" (bids) 数组
- let bids_array = data.get("b")
- .and_then(|v| v.as_array())
- .ok_or_else(|| anyhow!("获取 'data.b' 数组失败,原始 JSON: {}", value_str))?;
- // 如若有发送asks信息
- if asks_array.len() > 0 {
- let ask_item = &asks_array[0];
- let p = ask_item.get("p")
- .and_then(|v| v.as_str())
- .ok_or_else(|| anyhow!("获取 'data.a.p' 字符串失败,原始 JSON: {}", value_str))?;
-
- self.best_ask = Decimal::from_str(p)
- .map_err(|e| anyhow!("将价格字符串 '{}' 解析为 Decimal 失败: {},原始 JSON: {}", p, e, value_str))?;
- }
- // 如若有发送bids信息
- if bids_array.len() > 0 {
- let bid_item = &bids_array[0];
- let p = bid_item.get("p")
- .and_then(|v| v.as_str())
- .ok_or_else(|| anyhow!("获取 'data.b.p' 字符串失败,原始 JSON: {}", value_str))?;
- self.best_bid = Decimal::from_str(p)
- .map_err(|e| anyhow!("将价格字符串 '{}' 解析为 Decimal 失败: {},原始 JSON: {}", p, e, value_str))?;
- }
-
- info!("{}, {}", self.best_ask, self.best_bid);
-
- Ok(())
- }
- }
|