use crate::DriverCallback;
use crate::DriverContext;
use crate::{
DriverCategory,
operating_system_memory::common::ProcessMemory,
types::{Driver, DriverParameter},
};
use anyhow::Result;
use serde_json::{Value, json};
use std::collections::HashMap;
#[derive(Debug)]
pub struct MemoryReadDriver;
#[async_trait::async_trait]
impl Driver for MemoryReadDriver {
fn name(&self) -> &str {
"memory_read"
}
fn description(&self) -> &str {
"Read memory from a running process at a specified address"
}
fn usage_hint(&self) -> &str {
"Use this skill to inspect process memory for debugging or analysis. This is a read-only operation."
}
fn parameters(&self) -> Vec<DriverParameter> {
vec![
DriverParameter {
name: "pid".to_string(),
param_type: "integer".to_string(),
description: "Process ID to read from".to_string(),
required: true,
default: None,
example: Some(Value::Number(1234.into())),
enum_values: None,
},
DriverParameter {
name: "address".to_string(),
param_type: "string".to_string(),
description: "Memory address to read (hex format, e.g., '0x7FF6A1B4C000')"
.to_string(),
required: true,
default: None,
example: Some(Value::String("0x7FF6A1B4C000".to_string())),
enum_values: None,
},
DriverParameter {
name: "size".to_string(),
param_type: "integer".to_string(),
description: "Number of bytes to read".to_string(),
required: true,
default: None,
example: Some(Value::Number(4.into())),
enum_values: None,
},
DriverParameter {
name: "data_type".to_string(),
param_type: "string".to_string(),
description: "Data type: bytes, u8, u16, u32, u64, f32, f64, string".to_string(),
required: false,
default: Some(Value::String("bytes".to_string())),
example: Some(Value::String("u32".to_string())),
enum_values: Some(vec![
"bytes".to_string(),
"u8".to_string(),
"u16".to_string(),
"u32".to_string(),
"u64".to_string(),
"f32".to_string(),
"f64".to_string(),
"string".to_string(),
]),
},
]
}
fn example_call(&self) -> Value {
json!({
"action": "memory_read",
"parameters": {
"pid": 1234,
"address": "0x7FF6A1B4C000",
"size": 4,
"data_type": "u32"
}
})
}
fn example_output(&self) -> String {
"Value: 12345".to_string()
}
fn category(&self) -> DriverCategory {
DriverCategory::OperatingSystemMemory
}
async fn execute(
&self,
parameters: &HashMap<String, Value>,
callback: Option<&dyn DriverCallback>,
context: Option<&DriverContext>,
) -> Result<String> {
let pid = parameters
.get("pid")
.and_then(|v| v.as_u64())
.ok_or_else(|| anyhow::anyhow!("Missing required parameter: pid"))?
as u32;
let address_str = parameters
.get("address")
.and_then(|v| v.as_str())
.ok_or_else(|| anyhow::anyhow!("Missing required parameter: address"))?;
let address = usize::from_str_radix(address_str.trim_start_matches("0x"), 16)
.map_err(|e| anyhow::anyhow!("Invalid address format: {}: {}", address_str, e))?;
let data_type = parameters
.get("data_type")
.and_then(|v| v.as_str())
.unwrap_or("bytes");
let size = parameters.get("size").and_then(|v| v.as_u64()).unwrap_or(0) as usize;
let mut memory = ProcessMemory::open(pid, true)
.map_err(|e| anyhow::anyhow!("Failed to open process: {}", e))?;
match data_type {
"u8" => {
let value = memory
.read_u8(address)
.map_err(|e| anyhow::anyhow!("Failed to read u8: {}", e))?;
Ok(format!("Value: {}", value))
}
"u16" => {
let value = memory
.read_u16(address)
.map_err(|e| anyhow::anyhow!("Failed to read u16: {}", e))?;
Ok(format!("Value: {}", value))
}
"u32" => {
let value = memory
.read_u32(address)
.map_err(|e| anyhow::anyhow!("Failed to read u32: {}", e))?;
Ok(format!("Value: {}", value))
}
"u64" => {
let value = memory
.read_u64(address)
.map_err(|e| anyhow::anyhow!("Failed to read u64: {}", e))?;
Ok(format!("Value: {}", value))
}
"f32" => {
let value = memory
.read_f32(address)
.map_err(|e| anyhow::anyhow!("Failed to read f32: {}", e))?;
Ok(format!("Value: {:.6}", value))
}
"f64" => {
let value = memory
.read_f64(address)
.map_err(|e| anyhow::anyhow!("Failed to read f64: {}", e))?;
Ok(format!("Value: {:.6}", value))
}
"string" => {
let max_len = if size > 0 { size } else { 256 };
let value = memory
.read_string(address, max_len)
.map_err(|e| anyhow::anyhow!("Failed to read string: {}", e))?;
Ok(format!("String: \"{}\"", value))
}
_ => {
let read_size = if size > 0 { size } else { 16 };
let mut buffer = vec![0u8; read_size];
let bytes_read = memory
.read_memory(address, &mut buffer)
.map_err(|e| anyhow::anyhow!("Failed to read memory: {}", e))?;
let hex: String = buffer[..bytes_read]
.iter()
.map(|b| format!("{:02X}", b))
.collect::<Vec<_>>()
.join(" ");
Ok(format!("Bytes ({} bytes): {}", bytes_read, hex))
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_memory_read_skill_metadata() {
let skill = MemoryReadDriver;
assert_eq!(skill.name(), "memory_read");
assert_eq!(skill.category(), DriverCategory::OperatingSystemMemory);
assert!(!skill.parameters().is_empty());
}
}