use crate::{Arc, Data};
pub trait Storage: Send + Sync {
fn size(&self) -> u64;
fn read(&self, start: u64, data: &mut [u8]);
fn write(&mut self, start: u64, data: &[u8]);
fn write_vec(&mut self, start: u64, data: Vec<u8>) {
let len = data.len();
let d = Arc::new(data);
self.write_data(start, d, 0, len);
}
fn write_data(&mut self, start: u64, data: Data, off: usize, len: usize) {
self.write(start, &data[off..off + len]);
}
fn commit(&mut self, size: u64);
fn write_u64(&mut self, start: u64, value: u64) {
self.write(start, &value.to_le_bytes());
}
fn read_u64(&self, start: u64) -> u64 {
let mut bytes = [0; 8];
self.read(start, &mut bytes);
u64::from_le_bytes(bytes)
}
}
#[derive(Default)]
pub struct MemFile {
v: Mutex<Vec<u8>>,
}
impl MemFile {
pub fn new() -> Box<Self> {
Box::<Self>::default()
}
}
impl Storage for MemFile {
fn size(&self) -> u64 {
let v = self.v.lock().unwrap();
v.len() as u64
}
fn read(&self, off: u64, bytes: &mut [u8]) {
let off = off as usize;
let len = bytes.len();
let mut v = self.v.lock().unwrap();
if off + len > v.len() {
v.resize(off + len, 0);
}
bytes.copy_from_slice(&v[off..off + len]);
}
fn write(&mut self, off: u64, bytes: &[u8]) {
let off = off as usize;
let len = bytes.len();
let mut v = self.v.lock().unwrap();
if off + len > v.len() {
v.resize(off + len, 0);
}
v[off..off + len].copy_from_slice(bytes);
}
fn commit(&mut self, size: u64) {
let mut v = self.v.lock().unwrap();
v.resize(size as usize, 0);
}
}
use crate::Mutex;
use std::{fs, fs::OpenOptions, io::Read, io::Seek, io::SeekFrom, io::Write};
pub struct SimpleFileStorage {
file: Mutex<fs::File>,
}
impl SimpleFileStorage {
pub fn new(filename: &str) -> Box<Self> {
Box::new(Self {
file: Mutex::new(
OpenOptions::new()
.read(true)
.write(true)
.create(true)
.open(filename)
.unwrap(),
),
})
}
}
impl Storage for SimpleFileStorage {
fn size(&self) -> u64 {
let mut f = self.file.lock().unwrap();
f.seek(SeekFrom::End(0)).unwrap()
}
fn read(&self, off: u64, bytes: &mut [u8]) {
let mut f = self.file.lock().unwrap();
f.seek(SeekFrom::Start(off)).unwrap();
let _ = f.read(bytes).unwrap();
}
fn write(&mut self, off: u64, bytes: &[u8]) {
let mut f = self.file.lock().unwrap();
#[cfg(not(any(target_os = "windows", target_os = "linux")))]
{
let size = f.seek(SeekFrom::End(0)).unwrap();
if off > size {
f.set_len(off).unwrap();
}
}
f.seek(SeekFrom::Start(off)).unwrap();
let _ = f.write(bytes).unwrap();
}
fn commit(&mut self, size: u64) {
let f = self.file.lock().unwrap();
f.set_len(size).unwrap();
f.sync_all().unwrap();
}
}
#[allow(clippy::vec_box)]
pub struct MultiFileStorage {
filename: String,
files: Mutex<Vec<Box<SimpleFileStorage>>>,
}
impl MultiFileStorage {
pub fn new(filename: &str) -> Box<Self> {
Box::new(Self {
filename: filename.to_string(),
files: Mutex::new(Vec::new()),
})
}
fn get_file(&self) -> Box<SimpleFileStorage> {
if let Some(f) = self.files.lock().unwrap().pop() {
f
} else {
SimpleFileStorage::new(&self.filename)
}
}
fn put_file(&self, f: Box<SimpleFileStorage>) {
self.files.lock().unwrap().push(f);
}
}
impl Storage for MultiFileStorage {
fn size(&self) -> u64 {
let f = self.get_file();
let result = f.size();
self.put_file(f);
result
}
fn read(&self, off: u64, bytes: &mut [u8]) {
let f = self.get_file();
f.read(off, bytes);
self.put_file(f);
}
fn write(&mut self, off: u64, bytes: &[u8]) {
let mut f = self.get_file();
f.write(off, bytes);
self.put_file(f);
}
fn commit(&mut self, size: u64) {
let mut f = self.get_file();
f.commit(size);
self.put_file(f);
}
}