2020-04-08 14:26:42 +00:00
|
|
|
use crate::commands::{Command, FileInfo, LocalCommand, RemoteInfo};
|
|
|
|
use async_std::{
|
|
|
|
fs,
|
|
|
|
path::PathBuf,
|
|
|
|
sync::{Mutex, RwLock},
|
|
|
|
};
|
|
|
|
use futures::{channel::mpsc, future::FutureExt, stream::StreamExt};
|
2020-07-09 11:42:38 +00:00
|
|
|
use network::{ProtocolAddr, Network, Participant, Pid, Stream, PROMISES_CONSISTENCY, PROMISES_ORDERED};
|
2020-04-08 14:26:42 +00:00
|
|
|
use std::{collections::HashMap, sync::Arc};
|
|
|
|
use tracing::*;
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
struct ControlChannels {
|
|
|
|
command_receiver: mpsc::UnboundedReceiver<LocalCommand>,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Server {
|
|
|
|
run_channels: Option<ControlChannels>,
|
|
|
|
network: Network,
|
|
|
|
served: RwLock<Vec<FileInfo>>,
|
|
|
|
remotes: RwLock<HashMap<Pid, Arc<Mutex<RemoteInfo>>>>,
|
|
|
|
receiving_files: Mutex<HashMap<u32, Option<String>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Server {
|
|
|
|
pub fn new() -> (Self, mpsc::UnboundedSender<LocalCommand>) {
|
|
|
|
let (command_sender, command_receiver) = mpsc::unbounded();
|
|
|
|
|
2020-07-14 23:34:41 +00:00
|
|
|
let (network, f) = Network::new(Pid::new());
|
2020-06-08 09:47:39 +00:00
|
|
|
std::thread::spawn(f);
|
2020-04-08 14:26:42 +00:00
|
|
|
|
|
|
|
let run_channels = Some(ControlChannels { command_receiver });
|
|
|
|
(
|
|
|
|
Server {
|
|
|
|
run_channels,
|
|
|
|
network,
|
|
|
|
served: RwLock::new(vec![]),
|
|
|
|
remotes: RwLock::new(HashMap::new()),
|
|
|
|
receiving_files: Mutex::new(HashMap::new()),
|
|
|
|
},
|
|
|
|
command_sender,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-07-09 11:42:38 +00:00
|
|
|
pub async fn run(mut self, address: ProtocolAddr) {
|
2020-04-08 14:26:42 +00:00
|
|
|
let run_channels = self.run_channels.take().unwrap();
|
|
|
|
|
|
|
|
self.network.listen(address).await.unwrap();
|
|
|
|
|
|
|
|
futures::join!(
|
|
|
|
self.command_manager(run_channels.command_receiver,),
|
|
|
|
self.connect_manager(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn command_manager(&self, command_receiver: mpsc::UnboundedReceiver<LocalCommand>) {
|
2020-07-05 22:13:53 +00:00
|
|
|
trace!("Start command_manager");
|
2020-04-08 14:26:42 +00:00
|
|
|
command_receiver
|
|
|
|
.for_each_concurrent(None, async move |cmd| {
|
|
|
|
match cmd {
|
|
|
|
LocalCommand::Shutdown => {
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Shutting down service");
|
2020-04-08 14:26:42 +00:00
|
|
|
return;
|
|
|
|
},
|
|
|
|
LocalCommand::Disconnect => {
|
|
|
|
self.remotes.write().await.clear();
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Disconnecting all connections");
|
2020-04-08 14:26:42 +00:00
|
|
|
return;
|
|
|
|
},
|
|
|
|
LocalCommand::Connect(addr) => {
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Trying to connect to: {:?}", &addr);
|
2020-04-08 14:26:42 +00:00
|
|
|
match self.network.connect(addr.clone()).await {
|
|
|
|
Ok(p) => self.loop_participant(p).await,
|
|
|
|
Err(e) => {
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Failled to connect to {:?}, err: {:?}", &addr, e);
|
2020-04-08 14:26:42 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
LocalCommand::Serve(fileinfo) => {
|
|
|
|
self.served.write().await.push(fileinfo.clone());
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Serving file: {:?}", fileinfo.path);
|
2020-04-08 14:26:42 +00:00
|
|
|
},
|
|
|
|
LocalCommand::List => {
|
|
|
|
let mut total_file_infos = vec![];
|
|
|
|
for ri in self.remotes.read().await.values() {
|
|
|
|
let mut ri = ri.lock().await;
|
|
|
|
ri.cmd_out.send(Command::List).unwrap();
|
|
|
|
let mut file_infos = ri.cmd_out.recv::<Vec<FileInfo>>().await.unwrap();
|
|
|
|
ri.insert_infos(file_infos.clone());
|
|
|
|
total_file_infos.append(&mut file_infos);
|
|
|
|
}
|
|
|
|
print_fileinfos(&total_file_infos);
|
|
|
|
},
|
|
|
|
LocalCommand::Get(id, path) => {
|
|
|
|
// i dont know the owner, just broadcast, i am laaaazyyy
|
|
|
|
for ri in self.remotes.read().await.values() {
|
|
|
|
let mut ri = ri.lock().await;
|
|
|
|
if ri.get_info(id).is_some() {
|
|
|
|
//found provider, send request.
|
|
|
|
self.receiving_files.lock().await.insert(id, path.clone());
|
|
|
|
ri.cmd_out.send(Command::Get(id)).unwrap();
|
|
|
|
// the answer is handled via the other stream!
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.await;
|
2020-07-05 22:13:53 +00:00
|
|
|
trace!("Stop command_manager");
|
2020-04-08 14:26:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn connect_manager(&self) {
|
2020-07-05 22:13:53 +00:00
|
|
|
trace!("Start connect_manager");
|
2020-04-08 14:26:42 +00:00
|
|
|
let iter = futures::stream::unfold((), |_| {
|
|
|
|
self.network.connected().map(|r| r.ok().map(|v| (v, ())))
|
|
|
|
});
|
|
|
|
|
|
|
|
iter.for_each_concurrent(/* limit */ None, async move |participant| {
|
|
|
|
self.loop_participant(participant).await;
|
|
|
|
})
|
|
|
|
.await;
|
2020-07-05 22:13:53 +00:00
|
|
|
trace!("Stop connect_manager");
|
2020-04-08 14:26:42 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 07:58:21 +00:00
|
|
|
async fn loop_participant(&self, p: Participant) {
|
2020-04-08 14:26:42 +00:00
|
|
|
if let (Ok(cmd_out), Ok(file_out), Ok(cmd_in), Ok(file_in)) = (
|
|
|
|
p.open(15, PROMISES_CONSISTENCY | PROMISES_ORDERED).await,
|
|
|
|
p.open(40, PROMISES_CONSISTENCY).await,
|
|
|
|
p.opened().await,
|
|
|
|
p.opened().await,
|
|
|
|
) {
|
2020-07-05 22:13:53 +00:00
|
|
|
debug!(?p, "Connection successfully initiated");
|
2020-04-08 14:26:42 +00:00
|
|
|
let id = p.remote_pid();
|
|
|
|
let ri = Arc::new(Mutex::new(RemoteInfo::new(cmd_out, file_out, p)));
|
|
|
|
self.remotes.write().await.insert(id, ri.clone());
|
|
|
|
futures::join!(
|
|
|
|
self.handle_remote_cmd(cmd_in, ri.clone()),
|
|
|
|
self.handle_files(file_in, ri.clone()),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn handle_remote_cmd(&self, mut stream: Stream, remote_info: Arc<Mutex<RemoteInfo>>) {
|
|
|
|
while let Ok(msg) = stream.recv::<Command>().await {
|
2020-07-05 22:13:53 +00:00
|
|
|
println!("Got message: {:?}", &msg);
|
2020-04-08 14:26:42 +00:00
|
|
|
match msg {
|
|
|
|
Command::List => {
|
2020-07-05 22:13:53 +00:00
|
|
|
info!("Request to send my list");
|
2020-04-08 14:26:42 +00:00
|
|
|
let served = self.served.read().await.clone();
|
|
|
|
stream.send(served).unwrap();
|
|
|
|
},
|
|
|
|
Command::Get(id) => {
|
|
|
|
for file_info in self.served.read().await.iter() {
|
|
|
|
if file_info.id() == id {
|
2020-07-05 22:13:53 +00:00
|
|
|
info!("Request to send file i got, sending it");
|
2020-04-08 14:26:42 +00:00
|
|
|
if let Ok(data) = file_info.load().await {
|
|
|
|
match remote_info.lock().await.file_out.send((file_info, data)) {
|
|
|
|
Ok(_) => debug!("send file"),
|
|
|
|
Err(e) => error!(?e, "sending file failed"),
|
|
|
|
}
|
|
|
|
} else {
|
2020-07-05 22:13:53 +00:00
|
|
|
warn!("Cannot send file as loading failed, oes it still exist?");
|
2020-04-08 14:26:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn handle_files(&self, mut stream: Stream, _remote_info: Arc<Mutex<RemoteInfo>>) {
|
|
|
|
while let Ok((fi, data)) = stream.recv::<(FileInfo, Vec<u8>)>().await {
|
2020-07-05 22:13:53 +00:00
|
|
|
debug!(?fi, "Got file");
|
2020-04-08 14:26:42 +00:00
|
|
|
let path = self.receiving_files.lock().await.remove(&fi.id()).flatten();
|
|
|
|
let path: PathBuf = match &path {
|
|
|
|
Some(path) => shellexpand::tilde(&path).parse().unwrap(),
|
|
|
|
None => {
|
|
|
|
let mut path = std::env::current_dir().unwrap();
|
|
|
|
path.push(fi.path().file_name().unwrap());
|
2020-07-05 22:13:53 +00:00
|
|
|
trace!("No path provided, saving down to {:?}", path);
|
2020-04-08 14:26:42 +00:00
|
|
|
PathBuf::from(path)
|
|
|
|
},
|
|
|
|
};
|
2020-07-05 22:13:53 +00:00
|
|
|
debug!("Received file, going to save it under {:?}", path);
|
2020-04-08 14:26:42 +00:00
|
|
|
fs::write(path, data).await.unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_fileinfos(infos: &Vec<FileInfo>) {
|
|
|
|
let mut i = 0;
|
|
|
|
for info in infos {
|
|
|
|
let bytes = info.size;
|
|
|
|
match bytes {
|
|
|
|
0..100_000 => println!("{}: {}bytes '{}'", info.id(), bytes, info.path),
|
|
|
|
100_000..100_000_000 => {
|
|
|
|
println!("{}: {}bytes '{}'", info.id(), bytes / 1024, info.path)
|
|
|
|
},
|
|
|
|
_ => println!(
|
|
|
|
"{}: {}bytes '{}'",
|
|
|
|
info.id(),
|
|
|
|
bytes / 1024 / 1024,
|
|
|
|
info.path
|
|
|
|
),
|
|
|
|
}
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
println!("-- {} files available", i);
|
|
|
|
}
|