flux-sftp/src/sftp.rs

257 lines
8 KiB
Rust
Raw Normal View History

use std::{collections::HashMap, io::SeekFrom, os::unix::fs::MetadataExt};
2025-07-05 20:09:01 +05:00
use chrono::{Local, TimeZone};
use regex::Regex;
use russh_sftp::{protocol::{Attrs, Data, File, FileAttributes, Handle as SftpHandle, Name, OpenFlags, Status, StatusCode}, server::Handler as SftpHandler};
2025-07-05 20:09:01 +05:00
use tokio::{fs::{self, OpenOptions, ReadDir}, io::{AsyncReadExt, AsyncSeekExt}};
enum Handle {
Dir(ReadDir),
File(fs::File)
}
2025-07-05 20:09:01 +05:00
pub struct SftpSession {
jail_dir: String,
cwd: String,
handles: HashMap<String, Handle>
2025-07-05 20:09:01 +05:00
}
impl SftpSession {
pub fn new(jail_dir: String) -> Self {
SftpSession { jail_dir, cwd: String::from("/"), handles: HashMap::new() }
}
}
impl SftpHandler for SftpSession {
type Error = StatusCode;
fn unimplemented(&self) -> Self::Error {
StatusCode::OpUnsupported
}
async fn realpath(
&mut self,
id: u32,
path: String,
) -> Result<Name, Self::Error> {
println!("realpath called, path: {}", path);
let re_1 = Regex::new(r"/[^/]+/\.\.").unwrap();
self.cwd = re_1.replace_all(&path, "").to_string();
while re_1.is_match(&self.cwd) {
self.cwd = re_1.replace_all(&self.cwd, "").to_string();
}
let re_2 = Regex::new(r"/\.").unwrap();
self.cwd = re_2.replace_all(&self.cwd, "").to_string();
if self.cwd == "." || self.cwd == "" {
self.cwd = String::from("/");
}
Ok(Name { id, files: vec![File::dummy(&self.cwd)] })
}
async fn open(
&mut self,
id: u32,
filename: String,
pflags: OpenFlags,
_attrs: FileAttributes,
) -> Result<SftpHandle, Self::Error> {
println!("open called, path: {}", filename);
let filename = format!("{}/{}", self.jail_dir, filename);
let mut options = OpenOptions::new();
if pflags.contains(OpenFlags::READ){
options.read(true);
}
if pflags.contains(OpenFlags::WRITE){
options.write(true);
}
if pflags.contains(OpenFlags::APPEND){
options.append(true);
}
if pflags.contains(OpenFlags::CREATE){
options.create(true);
}
if pflags.contains(OpenFlags::TRUNCATE){
options.truncate(true);
}
match options.open(&filename).await {
Ok(file) => {
self.handles.insert(filename.clone(), Handle::File(file));
Ok(SftpHandle { id, handle: filename })
}
Err(_) => Err(StatusCode::NoSuchFile)
}
}
async fn read(
&mut self,
id: u32,
handle: String,
offset: u64,
len: u32,
) -> Result<Data, Self::Error> {
if let Handle::File(file) = self.handles.get_mut(&handle).unwrap() {
let mut buf = vec![0u8; len as usize];
match file.seek(SeekFrom::Start(offset)).await {
Ok(_) => {
match file.read(&mut buf).await {
Ok(bytes) => {
if bytes != 0 {
buf.truncate(bytes);
Ok(Data { id, data: buf })
}
else {
Err(StatusCode::Eof)
}
}
Err(e) => {
println!("Error in reading from offset in file: {}", e);
Err(StatusCode::Failure)
2025-07-05 20:09:01 +05:00
}
}
}
Err(e) => {
println!("Error in seeking offset in file: {}", e);
Err(StatusCode::Failure)
}
2025-07-05 20:09:01 +05:00
}
}
else {
Err(StatusCode::Ok)
}
2025-07-05 20:09:01 +05:00
}
async fn opendir(
&mut self,
id: u32,
path: String,
) -> Result<SftpHandle, Self::Error> {
println!("opendir called");
2025-07-05 20:09:01 +05:00
let path = format!("{}/{}", self.jail_dir, path);
match fs::read_dir(&path).await {
Ok(entries) => {
self.handles.insert(path.clone(), Handle::Dir(entries));
Ok(SftpHandle { id, handle: path })
2025-07-05 20:09:01 +05:00
}
Err(e) => {
println!("Error in reading dir: {}", e);
Err(StatusCode::NoSuchFile)
}
}
}
async fn readdir(
&mut self,
id: u32,
handle: String,
) -> Result<Name, Self::Error> {
println!("readdir called");
if let Handle::Dir(handle) = self.handles.get_mut(&handle).unwrap() {
match handle.next_entry().await {
Ok(Some(entry)) => {
let metadata = entry.metadata().await.unwrap();
let dt = Local.timestamp_opt(metadata.mtime(), 0).unwrap();
let longname = format!("{} {} {}", metadata.size(), dt.format("%b %e %Y"), entry.file_name().to_string_lossy());
Ok(Name { id, files: vec![
File {
filename: entry.file_name().to_string_lossy().into(),
longname: longname,
attrs: FileAttributes {
size: Some(metadata.size()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}
2025-07-05 20:09:01 +05:00
}
] })
}
Ok(None) => Err(StatusCode::Eof),
Err(e) => {
println!("Error listing file: {}", e);
Err(StatusCode::Failure)
}
2025-07-05 20:09:01 +05:00
}
}
else {
println!("handle is not a dirhandle");
Err(StatusCode::Failure)
}
2025-07-05 20:09:01 +05:00
}
async fn close(
&mut self,
id: u32,
handle: String,
) -> Result<Status, Self::Error> {
println!("close called");
2025-07-05 20:09:01 +05:00
self.handles.remove(&handle);
Ok(Status {
id,
status_code: StatusCode::Ok,
error_message: "Ok".to_string(),
language_tag: "en-US".to_string(),
})
}
async fn stat(
&mut self,
id: u32,
path: String,
) -> Result<Attrs, Self::Error> {
println!("stat called");
let path = format!("{}/{}", self.jail_dir, path);
match fs::metadata(path).await {
Ok(metadata) => Ok(Attrs { id, attrs: FileAttributes {
size: Some(metadata.size()),
uid: Some(metadata.uid()),
user: None,
gid: Some(metadata.gid()),
group: None,
permissions: Some(metadata.mode()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}}),
Err(_) => Err(StatusCode::NoSuchFile)
}
}
async fn lstat(
&mut self,
id: u32,
path: String,
) -> Result<Attrs, Self::Error> {
println!("lstat called");
let path = format!("{}/{}", self.jail_dir, path);
match fs::symlink_metadata(path).await {
Ok(metadata) => Ok(Attrs { id, attrs: FileAttributes {
size: Some(metadata.size()),
uid: Some(metadata.uid()),
user: None,
gid: Some(metadata.gid()),
group: None,
permissions: Some(metadata.mode()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32)
}}),
Err(_) => Err(StatusCode::OpUnsupported)
}
}
async fn fstat(
&mut self,
id: u32,
handle: String,
) -> Result<Attrs, Self::Error> {
println!("fstat called");
self.stat(id, handle).await
}
2025-07-05 20:09:01 +05:00
}