hippox-drivers 0.3.1

🦛All indivisible atomic driver units in Hippox.
//! WiFi get saved passwords skill - retrieve saved WiFi passwords

use super::common::list_saved_networks;
use crate::DriverCallback;
use crate::DriverContext;
use crate::{
    DriverCategory,
    types::{Driver, DriverParameter},
};
use anyhow::Result;
use serde_json::{Value, json};
use std::collections::HashMap;
use std::process::Command;

#[derive(Debug)]
pub struct WifiGetSavedPasswordsDriver;

#[async_trait::async_trait]
impl Driver for WifiGetSavedPasswordsDriver {
    fn name(&self) -> &str {
        "wifi_get_saved_passwords"
    }

    fn description(&self) -> &str {
        "Get saved WiFi passwords for networks the device has connected to"
    }

    fn usage_hint(&self) -> &str {
        "Use this skill to retrieve saved WiFi passwords. May require administrator privileges."
    }

    fn parameters(&self) -> Vec<DriverParameter> {
        vec![DriverParameter {
            name: "ssid".to_string(),
            param_type: "string".to_string(),
            description: "Specific SSID to get password for (omit for all networks)".to_string(),
            required: false,
            default: None,
            example: Some(Value::String("MyWiFi".to_string())),
            enum_values: None,
        }]
    }

    fn example_call(&self) -> Value {
        json!({
            "action": "wifi_get_saved_passwords",
            "parameters": {
                "ssid": "MyWiFi"
            }
        })
    }

    fn example_output(&self) -> String {
        "Password for MyWiFi: password123".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 specific_ssid = parameters.get("ssid").and_then(|v| v.as_str());

        let mut result = String::new();

        #[cfg(target_os = "windows")]
        {
            if let Some(ssid) = specific_ssid {
                let output = Command::new("netsh")
                    .args(["wlan", "show", "profile", "name=", ssid, "key=clear"])
                    .output()?;
                let stdout = String::from_utf8_lossy(&output.stdout);

                for line in stdout.lines() {
                    if line.contains("Key Content") || line.contains("关键内容") {
                        if let Some(pwd) = line.split(':').nth(1) {
                            result = format!("Password for {}: {}", ssid, pwd.trim());
                            break;
                        }
                    }
                }
                if result.is_empty() {
                    result = format!("No password found for {}", ssid);
                }
            } else {
                let networks = list_saved_networks()?;
                for network in networks {
                    let output = Command::new("netsh")
                        .args([
                            "wlan",
                            "show",
                            "profile",
                            "name=",
                            &network.ssid,
                            "key=clear",
                        ])
                        .output()?;
                    let stdout = String::from_utf8_lossy(&output.stdout);

                    for line in stdout.lines() {
                        if line.contains("Key Content") || line.contains("关键内容") {
                            if let Some(pwd) = line.split(':').nth(1) {
                                result.push_str(&format!("{}: {}\n", network.ssid, pwd.trim()));
                                break;
                            }
                        }
                    }
                }
            }
        }

        #[cfg(any(target_os = "linux", target_os = "macos"))]
        {
            result = "Password retrieval on this platform requires root privileges. Please run with sudo.".to_string();
        }

        if result.is_empty() {
            result = "No saved passwords found".to_string();
        }

        Ok(result)
    }
}