Compare commits

..

10 commits

Author SHA1 Message Date
RafayAhmad7548
c717f9f4f6 simple auth using sqlite 2025-07-07 07:46:50 +05:00
RafayAhmad7548
2ce96ae74c added a macro 2025-07-06 16:51:27 +05:00
RafayAhmad7548
649263ff5f update fstat and fix exclude opening files 2025-07-06 14:45:01 +05:00
RafayAhmad7548
f1bc7e9a1b rename pogg 2025-07-06 08:14:21 +05:00
RafayAhmad7548
2328982583 rm mkdir and rmdir pooog 2025-07-06 07:56:51 +05:00
RafayAhmad7548
2840b5bed7 yoo, nautlius also work now 2025-07-06 07:17:51 +05:00
RafayAhmad7548
464eb19ee4 write added so much pogggers 2025-07-06 07:05:12 +05:00
RafayAhmad7548
34db55a70c stats opening and reading file, very poggers 2025-07-06 06:33:59 +05:00
RafayAhmad7548
611f29f325 new file and progress 2025-07-05 20:09:01 +05:00
RafayAhmad7548
899d5a1e17 added some fns 2025-07-04 12:02:10 +05:00
6 changed files with 1379 additions and 59 deletions

2
.gitignore vendored
View file

@ -1 +1,3 @@
/target /target
sftptest
sftptest.pub

955
Cargo.lock generated

File diff suppressed because it is too large Load diff

View file

@ -4,6 +4,9 @@ version = "0.1.0"
edition = "2024" edition = "2024"
[dependencies] [dependencies]
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"] }
sqlx = { version = "0.8.6", features = [ "runtime-tokio", "postgres", "sqlite", "mysql" ] }

BIN
auth.db Normal file

Binary file not shown.

View file

@ -1,38 +1,65 @@
mod sftp;
use std::{net::SocketAddr, sync::Arc, time::Duration}; use std::{net::SocketAddr, sync::Arc, time::Duration};
use russh::{keys::ssh_key::{rand_core::OsRng, PublicKey}, server::{Auth, Handler as SshHandler, Msg, Server, Session}, Channel, ChannelId};
use russh::{keys::ssh_key::{rand_core::OsRng, PublicKey}, server::{Auth, Handler as SshHandler, Msg, Server, Session}, Channel, ChannelId, Error}; use sftp::SftpSession;
use russh_sftp::{protocol::{Name, StatusCode}, server::Handler as SftpHandler}; use sqlx::{sqlite::SqlitePoolOptions, Pool, Row, Sqlite};
struct SftpServer; struct SftpServer {
pool: Arc<Pool<Sqlite>>
}
impl Server for SftpServer { impl Server for SftpServer {
type Handler = SshSession; type Handler = SshSession;
fn new_client(&mut self, _peer_addr: Option<SocketAddr>) -> Self::Handler { fn new_client(&mut self, _peer_addr: Option<SocketAddr>) -> Self::Handler {
SshSession{ channel: None } let session_pool = self.pool.clone();
SshSession { channel: None, user: None, pool: session_pool }
} }
} }
struct SshSession { struct SshSession {
channel: Option<Channel<Msg>> channel: Option<Channel<Msg>>,
user: Option<String>,
pool: Arc<Pool<Sqlite>>
} }
impl SshHandler for SshSession { impl SshHandler for SshSession {
type Error = Error; type Error = russh::Error;
async fn auth_publickey_offered( async fn auth_publickey_offered(
&mut self, &mut self,
user: &str, user: &str,
public_key: &PublicKey, public_key: &PublicKey,
) -> Result<Auth, Self::Error> { ) -> Result<Auth, Self::Error> {
self.user = Some(user.to_string());
let row_res = sqlx::query("SELECT * FROM users WHERE username = ?")
.bind(user)
.fetch_one(&*self.pool).await;
match row_res {
Ok(row) => {
let stored_key: String = row.get("public_key");
let offered_key = public_key.to_string();
if stored_key == offered_key {
Ok(Auth::Accept) Ok(Auth::Accept)
} }
else {
Ok(Auth::reject())
}
}
Err(e) => {
println!("User Not found: {}", e);
Ok(Auth::reject())
}
}
}
async fn auth_publickey( async fn auth_publickey(
&mut self, &mut self,
user: &str, _user: &str,
public_key: &PublicKey, _public_key: &PublicKey,
) -> Result<Auth, Self::Error> { ) -> Result<Auth, Self::Error> {
Ok(Auth::Accept) Ok(Auth::Accept)
} }
@ -62,7 +89,8 @@ 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 sftp_handler = SftpSession {}; let jail_dir = format!("/srv/sftp/{}", self.user.as_ref().unwrap());
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;
} }
else { else {
@ -72,21 +100,14 @@ impl SshHandler for SshSession {
} }
} }
struct SftpSession;
impl SftpHandler for SftpSession {
type Error = StatusCode;
fn unimplemented(&self) -> Self::Error {
Self::Error::OpUnsupported
}
}
#[tokio::main] #[tokio::main]
async fn main() { async fn main() -> Result<(), sqlx::Error> {
let pool = SqlitePoolOptions::new()
.max_connections(3)
.connect("sqlite:/home/rafayahmad/Stuff/Coding/Rust/flux-sftp/auth.db").await?;
let mut server = SftpServer { pool: Arc::new(pool) };
let config = russh::server::Config { let config = russh::server::Config {
auth_rejection_time: Duration::from_secs(3), auth_rejection_time: Duration::from_secs(3),
@ -96,7 +117,7 @@ async fn main() {
], ],
..Default::default() ..Default::default()
}; };
let mut server = SftpServer;
server.run_on_address(Arc::new(config), ("0.0.0.0", 2222)).await.unwrap(); server.run_on_address(Arc::new(config), ("0.0.0.0", 2222)).await.unwrap();
Ok(())
} }

369
src/sftp.rs Normal file
View file

@ -0,0 +1,369 @@
use std::{collections::HashMap, io::{ErrorKind, SeekFrom}, os::unix::fs::MetadataExt};
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};
use tokio::{fs::{self, OpenOptions, ReadDir}, io::{AsyncReadExt, AsyncSeekExt, AsyncWriteExt}};
macro_rules! match_expr {
($match:expr, $err_msg:literal, $id:ident) => {
match $match {
Ok(()) => Ok(Status { $id, status_code: StatusCode::Ok, error_message: "Ok".to_string(), language_tag: "en-US".to_string() }),
Err(e) => {
println!($err_msg, e);
match e.kind() {
ErrorKind::NotFound => Ok(Status { $id, status_code: StatusCode::NoSuchFile, error_message: e.to_string(), language_tag: "en-US".to_string() }),
ErrorKind::PermissionDenied => Ok(Status { $id, status_code: StatusCode::PermissionDenied, error_message: e.to_string(), language_tag: "en-US".to_string() }),
ErrorKind::ConnectionReset => Ok(Status { $id, status_code: StatusCode::ConnectionLost, error_message: e.to_string(), language_tag: "en-US".to_string() }),
ErrorKind::NotConnected => Ok(Status { $id, status_code: StatusCode::NoConnection, error_message: e.to_string(), language_tag: "en-US".to_string() }),
_ => Ok(Status { $id, status_code: StatusCode::Failure, error_message: e.to_string(), language_tag: "en-US".to_string() })
}
}
}
};
}
enum Handle {
Dir(ReadDir),
File(fs::File)
}
pub struct SftpSession {
jail_dir: String,
cwd: String,
handles: HashMap<String, Handle>
}
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);
println!("pflags raw: {:b}", pflags.bits());
println!("pflags: read: {}, write: {}, append: {}, create: {}, truncate: {}", pflags.contains(OpenFlags::READ), pflags.contains(OpenFlags::WRITE), pflags.contains(OpenFlags::APPEND), pflags.contains(OpenFlags::CREATE), pflags.contains(OpenFlags::TRUNCATE));
let path = format!("{}{}", self.jail_dir, filename);
if pflags.contains(OpenFlags::EXCLUDE) && fs::metadata(&path).await.is_ok() {
return Err(StatusCode::Failure)
}
let mut options = OpenOptions::new();
options
.read(pflags.contains(OpenFlags::READ))
.write(pflags.contains(OpenFlags::WRITE))
.append(pflags.contains(OpenFlags::APPEND))
.create(pflags.contains(OpenFlags::CREATE))
.truncate(pflags.contains(OpenFlags::TRUNCATE));
match options.open(&path).await {
Ok(file) => {
self.handles.insert(filename.clone(), Handle::File(file));
Ok(SftpHandle { id, handle: filename })
}
Err(e) => {
println!("error opeing file: {}", e);
match e.kind() {
ErrorKind::NotFound => Err(StatusCode::NoSuchFile),
ErrorKind::PermissionDenied => Err(StatusCode::PermissionDenied),
ErrorKind::ConnectionReset => Err(StatusCode::ConnectionLost),
ErrorKind::NotConnected => Err(StatusCode::NoConnection),
_ => Err(StatusCode::Failure)
}
}
}
}
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 {
println!("handle is not a filehandle");
Err(StatusCode::Failure)
}
}
async fn write(
&mut self,
id: u32,
handle: String,
offset: u64,
data: Vec<u8>,
) -> Result<Status, Self::Error> {
println!("write called, offset: {}, data: {:?}", offset, String::from_utf8(data.clone()));
if let Handle::File(file) = self.handles.get_mut(&handle).unwrap() {
match file.seek(SeekFrom::Start(offset)).await {
Ok(_) => {
match file.write_all(&data).await {
Ok(()) => {
Ok(Status {
id,
status_code: StatusCode::Ok,
error_message: "Ok".to_string(),
language_tag: "en-US".to_string(),
})
}
Err(e) => {
println!("Error in writing at offset in file: {}", e);
Ok(Status { id, status_code: StatusCode::Failure, error_message: e.to_string(), language_tag: "en-US".to_string() })
}
}
}
Err(e) => {
println!("Error in seeking offset in file: {}", e);
Ok(Status { id, status_code: StatusCode::Failure, error_message: e.to_string(), language_tag: "en-US".to_string() })
}
}
}
else {
println!("handle is not a filehandle");
Err(StatusCode::Failure)
}
}
async fn opendir(
&mut self,
id: u32,
path: String,
) -> Result<SftpHandle, Self::Error> {
println!("opendir called: {}", path);
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 })
}
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()),
permissions: Some(metadata.mode()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}
}
] })
}
Ok(None) => Err(StatusCode::Eof),
Err(e) => {
println!("Error listing file: {}", e);
Err(StatusCode::Failure)
}
}
}
else {
println!("handle is not a dirhandle");
Err(StatusCode::Failure)
}
}
async fn close(
&mut self,
id: u32,
handle: String,
) -> Result<Status, Self::Error> {
println!("close called");
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: {}", path);
let path = format!("{}{}", self.jail_dir, path);
match fs::metadata(path).await {
Ok(metadata) => Ok(Attrs { id, attrs: FileAttributes {
size: Some(metadata.size()),
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: {}", path);
let path = format!("{}{}", self.jail_dir, path);
match fs::symlink_metadata(path).await {
Ok(metadata) => Ok(Attrs { id, attrs: FileAttributes {
size: Some(metadata.size()),
permissions: Some(metadata.mode()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}}),
Err(_) => Err(StatusCode::OpUnsupported)
}
}
async fn fstat(
&mut self,
id: u32,
handle: String,
) -> Result<Attrs, Self::Error> {
println!("fstat called: {}", handle);
if let Handle::File(file) = self.handles.get(&handle).unwrap() {
let metadata = file.metadata().await.unwrap();
Ok(Attrs { id, attrs: FileAttributes {
size: Some(metadata.size()),
permissions: Some(metadata.mode()),
atime: Some(metadata.atime() as u32),
mtime: Some(metadata.mtime() as u32),
..Default::default()
}})
}
else {
println!("handle is not a filehandle");
Err(StatusCode::Failure)
}
}
async fn remove(
&mut self,
id: u32,
filename: String,
) -> Result<Status, Self::Error> {
println!("remove called: {}", filename);
let path = format!("{}{}", self.jail_dir, filename);
match_expr!(fs::remove_file(path).await, "error removing file: {}", id)
}
async fn mkdir(
&mut self,
id: u32,
path: String,
_attrs: FileAttributes,
) -> Result<Status, Self::Error> {
println!("mkdir called: {}", path);
let path = format!("{}{}", self.jail_dir, path);
match_expr!(fs::create_dir(path).await, "error creating dir: {}", id)
}
async fn rmdir(
&mut self,
id: u32,
path: String,
) -> Result<Status, Self::Error> {
println!("rmdir called: {}", path);
let path = format!("{}{}", self.jail_dir, path);
match_expr!(fs::remove_dir(path).await, "error removing file: {}", id)
}
async fn rename(
&mut self,
id: u32,
oldpath: String,
newpath: String,
) -> Result<Status, Self::Error> {
println!("rename called from: {}, to: {}", oldpath, newpath);
let oldpath = format!("{}{}", self.jail_dir, oldpath);
let newpath = format!("{}{}", self.jail_dir, newpath);
match_expr!(fs::rename(oldpath, newpath).await, "error renaming file: {}", id)
}
}