use super::{CacheKey, RandomAccessFileReader, ZstdFrameCache};
use crate::blocks::{Tipset, TipsetKey};
use crate::chain::FilecoinSnapshotMetadata;
use crate::utils::io::EitherMmapOrRandomAccessFile;
use cid::Cid;
use fvm_ipld_blockstore::Blockstore;
use itertools::Either;
use positioned_io::ReadAt;
use std::{
borrow::Cow,
io::{Error, ErrorKind, Read, Result},
path::{Path, PathBuf},
};
#[derive(derive_more::From)]
pub enum AnyCar<ReaderT> {
Plain(super::PlainCar<ReaderT>),
Forest(super::ForestCar<ReaderT>),
#[from(skip)]
Memory(super::PlainCar<Vec<u8>>),
}
impl<ReaderT: RandomAccessFileReader> AnyCar<ReaderT> {
pub fn new(reader: ReaderT) -> Result<Self> {
if let Ok(validation_result) = super::ForestCar::validate_car(&reader) {
return Ok(
super::ForestCar::new_from_validation_result(reader, validation_result)?.into(),
);
}
if let Ok(decompressed) = zstd::stream::decode_all(positioned_io::Cursor::new(&reader))
&& let Ok(mem_car) = super::PlainCar::new(decompressed)
{
return Ok(AnyCar::Memory(mem_car));
}
if let Ok(plain_car) = super::PlainCar::new(reader) {
return Ok(plain_car.into());
}
Err(Error::new(
ErrorKind::InvalidData,
"input not recognized as any kind of CAR data (.car, .car.zst, .forest.car)",
))
}
pub fn metadata(&self) -> Option<&FilecoinSnapshotMetadata> {
match self {
AnyCar::Forest(forest) => forest.metadata(),
AnyCar::Plain(plain) => plain.metadata(),
AnyCar::Memory(mem) => mem.metadata(),
}
}
pub fn heaviest_tipset_key(&self) -> TipsetKey {
match self {
AnyCar::Forest(forest) => forest.heaviest_tipset_key(),
AnyCar::Plain(plain) => plain.heaviest_tipset_key(),
AnyCar::Memory(mem) => mem.heaviest_tipset_key(),
}
}
pub fn heaviest_tipset(&self) -> anyhow::Result<Tipset> {
match self {
AnyCar::Forest(forest) => forest.heaviest_tipset(),
AnyCar::Plain(plain) => plain.heaviest_tipset(),
AnyCar::Memory(mem) => mem.heaviest_tipset(),
}
}
pub fn variant(&self) -> Cow<'static, str> {
match self {
AnyCar::Forest(_) => "ForestCARv1.zst".into(),
AnyCar::Plain(car) => format!("CARv{}", car.version()).into(),
AnyCar::Memory(car) => format!("CARv{}.zst", car.version()).into(),
}
}
pub fn into_dyn(self) -> AnyCar<Box<dyn super::RandomAccessFileReader>> {
match self {
AnyCar::Forest(f) => AnyCar::Forest(f.into_dyn()),
AnyCar::Plain(p) => AnyCar::Plain(p.into_dyn()),
AnyCar::Memory(m) => AnyCar::Memory(m),
}
}
pub fn with_cache(self, cache: ZstdFrameCache, key: CacheKey) -> Self {
match self {
AnyCar::Forest(f) => AnyCar::Forest(f.with_cache(cache, key)),
AnyCar::Plain(p) => AnyCar::Plain(p),
AnyCar::Memory(m) => AnyCar::Memory(m),
}
}
pub fn index_size_bytes(&self) -> Option<u64> {
match self {
Self::Forest(car) => Some(car.index_size_bytes()),
_ => None,
}
}
pub fn get_reader(&self, k: Cid) -> anyhow::Result<Option<impl Read>> {
match self {
Self::Forest(car) => Ok(car.get_reader(k)?.map(Either::Left)),
Self::Plain(car) => Ok(car.get_reader(k).map(|r| Either::Right(Either::Left(r)))),
Self::Memory(car) => Ok(car.get_reader(k).map(|r| Either::Right(Either::Right(r)))),
}
}
}
impl TryFrom<&'static [u8]> for AnyCar<&'static [u8]> {
type Error = std::io::Error;
fn try_from(bytes: &'static [u8]) -> std::io::Result<Self> {
Ok(AnyCar::Plain(super::PlainCar::new(bytes)?))
}
}
impl TryFrom<&Path> for AnyCar<EitherMmapOrRandomAccessFile> {
type Error = std::io::Error;
fn try_from(path: &Path) -> std::io::Result<Self> {
AnyCar::new(EitherMmapOrRandomAccessFile::open(path)?)
}
}
impl TryFrom<&PathBuf> for AnyCar<EitherMmapOrRandomAccessFile> {
type Error = std::io::Error;
fn try_from(path: &PathBuf) -> std::io::Result<Self> {
Self::try_from(path.as_path())
}
}
impl<ReaderT> Blockstore for AnyCar<ReaderT>
where
ReaderT: ReadAt,
{
fn get(&self, k: &Cid) -> anyhow::Result<Option<Vec<u8>>> {
match self {
AnyCar::Forest(forest) => forest.get(k),
AnyCar::Plain(plain) => plain.get(k),
AnyCar::Memory(mem) => mem.get(k),
}
}
fn put_keyed(&self, k: &Cid, block: &[u8]) -> anyhow::Result<()> {
match self {
AnyCar::Forest(forest) => forest.put_keyed(k, block),
AnyCar::Plain(plain) => plain.put_keyed(k, block),
AnyCar::Memory(mem) => mem.put_keyed(k, block),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::networks::{calibnet, mainnet};
#[test]
fn forest_any_load_calibnet() {
let forest_car = AnyCar::new(calibnet::DEFAULT_GENESIS).unwrap();
assert!(forest_car.has(&calibnet::GENESIS_CID).unwrap());
}
#[test]
fn forest_any_load_calibnet_zstd() {
let data = zstd::encode_all(calibnet::DEFAULT_GENESIS, 3).unwrap();
let forest_car = AnyCar::new(data).unwrap();
assert!(forest_car.has(&calibnet::GENESIS_CID).unwrap());
}
#[test]
fn forest_any_load_mainnet() {
let forest_car = AnyCar::new(mainnet::DEFAULT_GENESIS).unwrap();
assert!(forest_car.has(&mainnet::GENESIS_CID).unwrap());
}
#[test]
fn forest_any_load_mainnet_zstd() {
let data = zstd::encode_all(mainnet::DEFAULT_GENESIS, 3).unwrap();
let forest_car = AnyCar::new(data).unwrap();
assert!(forest_car.has(&mainnet::GENESIS_CID).unwrap());
}
}