shrupl/src/sharry/file/uploading.rs

134 lines
3.2 KiB
Rust
Raw Normal View History

use std::{
2025-06-05 22:08:24 +00:00
ffi::OsStr,
fs,
io::{self, Read, Seek, SeekFrom},
path::PathBuf,
};
use log::debug;
use serde::{Deserialize, Serialize};
2025-06-05 17:37:35 +00:00
use ureq::http::{HeaderValue, StatusCode};
2025-06-06 15:21:49 +00:00
use super::{Alias, SharryAlias, SharryFile};
#[derive(Serialize, Deserialize, Debug)]
pub struct FileUploading {
2025-06-04 16:58:18 +00:00
path: PathBuf,
2025-06-06 23:42:18 +00:00
size: u64,
2025-06-04 16:58:18 +00:00
uri: String,
2025-06-06 23:42:18 +00:00
offset: u64,
}
#[derive(Debug)]
pub enum UploadError {
FileIO(io::Error),
Request,
ResponseStatus,
ResponseOffset,
}
pub enum ChunkState {
Ok(FileUploading),
Err(FileUploading, UploadError),
2025-06-04 16:54:03 +00:00
Finished(PathBuf),
}
2025-06-06 15:21:49 +00:00
impl std::fmt::Display for FileUploading {
2025-06-05 01:13:54 +00:00
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"Uploading ({:?}, {}, {})",
2025-06-05 01:45:25 +00:00
self.path.display(),
self.size,
self.offset
2025-06-05 01:13:54 +00:00
)
}
}
impl FileUploading {
2025-06-06 23:42:18 +00:00
pub(super) fn new(path: PathBuf, size: u64, uri: String) -> Self {
2025-06-04 16:58:18 +00:00
Self {
path,
size,
uri,
offset: 0,
}
}
2025-06-06 23:42:18 +00:00
pub fn get_offset(&self) -> u64 {
self.offset
}
2025-06-06 23:42:18 +00:00
fn read_chunk(&self, chunk_size: usize) -> io::Result<Vec<u8>> {
2025-06-05 22:08:24 +00:00
let mut f = fs::File::open(&self.path)?;
2025-06-06 23:42:18 +00:00
f.seek(SeekFrom::Start(self.offset))?;
let mut bytes = vec![0; chunk_size];
let read_len = f.read(&mut bytes)?;
bytes.truncate(read_len);
Ok(bytes)
}
pub fn upload_chunk(
mut self,
http: &ureq::Agent,
alias: &Alias,
chunk_size: usize,
) -> ChunkState {
let chunk = match self.read_chunk(chunk_size) {
Err(e) => return ChunkState::Err(self, UploadError::FileIO(e)),
Ok(value) => value,
};
let Ok(res) = (http.patch(&self.uri))
.sharry_header(alias)
.header("Upload-Offset", self.offset)
.send(&chunk)
else {
return ChunkState::Err(self, UploadError::Request);
};
if res.status() != StatusCode::NO_CONTENT {
return ChunkState::Err(self, UploadError::ResponseStatus);
}
let Some(Ok(Ok(res_offset))) = (res.headers().get("Upload-Offset"))
.map(HeaderValue::to_str)
2025-06-06 23:42:18 +00:00
.map(|v| v.map(str::parse::<u64>))
else {
return ChunkState::Err(self, UploadError::ResponseOffset);
};
2025-06-06 23:42:18 +00:00
// convert chunk.len() into an `u64`
//
// BOOKMARK this might panic on platforms where `usize` > 64 bit.
// Also whoa, you've sent more than 2 EiB ... in ONE chunk.
// Maybe just chill?
let chunk_len = u64::try_from(chunk.len())
.unwrap_or_else(|e| panic!("usize={} did not fit into u64: {}", chunk.len(), e));
if self.offset + chunk_len != res_offset {
return ChunkState::Err(self, UploadError::ResponseOffset);
}
self.offset = res_offset;
if self.offset == self.size {
2025-06-04 16:54:03 +00:00
return ChunkState::Finished(self.path);
}
ChunkState::Ok(self)
}
}
2025-06-06 15:21:49 +00:00
impl<'t> SharryFile<'t> for FileUploading {
2025-06-06 23:42:18 +00:00
fn get_name(&'t self) -> &'t str {
2025-06-06 15:21:49 +00:00
<Self as SharryFile>::extract_file_name(&self.path)
}
2025-06-06 23:42:18 +00:00
fn get_size(&self) -> u64 {
self.size
2025-06-06 15:21:49 +00:00
}
}