stats opening and reading file, very poggers

This commit is contained in:
RafayAhmad7548 2025-07-06 06:33:59 +05:00
parent 611f29f325
commit 34db55a70c
4 changed files with 224 additions and 42 deletions

39
Cargo.lock generated
View file

@ -65,6 +65,15 @@ dependencies = [
"zerocopy", "zerocopy",
] ]
[[package]]
name = "aho-corasick"
version = "1.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
dependencies = [
"memchr",
]
[[package]] [[package]]
name = "android-tzdata" name = "android-tzdata"
version = "0.1.1" version = "0.1.1"
@ -535,6 +544,7 @@ name = "flux-sftp"
version = "0.1.0" version = "0.1.0"
dependencies = [ dependencies = [
"chrono", "chrono",
"regex",
"russh", "russh",
"russh-sftp", "russh-sftp",
"tokio", "tokio",
@ -1222,6 +1232,35 @@ dependencies = [
"bitflags", "bitflags",
] ]
[[package]]
name = "regex"
version = "1.11.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191"
dependencies = [
"aho-corasick",
"memchr",
"regex-automata",
"regex-syntax",
]
[[package]]
name = "regex-automata"
version = "0.4.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908"
dependencies = [
"aho-corasick",
"memchr",
"regex-syntax",
]
[[package]]
name = "regex-syntax"
version = "0.8.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
[[package]] [[package]]
name = "rfc6979" name = "rfc6979"
version = "0.4.0" version = "0.4.0"

View file

@ -5,6 +5,7 @@ edition = "2024"
[dependencies] [dependencies]
chrono = "0.4.41" chrono = "0.4.41"
regex = "1.11.1"
russh = "0.52.1" russh = "0.52.1"
russh-sftp = "2.1.1" russh-sftp = "2.1.1"
tokio = { version = "1.45.1", features = ["full"] } tokio = { version = "1.45.1", features = ["full"] }

View file

@ -67,7 +67,7 @@ impl SshHandler for SshSession {
) -> Result<(), Self::Error> { ) -> Result<(), Self::Error> {
if name == "sftp" { if name == "sftp" {
session.channel_success(channel_id)?; session.channel_success(channel_id)?;
let jail_dir = format!("/srv/sftp/{}", self.user.take().unwrap()); let jail_dir = format!("/srv/sftp/{}", self.user.as_ref().unwrap());
let sftp_handler = SftpSession::new(jail_dir); let sftp_handler = SftpSession::new(jail_dir);
russh_sftp::server::run(self.channel.take().ok_or(Self::Error::WrongChannel)?.into_stream(), sftp_handler).await; russh_sftp::server::run(self.channel.take().ok_or(Self::Error::WrongChannel)?.into_stream(), sftp_handler).await;
} }

View file

@ -1,14 +1,20 @@
use std::{collections::HashMap, os::unix::fs::MetadataExt}; use std::{collections::HashMap, io::SeekFrom, os::unix::fs::MetadataExt};
use chrono::{Local, TimeZone}; use chrono::{Local, TimeZone};
use russh_sftp::{protocol::{File, FileAttributes, Handle, Name, Status, StatusCode}, server::Handler as SftpHandler}; use regex::Regex;
use russh_sftp::{protocol::{Attrs, Data, File, FileAttributes, Handle as SftpHandle, Name, OpenFlags, Status, StatusCode}, server::Handler as SftpHandler};
use tokio::fs::{self, ReadDir}; use tokio::{fs::{self, OpenOptions, ReadDir}, io::{AsyncReadExt, AsyncSeekExt}};
enum Handle {
Dir(ReadDir),
File(fs::File)
}
pub struct SftpSession { pub struct SftpSession {
jail_dir: String, jail_dir: String,
cwd: String, cwd: String,
handles: HashMap<String, ReadDir> handles: HashMap<String, Handle>
} }
impl SftpSession { impl SftpSession {
@ -29,34 +35,107 @@ impl SftpHandler for SftpSession {
id: u32, id: u32,
path: String, path: String,
) -> Result<Name, Self::Error> { ) -> Result<Name, Self::Error> {
let path_parts = path.split('/'); println!("realpath called, path: {}", path);
for path_part in path_parts {
match path_part { let re_1 = Regex::new(r"/[^/]+/\.\.").unwrap();
".." => { self.cwd = re_1.replace_all(&path, "").to_string();
if self.cwd != "/" { while re_1.is_match(&self.cwd) {
if let Some(pos) = self.cwd.rfind('/') { self.cwd = re_1.replace_all(&self.cwd, "").to_string();
self.cwd.truncate(pos); }
}
} let re_2 = Regex::new(r"/\.").unwrap();
}, self.cwd = re_2.replace_all(&self.cwd, "").to_string();
"." => {},
_ => self.cwd.push_str(&format!("/{}", path_part)) if self.cwd == "." || self.cwd == "" {
} self.cwd = String::from("/");
} }
Ok(Name { id, files: vec![File::dummy(&self.cwd)] }) 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)
}
}
}
Err(e) => {
println!("Error in seeking offset in file: {}", e);
Err(StatusCode::Failure)
}
}
}
else {
Err(StatusCode::Ok)
}
}
async fn opendir( async fn opendir(
&mut self, &mut self,
id: u32, id: u32,
path: String, path: String,
) -> Result<Handle, Self::Error> { ) -> Result<SftpHandle, Self::Error> {
println!("opendir called");
let path = format!("{}/{}", self.jail_dir, path); let path = format!("{}/{}", self.jail_dir, path);
match fs::read_dir(&path).await { match fs::read_dir(&path).await {
Ok(entries) => { Ok(entries) => {
self.handles.insert(path.clone(), entries); self.handles.insert(path.clone(), Handle::Dir(entries));
Ok(Handle { id, handle: path }) Ok(SftpHandle { id, handle: path })
} }
Err(e) => { Err(e) => {
println!("Error in reading dir: {}", e); println!("Error in reading dir: {}", e);
@ -70,30 +149,37 @@ impl SftpHandler for SftpSession {
id: u32, id: u32,
handle: String, handle: String,
) -> Result<Name, Self::Error> { ) -> Result<Name, Self::Error> {
match self.handles.get_mut(&handle).unwrap().next_entry().await { println!("readdir called");
Ok(Some(entry)) => { if let Handle::Dir(handle) = self.handles.get_mut(&handle).unwrap() {
let metadata = entry.metadata().await.unwrap(); match handle.next_entry().await {
let dt = Local.timestamp_opt(metadata.mtime(), 0).unwrap(); Ok(Some(entry)) => {
let longname = format!("{} {} {}", metadata.size(), dt.format("%b %e %Y"), entry.file_name().to_string_lossy()); let metadata = entry.metadata().await.unwrap();
Ok(Name { id, files: vec![ let dt = Local.timestamp_opt(metadata.mtime(), 0).unwrap();
File { let longname = format!("{} {} {}", metadata.size(), dt.format("%b %e %Y"), entry.file_name().to_string_lossy());
filename: entry.file_name().to_string_lossy().into(), Ok(Name { id, files: vec![
longname: longname, File {
attrs: FileAttributes { filename: entry.file_name().to_string_lossy().into(),
size: Some(metadata.size()), longname: longname,
atime: Some(metadata.atime() as u32), attrs: FileAttributes {
mtime: Some(metadata.mtime() as u32), size: Some(metadata.size()),
..Default::default() atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}
} }
} ] })
] }) }
} Ok(None) => Err(StatusCode::Eof),
Ok(None) => Err(StatusCode::Eof), Err(e) => {
Err(e) => { println!("Error listing file: {}", e);
println!("Error listing file: {}", e); Err(StatusCode::Failure)
Err(StatusCode::Failure) }
} }
} }
else {
println!("handle is not a dirhandle");
Err(StatusCode::Failure)
}
} }
async fn close( async fn close(
@ -101,6 +187,7 @@ impl SftpHandler for SftpSession {
id: u32, id: u32,
handle: String, handle: String,
) -> Result<Status, Self::Error> { ) -> Result<Status, Self::Error> {
println!("close called");
self.handles.remove(&handle); self.handles.remove(&handle);
Ok(Status { Ok(Status {
id, id,
@ -110,5 +197,60 @@ impl SftpHandler for SftpSession {
}) })
} }
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
}
} }