Files
brk/crates/brk_fetcher/src/brk.rs
T
2026-01-02 19:08:20 +01:00

161 lines
4.8 KiB
Rust

use std::collections::BTreeMap;
use brk_error::{Error, Result};
use brk_types::{Cents, CheckedSub, Close, Date, DateIndex, Dollars, Height, High, Low, OHLCCents, Open, Timestamp};
use log::info;
use serde_json::Value;
use crate::{PriceSource, default_retry};
#[derive(Default, Clone)]
#[allow(clippy::upper_case_acronyms)]
pub struct BRK {
height_to_ohlc: BTreeMap<Height, Vec<OHLCCents>>,
dateindex_to_ohlc: BTreeMap<DateIndex, Vec<OHLCCents>>,
}
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<OHLCCents> {
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<OHLCCents>> {
default_retry(|_| {
let url = format!(
"{API_URL}/height-to-price-ohlc?from={}&to={}",
height,
height + CHUNK_SIZE
);
info!("Fetching {url} ...");
let body: Value =
serde_json::from_slice(minreq::get(url).with_timeout(30).send()?.as_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<OHLCCents> {
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<OHLCCents>> {
default_retry(|_| {
let url = format!(
"{API_URL}/dateindex-to-price-ohlc?from={}&to={}",
dateindex,
dateindex + CHUNK_SIZE
);
info!("Fetching {url}...");
let body: Value =
serde_json::from_slice(minreq::get(url).with_timeout(30).send()?.as_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<OHLCCents> {
let ohlc = value
.as_array()
.ok_or(Error::Parse("Expected OHLC array".into()))?;
let get_value = |index: usize| -> Result<_> {
Ok(Cents::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(OHLCCents::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<OHLCCents>> {
Some(self.get_from_date(date))
}
fn get_1mn(
&mut self,
_timestamp: Timestamp,
_previous_timestamp: Option<Timestamp>,
) -> Option<Result<OHLCCents>> {
None // BRK doesn't support timestamp-based queries
}
fn get_height(&mut self, height: Height) -> Option<Result<OHLCCents>> {
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();
}
}