use crate::server::ControlChanMsg;
use crate::server::controlchan::{
Reply, ReplyCode,
error::ControlChanError,
handler::{CommandContext, CommandHandler},
};
use async_trait::async_trait;
use std::{path::PathBuf, sync::Arc};
use unftp_core::auth::UserDetail;
use unftp_core::storage::{Metadata, StorageBackend};
#[derive(Debug)]
pub struct Rnto {
path: PathBuf,
}
impl Rnto {
pub fn new(path: PathBuf) -> Self {
Rnto { path }
}
}
#[async_trait]
impl<Storage, User> CommandHandler<Storage, User> for Rnto
where
User: UserDetail + 'static,
Storage: StorageBackend<User> + 'static,
Storage::Metadata: Metadata,
{
#[tracing_attributes::instrument]
async fn handle(&self, args: CommandContext<Storage, User>) -> Result<Reply, ControlChanError> {
let CommandContext {
logger,
session,
tx_control_chan,
..
} = args;
let mut session = session.lock().await;
let storage = Arc::clone(&session.storage);
let (from, to) = match session.rename_from.take() {
Some(from) => {
let to = session.cwd.join(self.path.clone());
(from, to)
}
None => return Ok(Reply::new(ReplyCode::TransientFileError, "Please tell me what file you want to rename first")),
};
let user = (*session.user).as_ref().unwrap();
let old_path = from.to_string_lossy().to_string();
let new_path = to.to_string_lossy().to_string();
match storage.rename(user, &from, &to).await {
Ok(_) => {
slog::info!(logger, "RNTO: Successfully renamed {:?} to {:?}", from, to);
if let Err(err) = tx_control_chan.send(ControlChanMsg::RenameSuccess { old_path, new_path }).await {
slog::warn!(logger, "RNTO: Could not send internal message to notify of RNTO success: {}", err);
}
}
Err(err) => {
if let Err(err) = tx_control_chan.send(ControlChanMsg::StorageError(err)).await {
slog::warn!(logger, "RNTO: Could not send internal message to notify of RNTO failure: {}", err);
}
}
}
Ok(Reply::none())
}
}