use anyhow::Result;
use serde_json::{Value, json};
use std::collections::HashMap;
use std::fs::File;
use std::io::Read;
use super::common::connect_wifi;
use crate::DriverCallback;
use crate::DriverContext;
use crate::{
DriverCategory,
types::{Driver, DriverParameter},
};
#[derive(Debug)]
pub struct WifiImportConfigDriver;
#[async_trait::async_trait]
impl Driver for WifiImportConfigDriver {
fn name(&self) -> &str {
"wifi_import_config"
}
fn description(&self) -> &str {
"Import WiFi configuration from a backup file to restore saved networks"
}
fn usage_hint(&self) -> &str {
"Use this skill to restore WiFi settings from a previously exported backup file."
}
fn parameters(&self) -> Vec<DriverParameter> {
vec![
DriverParameter {
name: "file_path".to_string(),
param_type: "string".to_string(),
description: "Path to the configuration file to import".to_string(),
required: true,
default: None,
example: Some(Value::String("./wifi_backup.json".to_string())),
enum_values: None,
},
DriverParameter {
name: "auto_connect".to_string(),
param_type: "boolean".to_string(),
description:
"Automatically connect to the best network after import (default: false)"
.to_string(),
required: false,
default: Some(Value::Bool(false)),
example: Some(Value::Bool(true)),
enum_values: None,
},
]
}
fn example_call(&self) -> Value {
json!({
"action": "wifi_import_config",
"parameters": {
"file_path": "./wifi_backup.json",
"auto_connect": false
}
})
}
fn example_output(&self) -> String {
"Imported 3 WiFi networks from: ./wifi_backup.json".to_string()
}
fn category(&self) -> DriverCategory {
DriverCategory::Wifi
}
async fn execute(
&self,
parameters: &HashMap<String, Value>,
callback: Option<&dyn DriverCallback>,
context: Option<&DriverContext>,
) -> Result<String> {
let file_path = parameters
.get("file_path")
.and_then(|v| v.as_str())
.ok_or_else(|| anyhow::anyhow!("Missing 'file_path' parameter"))?;
let auto_connect = parameters
.get("auto_connect")
.and_then(|v| v.as_bool())
.unwrap_or(false);
let mut file = File::open(file_path)?;
let mut contents = String::new();
file.read_to_string(&mut contents)?;
let import_data: serde_json::Value = serde_json::from_str(&contents)?;
let networks = import_data["networks"].as_array().ok_or_else(|| {
anyhow::anyhow!("Invalid backup file format: missing 'networks' array")
})?;
let mut imported_count = 0;
let mut best_network: Option<String> = None;
let mut best_signal = -100;
for network in networks {
let ssid = network["ssid"]
.as_str()
.ok_or_else(|| anyhow::anyhow!("Network missing 'ssid' field"))?;
let password = network["password"].as_str();
connect_wifi(ssid, password)?;
imported_count += 1;
if auto_connect {
tokio::time::sleep(std::time::Duration::from_millis(500)).await;
let status = super::common::get_wifi_status()?;
if let Some(signal) = status.signal_strength {
if signal > best_signal {
best_signal = signal;
best_network = Some(ssid.to_string());
}
}
}
}
let auto_connect_msg = if auto_connect {
if let Some(ref network) = best_network {
connect_wifi(network, None)?;
tokio::time::sleep(std::time::Duration::from_secs(2)).await;
format!(" (auto-connected to: {})", network)
} else {
String::new()
}
} else {
String::new()
};
Ok(format!(
"Imported {} WiFi networks from: {}{}",
imported_count, file_path, auto_connect_msg
))
}
}