dota-gsi 0.2.1

Game State Integration with Dota 2 in rust. Provides a server that listens for JSON events sent by Dota 2.
Documentation
use std::path::PathBuf;

use async_trait::async_trait;
use clap::Parser;
use tokio::fs::File;
use tokio::io::AsyncWriteExt;

use dota::{GSIServer, GameStateHandler};

#[derive(Clone, Debug)]
struct RecallHandler {
    output_dir: PathBuf,
}

#[async_trait]
impl GameStateHandler<serde_json::Value> for RecallHandler {
    /// Save raw GameState Integration as JSON for later recalling
    async fn handle(self, gs: serde_json::Value) {
        let file_name = format!("DotaGSI_{}.json", chrono::offset::Local::now());
        let mut file_path = self.output_dir.clone();
        file_path.push(file_name);

        let json_str = serde_json::to_string(&gs).expect("Unable to cast to JSON string.");

        let mut file = File::create(file_path)
            .await
            .expect("Failed to create file for DotaGSI JSON.");
        file.write_all(json_str.as_bytes())
            .await
            .expect("Failed to write DotaGSI JSON file.");
    }
}

/// Listen for Dota 2 events to store them as JSON for recalling later.
#[derive(Parser, Debug)]
#[command(author, version, about, long_about = None)]
struct Args {
    /// URI for the server to listen for events.
    /// This must be the same URI used in the Game State configuration file.
    uri: String,

    /// Optional directory where to store JSON event files.
    #[arg(short, long)]
    output_dir: Option<PathBuf>,
}

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    env_logger::init();

    let args = Args::parse();
    let output_dir = match args.output_dir {
        Some(p) => p,
        None => {
            std::env::current_dir().expect("Not enough permissions to write to current directory.")
        }
    };

    let handler = RecallHandler {
        output_dir: output_dir.clone(),
    };

    let server = GSIServer::new(&args.uri);
    server.run_with_handler(handler).await?;

    Ok(())
}