Skip to content

Commit

Permalink
Add file server endpoint
Browse files Browse the repository at this point in the history
Signed-off-by: Patrick José Pereira <[email protected]>
  • Loading branch information
patrickelectric committed Aug 22, 2024
1 parent 3706238 commit 6ad2daf
Show file tree
Hide file tree
Showing 7 changed files with 126 additions and 4 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ shellexpand = "3.1"
chrono = "0.4"
url = { version = "2.5.2", features = ["serde"] }
ctrlc = "3.4"
byteorder = "1.5.0"

tracing = { version = "0.1.40", features = ["log", "async-await"] }
tracing-subscriber = { version = "0.3.18", features = ["env-filter"] }
Expand Down
14 changes: 11 additions & 3 deletions src/cli.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,14 +55,17 @@ fn endpoints_parser(endpoint: &str) -> Result<String, String> {
let endpoint = endpoint.to_lowercase();

let mut split = endpoint.split(':');
if split.clone().count() != 3 {
if split.clone().count() != 2 && split.clone().count() != 3 {
return Err("Wrong endpoint format".to_string());
}

let kind = split.next().expect(
"Endpoint should start with one of the kinds: udps, udpc, udpb, tcps, tcpc, or serial",
"Endpoint should start with one of the kinds: file, udps, udpc, udpb, tcps, tcpc, or serial",
);
if !matches!(kind, "udps" | "udpc" | "udpb" | "tcps" | "tcpc" | "serial") {
if !matches!(
kind,
"file" | "udps" | "udpc" | "udpb" | "tcps" | "tcpc" | "serial"
) {
return Err(format!("Unknown kind: {kind:?} for endpoint"));
}

Expand Down Expand Up @@ -126,6 +129,11 @@ pub fn tcp_client_endpoints() -> Vec<String> {
get_endpoint_with_kind("tcpc")
}

#[instrument(level = "debug")]
pub fn file_server_endpoints() -> Vec<String> {
get_endpoint_with_kind("file")
}

#[instrument(level = "debug")]
pub fn tcp_server_endpoints() -> Vec<String> {
get_endpoint_with_kind("tcps")
Expand Down
1 change: 1 addition & 0 deletions src/drivers/file/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
pub mod server;
103 changes: 103 additions & 0 deletions src/drivers/file/server.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
use std::sync::Arc;

use anyhow::Result;
use chrono::DateTime;
use mavlink::ardupilotmega::MavMessage;
use std::path::PathBuf;
use tokio::io::AsyncReadExt;
use tokio::sync::broadcast;
use tracing::*;

use crate::drivers::{Driver, DriverInfo};
use crate::protocol::Protocol;

#[derive(Clone, Debug)]
pub struct FileServer {
pub path: PathBuf,
}

impl FileServer {
#[instrument(level = "debug")]
pub fn try_new(file_path: &str) -> Result<Self> {
let path = PathBuf::from(file_path);
Ok(Self { path })
}

#[instrument(level = "debug", skip(reader, hub_sender))]
async fn handle_client(
server: FileServer,
mut reader: tokio::io::BufReader<tokio::fs::File>,
hub_sender: Arc<broadcast::Sender<Protocol>>,
) -> Result<()> {
let source_name = server.path.as_path().display().to_string();
loop {
// Tlog files follow the byte format of <unix_timestamp_us><raw_mavlink_messsage>
let Ok(us_since_epoch) = reader.read_u64().await else {
info!("End of file reached");
break;
};

let Some(_date_time) = DateTime::from_timestamp_micros(us_since_epoch as i64) else {
warn!("Failed to convert unix time");
continue;
};

// Ensure that we have at least a single byte before checking for a valid mavlink message
if (reader.buffer().is_empty()) {
info!("End of file reached");
break;
}

// Since the source is a tlog file that includes timestamps + raw mavlink messages.
// We first need to be sure that the next byte is the start of a mavlink message,
// otherwise the `read_v2_raw_message_async` will process valid timestamps as garbage.
if (reader.buffer()[0] != mavlink::MAV_STX_V2) {
warn!("Invalid MAVLink start byte, skipping");
continue;
}

let message =
match mavlink::read_v2_raw_message_async::<MavMessage, _>(&mut reader).await {
Ok(message) => message,
Err(error) => {
error!("Failed to parse MAVLink message: {error:?}");
continue; // Skip this iteration on error
}
};

let message = Protocol::new(&source_name, message);

trace!("Received File message: {message:?}");
if let Err(error) = hub_sender.send(message) {
error!("Failed to send message to hub: {error:?}");
}
}

debug!("File Receive task for {source_name} finished");
Ok(())
}
}

#[async_trait::async_trait]
impl Driver for FileServer {
#[instrument(level = "debug", skip(self, hub_sender))]
async fn run(&self, hub_sender: broadcast::Sender<Protocol>) -> Result<()> {
let file = tokio::fs::File::open(self.path.clone()).await.unwrap();
let mut reader = tokio::io::BufReader::new(file);

tokio::spawn(FileServer::handle_client(
self.clone(),
reader,
Arc::new(hub_sender),
));

Ok(())
}

#[instrument(level = "debug", skip(self))]
fn info(&self) -> DriverInfo {
DriverInfo {
name: "FileServer".to_string(),
}
}
}
1 change: 1 addition & 0 deletions src/drivers/mod.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
pub mod fake;
pub mod file;
pub mod tcp;
pub mod udp;

Expand Down
9 changes: 8 additions & 1 deletion src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ async fn main() -> Result<()> {
logger::init();

let hub = hub::Hub::new(
100,
10000,
Arc::new(RwLock::new(
mavlink::ardupilotmega::MavComponent::MAV_COMP_ID_ONBOARD_COMPUTER as u8,
)),
Expand All @@ -29,6 +29,13 @@ async fn main() -> Result<()> {

// Endpoints creation
{
for endpoint in cli::file_server_endpoints() {
debug!("Creating File Server to {endpoint:?}");
hub.add_driver(Arc::new(
drivers::file::server::FileServer::try_new(&endpoint).unwrap(),
))
.await?;
}
for endpoint in cli::tcp_client_endpoints() {
debug!("Creating TCP Client to {endpoint:?}");
hub.add_driver(Arc::new(drivers::tcp::client::TcpClient::new(&endpoint)))
Expand Down

0 comments on commit 6ad2daf

Please sign in to comment.