2025-06-04 13:25:00 +00:00
|
|
|
use std::{
|
2025-06-05 17:37:35 +00:00
|
|
|
fmt::{Debug, Display},
|
2025-06-04 13:25:00 +00:00
|
|
|
fs::File,
|
|
|
|
|
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-04 13:25:00 +00:00
|
|
|
|
|
|
|
|
use super::{Alias, SharryAlias};
|
|
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize, Debug)]
|
|
|
|
|
pub struct FileUploading {
|
2025-06-04 16:58:18 +00:00
|
|
|
path: PathBuf,
|
|
|
|
|
size: usize,
|
|
|
|
|
uri: String,
|
|
|
|
|
offset: usize,
|
2025-06-04 13:25:00 +00:00
|
|
|
}
|
|
|
|
|
|
2025-06-04 16:44:16 +00:00
|
|
|
#[derive(Debug)]
|
2025-06-04 13:25:00 +00:00
|
|
|
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-04 13:25:00 +00:00
|
|
|
}
|
|
|
|
|
|
2025-06-05 01:13:54 +00:00
|
|
|
impl Display for FileUploading {
|
|
|
|
|
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
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-04 13:25:00 +00:00
|
|
|
impl FileUploading {
|
2025-06-04 16:58:18 +00:00
|
|
|
pub(super) fn new(path: PathBuf, size: usize, uri: String) -> Self {
|
|
|
|
|
Self {
|
|
|
|
|
path,
|
|
|
|
|
size,
|
|
|
|
|
uri,
|
|
|
|
|
offset: 0,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-04 13:25:00 +00:00
|
|
|
fn read_chunk(&self, chunk_size: usize) -> io::Result<Vec<u8>> {
|
|
|
|
|
let offset = u64::try_from(self.offset).map_err(io::Error::other)?;
|
|
|
|
|
|
|
|
|
|
let mut f = File::open(&self.path)?;
|
|
|
|
|
f.seek(SeekFrom::Start(offset))?;
|
|
|
|
|
|
|
|
|
|
let mut bytes = vec![0; chunk_size];
|
|
|
|
|
let read_len = f.read(&mut bytes)?;
|
|
|
|
|
bytes.truncate(read_len);
|
|
|
|
|
|
|
|
|
|
Ok(bytes)
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-05 17:37:35 +00:00
|
|
|
pub fn file_name(&self) -> &str {
|
|
|
|
|
self.path.file_name().unwrap().to_str().unwrap()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn offset<T>(&self) -> T
|
|
|
|
|
where
|
|
|
|
|
T: TryFrom<usize>,
|
|
|
|
|
<T as TryFrom<usize>>::Error: Debug,
|
|
|
|
|
{
|
|
|
|
|
self.offset.try_into().unwrap()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn size<T>(&self) -> T
|
|
|
|
|
where
|
|
|
|
|
T: TryFrom<usize>,
|
|
|
|
|
<T as TryFrom<usize>>::Error: Debug,
|
|
|
|
|
{
|
|
|
|
|
self.size.try_into().unwrap()
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-04 13:25:00 +00:00
|
|
|
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)
|
|
|
|
|
.map(|v| v.map(str::parse::<usize>))
|
|
|
|
|
else {
|
|
|
|
|
return ChunkState::Err(self, UploadError::ResponseOffset);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if self.offset + chunk.len() != res_offset {
|
|
|
|
|
return ChunkState::Err(self, UploadError::ResponseOffset);
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-04 16:44:16 +00:00
|
|
|
self.offset = res_offset;
|
2025-06-04 13:25:00 +00:00
|
|
|
|
|
|
|
|
if self.offset == self.size {
|
2025-06-04 16:54:03 +00:00
|
|
|
return ChunkState::Finished(self.path);
|
2025-06-04 13:25:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ChunkState::Ok(self)
|
|
|
|
|
}
|
|
|
|
|
}
|