use crate::{
error::Error,
response::{Code, Response, SubFunction},
utils, DataFormatIdentifier, Configuration, LengthFormatIdentifier, ModeOfOperation,
ResponseData, Service,
};
use std::{collections::HashSet, sync::LazyLock};
pub static REQUEST_FILE_TRANSFER_NEGATIVES: LazyLock<HashSet<Code>> = LazyLock::new(|| {
HashSet::from([
Code::IncorrectMessageLengthOrInvalidFormat,
Code::ConditionsNotCorrect,
#[cfg(any(feature = "std2020"))]
Code::RequestSequenceError, Code::RequestOutOfRange,
Code::SecurityAccessDenied,
#[cfg(any(feature = "std2020"))]
Code::AuthenticationRequired,
Code::UploadDownloadNotAccepted,
])
});
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub enum RequestFileTransfer {
AddFile {
lfi: LengthFormatIdentifier,
max_block_len: u128,
dfi: DataFormatIdentifier,
},
DeleteFile, ReplaceFile {
lfi: LengthFormatIdentifier,
max_block_len: u128,
dfi: DataFormatIdentifier,
},
ReadFile {
lfi: LengthFormatIdentifier,
max_block_len: u128,
dfi: DataFormatIdentifier,
filesize_or_dir_param_len: u16,
uncompressed_size_or_dir_len: u128,
compressed_size: u128,
},
ReadDir {
lfi: LengthFormatIdentifier,
max_block_len: u128,
dfi: DataFormatIdentifier, filesize_or_dir_param_len: u16,
uncompressed_size_or_dir_len: u128,
},
ResumeFile {
lfi: LengthFormatIdentifier,
max_block_len: u128,
dfi: DataFormatIdentifier,
file_pos: [u8; 8],
},
}
impl From<RequestFileTransfer> for Vec<u8> {
fn from(v: RequestFileTransfer) -> Self {
let mut result = Vec::new();
match v {
RequestFileTransfer::AddFile {
lfi,
max_block_len,
dfi,
} => {
result.push(lfi.into());
result.extend(utils::u128_to_vec(
max_block_len,
lfi.max_number_of_block_length(),
));
result.push(dfi.into());
}
RequestFileTransfer::DeleteFile => {}
RequestFileTransfer::ReplaceFile {
lfi,
max_block_len,
dfi,
} => {
result.push(lfi.into());
result.extend(utils::u128_to_vec(
max_block_len,
lfi.max_number_of_block_length(),
));
result.push(dfi.into());
}
RequestFileTransfer::ReadFile {
lfi,
max_block_len,
dfi,
filesize_or_dir_param_len,
uncompressed_size_or_dir_len,
compressed_size,
} => {
result.push(lfi.into());
result.extend(utils::u128_to_vec(
max_block_len,
lfi.max_number_of_block_length(),
));
result.push(dfi.into());
result.extend(filesize_or_dir_param_len.to_be_bytes());
result.extend(utils::u128_to_vec_fix(uncompressed_size_or_dir_len));
result.extend(utils::u128_to_vec_fix(compressed_size));
}
RequestFileTransfer::ReadDir {
lfi,
max_block_len,
dfi,
filesize_or_dir_param_len,
uncompressed_size_or_dir_len,
} => {
result.push(lfi.into());
result.extend(utils::u128_to_vec(
max_block_len,
lfi.max_number_of_block_length(),
));
result.push(dfi.into());
result.extend(filesize_or_dir_param_len.to_be_bytes());
result.extend(utils::u128_to_vec_fix(uncompressed_size_or_dir_len));
}
RequestFileTransfer::ResumeFile {
lfi,
max_block_len,
dfi,
file_pos,
} => {
result.push(lfi.into());
result.extend(utils::u128_to_vec(
max_block_len,
lfi.max_number_of_block_length(),
));
result.push(dfi.into());
result.extend(file_pos);
}
}
result
}
}
impl ResponseData for RequestFileTransfer {
fn new_response<T: AsRef<[u8]>>(
data: T,
sub_func: Option<u8>,
_: &Configuration,
) -> Result<Response, Error> {
let data = data.as_ref();
match sub_func {
Some(sub_func) => {
let data_len = data.len();
match ModeOfOperation::try_from(sub_func)? {
ModeOfOperation::AddFile => utils::data_length_check(data_len, 2, false)?,
ModeOfOperation::DeleteFile => utils::data_length_check(data_len, 1, true)?,
ModeOfOperation::ReplaceFile => utils::data_length_check(data_len, 2, false)?,
ModeOfOperation::ReadFile => utils::data_length_check(data_len, 2, false)?,
ModeOfOperation::ReadDir => utils::data_length_check(data_len, 2, false)?,
ModeOfOperation::ResumeFile => utils::data_length_check(data_len, 2, false)?,
}
Ok(Response {
service: Service::RequestFileTransfer,
negative: false,
sub_func: Some(SubFunction::new(sub_func)),
data: data.to_vec(),
})
}
None => Err(Error::SubFunctionError(Service::RequestFileTransfer)),
}
}
}
impl TryFrom<(&Response, &Configuration)> for RequestFileTransfer {
type Error = Error;
fn try_from((resp, _): (&Response, &Configuration)) -> Result<Self, Self::Error> {
let service = resp.service();
if service != Service::RequestFileTransfer || resp.sub_func.is_none() {
return Err(Error::ServiceError(service));
}
let sub_func: ModeOfOperation = resp.sub_function().unwrap().function()?;
let data = &resp.data;
let data_len = data.len();
let mut offset = 0;
match sub_func {
ModeOfOperation::AddFile => {
utils::data_length_check(data_len, offset + 1, false)?;
let lfi = LengthFormatIdentifier::try_from(data[offset])?;
offset += 1;
utils::data_length_check(
data_len,
offset + lfi.max_number_of_block_length() + 1,
true,
)?;
let max_block_len =
utils::slice_to_u128(&data[offset..offset + lfi.max_number_of_block_length()]);
offset += lfi.max_number_of_block_length();
let dfi = DataFormatIdentifier::from(data[offset]);
Ok(Self::AddFile {
lfi,
max_block_len,
dfi,
})
}
ModeOfOperation::DeleteFile => Ok(Self::DeleteFile),
ModeOfOperation::ReplaceFile => {
utils::data_length_check(data_len, offset + 1, false)?;
let lfi = LengthFormatIdentifier::try_from(data[offset])?;
offset += 1;
utils::data_length_check(
data_len,
offset + lfi.max_number_of_block_length() + 1,
true,
)?;
let max_block_len =
utils::slice_to_u128(&data[offset..offset + lfi.max_number_of_block_length()]);
offset += lfi.max_number_of_block_length();
let dfi = DataFormatIdentifier::from(data[offset]);
Ok(Self::ReplaceFile {
lfi,
max_block_len,
dfi,
})
}
ModeOfOperation::ReadFile => {
utils::data_length_check(data_len, offset + 1, false)?;
let lfi = LengthFormatIdentifier::try_from(data[offset])?;
offset += 1;
utils::data_length_check(
data_len,
offset + lfi.max_number_of_block_length() + 4,
false,
)?;
let max_block_len =
utils::slice_to_u128(&data[offset..offset + lfi.max_number_of_block_length()]);
offset += lfi.max_number_of_block_length();
let dfi = DataFormatIdentifier::from(data[offset]);
offset += 1;
let filesize_or_dir_param_len =
u16::from_be_bytes([data[offset], data[offset + 1]]);
offset += 2;
utils::data_length_check(
data_len,
offset + filesize_or_dir_param_len as usize + 1,
false,
)?;
let uncompressed_size_or_dir_len = utils::slice_to_u128(
&data[offset..offset + filesize_or_dir_param_len as usize],
);
offset += filesize_or_dir_param_len as usize;
let compressed_size = utils::slice_to_u128(&data[offset..]);
Ok(Self::ReadFile {
lfi,
max_block_len,
dfi,
filesize_or_dir_param_len,
uncompressed_size_or_dir_len,
compressed_size,
})
}
ModeOfOperation::ReadDir => {
utils::data_length_check(data_len, offset + 1, false)?;
let lfi = LengthFormatIdentifier::try_from(data[offset])?;
offset += 1;
utils::data_length_check(
data_len,
offset + lfi.max_number_of_block_length() + 4,
false,
)?;
let max_block_len =
utils::slice_to_u128(&data[offset..offset + lfi.max_number_of_block_length()]);
offset += lfi.max_number_of_block_length();
let dfi = data[offset];
offset += 1;
if dfi != 0x00 {
return Err(Error::InvalidData(hex::encode(data)));
}
let dfi = DataFormatIdentifier(dfi);
let filesize_or_dir_param_len =
u16::from_be_bytes([data[offset], data[offset + 1]]);
offset += 2;
let uncompressed_size_or_dir_len = utils::slice_to_u128(
&data[offset..offset + filesize_or_dir_param_len as usize],
);
Ok(Self::ReadDir {
lfi,
max_block_len,
dfi,
filesize_or_dir_param_len,
uncompressed_size_or_dir_len,
})
}
ModeOfOperation::ResumeFile => {
utils::data_length_check(data_len, offset + 1, false)?;
let lfi = LengthFormatIdentifier::try_from(data[offset])?;
offset += 1;
utils::data_length_check(
data_len,
offset + lfi.max_number_of_block_length() + 9,
true,
)?;
let max_block_len =
utils::slice_to_u128(&data[offset..offset + lfi.max_number_of_block_length()]);
offset += lfi.max_number_of_block_length();
let dfi = DataFormatIdentifier::from(data[offset]);
offset += 1;
let file_pos = <[u8; 8]>::try_from(&data[offset..])
.map_err(|_| Error::InvalidData(hex::encode(data)))?;
Ok(Self::ResumeFile {
lfi,
max_block_len,
dfi,
file_pos,
})
}
}
}
}