use std::collections::BTreeMap;
use brk_error::{Error, Result};
use brk_types::{
CentsUnsigned, CheckedSub, Close, Date, DateIndex, Dollars, Height, High, Low,
OHLCCentsUnsigned, Open, Timestamp,
};
use serde_json::Value;
use tracing::info;
use crate::{PriceSource, check_response, default_retry};
#[derive(Default, Clone)]
#[allow(clippy::upper_case_acronyms)]
pub struct BRK {
height_to_ohlc: BTreeMap<Height, Vec<OHLCCentsUnsigned>>,
dateindex_to_ohlc: BTreeMap<DateIndex, Vec<OHLCCentsUnsigned>>,
}
const API_URL: &str = "https://bitview.space/api/vecs";
const CHUNK_SIZE: usize = 10_000;
impl BRK {
pub fn get_from_height(&mut self, height: Height) -> Result<OHLCCentsUnsigned> {
let key = height.checked_sub(height % CHUNK_SIZE).unwrap();
#[allow(clippy::map_entry)]
if !self.height_to_ohlc.contains_key(&key)
|| ((key + self.height_to_ohlc.get(&key).unwrap().len()) <= height)
{
self.height_to_ohlc
.insert(key, Self::fetch_height_prices(key)?);
}
self.height_to_ohlc
.get(&key)
.unwrap()
.get(usize::from(height.checked_sub(key).unwrap()))
.cloned()
.ok_or(Error::NotFound("Couldn't find height in BRK".into()))
}
fn fetch_height_prices(height: Height) -> Result<Vec<OHLCCentsUnsigned>> {
default_retry(|_| {
let url = format!(
"{API_URL}/height-to-price-ohlc?from={}&to={}",
height,
height + CHUNK_SIZE
);
info!("Fetching {url} ...");
let bytes = check_response(minreq::get(&url).with_timeout(30).send()?, &url)?;
let body: Value = serde_json::from_slice(&bytes)?;
body.as_array()
.ok_or(Error::Parse("Expected JSON array".into()))?
.iter()
.map(Self::value_to_ohlc)
.collect::<Result<Vec<_>, _>>()
})
}
pub fn get_from_date(&mut self, date: Date) -> Result<OHLCCentsUnsigned> {
let dateindex = DateIndex::try_from(date)?;
let key = dateindex.checked_sub(dateindex % CHUNK_SIZE).unwrap();
#[allow(clippy::map_entry)]
if !self.dateindex_to_ohlc.contains_key(&key)
|| ((key + self.dateindex_to_ohlc.get(&key).unwrap().len()) <= dateindex)
{
self.dateindex_to_ohlc
.insert(key, Self::fetch_date_prices(key)?);
}
self.dateindex_to_ohlc
.get(&key)
.unwrap()
.get(usize::from(dateindex.checked_sub(key).unwrap()))
.cloned()
.ok_or(Error::NotFound("Couldn't find date in BRK".into()))
}
fn fetch_date_prices(dateindex: DateIndex) -> Result<Vec<OHLCCentsUnsigned>> {
default_retry(|_| {
let url = format!(
"{API_URL}/dateindex-to-price-ohlc?from={}&to={}",
dateindex,
dateindex + CHUNK_SIZE
);
info!("Fetching {url}...");
let bytes = check_response(minreq::get(&url).with_timeout(30).send()?, &url)?;
let body: Value = serde_json::from_slice(&bytes)?;
body.as_array()
.ok_or(Error::Parse("Expected JSON array".into()))?
.iter()
.map(Self::value_to_ohlc)
.collect::<Result<Vec<_>, _>>()
})
}
fn value_to_ohlc(value: &Value) -> Result<OHLCCentsUnsigned> {
let ohlc = value
.as_array()
.ok_or(Error::Parse("Expected OHLC array".into()))?;
let get_value = |index: usize| -> Result<_> {
Ok(CentsUnsigned::from(Dollars::from(
ohlc.get(index)
.ok_or(Error::Parse("Missing OHLC value at index".into()))?
.as_f64()
.ok_or(Error::Parse("Invalid OHLC value type".into()))?,
)))
};
Ok(OHLCCentsUnsigned::from((
Open::new(get_value(0)?),
High::new(get_value(1)?),
Low::new(get_value(2)?),
Close::new(get_value(3)?),
)))
}
pub fn ping() -> Result<()> {
minreq::get(API_URL).with_timeout(10).send()?;
Ok(())
}
}
impl PriceSource for BRK {
fn name(&self) -> &'static str {
"BRK"
}
fn get_date(&mut self, date: Date) -> Option<Result<OHLCCentsUnsigned>> {
Some(self.get_from_date(date))
}
fn get_1mn(
&mut self,
_timestamp: Timestamp,
_previous_timestamp: Option<Timestamp>,
) -> Option<Result<OHLCCentsUnsigned>> {
None }
fn get_height(&mut self, height: Height) -> Option<Result<OHLCCentsUnsigned>> {
Some(self.get_from_height(height))
}
fn ping(&self) -> Result<()> {
Self::ping()
}
fn clear(&mut self) {
self.height_to_ohlc.clear();
self.dateindex_to_ohlc.clear();
}
}