2025-07-05 20:09:01 +05:00
|
|
|
mod sftp;
|
|
|
|
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 sftp::SftpSession;
|
2025-07-07 07:46:50 +05:00
|
|
|
use sqlx::{sqlite::SqlitePoolOptions, Pool, Row, Sqlite};
|
2025-07-02 06:39:31 +05:00
|
|
|
|
|
|
|
|
2025-07-07 07:46:50 +05:00
|
|
|
struct SftpServer {
|
|
|
|
pool: Arc<Pool<Sqlite>>
|
|
|
|
}
|
2025-07-02 06:39:31 +05:00
|
|
|
|
|
|
|
impl Server for SftpServer {
|
|
|
|
type Handler = SshSession;
|
|
|
|
|
|
|
|
fn new_client(&mut self, _peer_addr: Option<SocketAddr>) -> Self::Handler {
|
2025-07-07 07:46:50 +05:00
|
|
|
let session_pool = self.pool.clone();
|
|
|
|
SshSession { channel: None, user: None, pool: session_pool }
|
2025-07-02 06:39:31 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SshSession {
|
2025-07-04 12:02:10 +05:00
|
|
|
channel: Option<Channel<Msg>>,
|
2025-07-07 07:46:50 +05:00
|
|
|
user: Option<String>,
|
|
|
|
pool: Arc<Pool<Sqlite>>
|
2025-07-02 06:39:31 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SshHandler for SshSession {
|
2025-07-05 20:09:01 +05:00
|
|
|
type Error = russh::Error;
|
2025-07-02 06:39:31 +05:00
|
|
|
|
|
|
|
async fn auth_publickey_offered(
|
2025-07-04 12:02:10 +05:00
|
|
|
&mut self,
|
|
|
|
user: &str,
|
|
|
|
public_key: &PublicKey,
|
|
|
|
) -> Result<Auth, Self::Error> {
|
|
|
|
self.user = Some(user.to_string());
|
2025-07-07 07:46:50 +05:00
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Ok(Auth::reject())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
println!("User Not found: {}", e);
|
|
|
|
Ok(Auth::reject())
|
|
|
|
}
|
|
|
|
}
|
2025-07-02 06:39:31 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn auth_publickey(
|
2025-07-04 12:02:10 +05:00
|
|
|
&mut self,
|
2025-07-07 07:46:50 +05:00
|
|
|
_user: &str,
|
|
|
|
_public_key: &PublicKey,
|
2025-07-04 12:02:10 +05:00
|
|
|
) -> Result<Auth, Self::Error> {
|
2025-07-02 06:39:31 +05:00
|
|
|
Ok(Auth::Accept)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn channel_open_session(
|
2025-07-04 12:02:10 +05:00
|
|
|
&mut self,
|
|
|
|
channel: russh::Channel<Msg>,
|
|
|
|
_session: &mut Session,
|
|
|
|
) -> Result<bool, Self::Error> {
|
2025-07-02 06:39:31 +05:00
|
|
|
self.channel = Some(channel);
|
|
|
|
Ok(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn channel_eof(
|
2025-07-04 12:02:10 +05:00
|
|
|
&mut self,
|
|
|
|
channel_id: ChannelId,
|
|
|
|
session: &mut Session,
|
|
|
|
) -> Result<(), Self::Error> {
|
2025-07-02 06:39:31 +05:00
|
|
|
session.close(channel_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn subsystem_request(
|
2025-07-04 12:02:10 +05:00
|
|
|
&mut self,
|
|
|
|
channel_id: ChannelId,
|
|
|
|
name: &str,
|
|
|
|
session: &mut Session,
|
|
|
|
) -> Result<(), Self::Error> {
|
2025-07-02 06:39:31 +05:00
|
|
|
if name == "sftp" {
|
|
|
|
session.channel_success(channel_id)?;
|
2025-07-06 06:33:59 +05:00
|
|
|
let jail_dir = format!("/srv/sftp/{}", self.user.as_ref().unwrap());
|
2025-07-05 20:09:01 +05:00
|
|
|
let sftp_handler = SftpSession::new(jail_dir);
|
2025-07-02 06:39:31 +05:00
|
|
|
russh_sftp::server::run(self.channel.take().ok_or(Self::Error::WrongChannel)?.into_stream(), sftp_handler).await;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
session.channel_failure(channel_id)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[tokio::main]
|
2025-07-07 07:46:50 +05:00
|
|
|
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) };
|
2025-07-02 06:39:31 +05:00
|
|
|
|
|
|
|
let config = russh::server::Config {
|
|
|
|
auth_rejection_time: Duration::from_secs(3),
|
|
|
|
auth_rejection_time_initial: Some(Duration::from_secs(0)),
|
|
|
|
keys: vec![
|
|
|
|
russh::keys::PrivateKey::random(&mut OsRng, russh::keys::Algorithm::Ed25519).unwrap(),
|
|
|
|
],
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
|
|
|
|
server.run_on_address(Arc::new(config), ("0.0.0.0", 2222)).await.unwrap();
|
2025-07-07 07:46:50 +05:00
|
|
|
Ok(())
|
2025-07-02 06:39:31 +05:00
|
|
|
}
|