use reqwest::Client; use rquickjs::{Context, Runtime, Function}; use serde_json::Value; use std::collections::HashMap; use std::time::Duration; use crate::error::AppError; /// 执行用量查询脚本 pub async fn execute_usage_script( script_code: &str, api_key: &str, base_url: &str, timeout_secs: u64, ) -> Result { // 1. 替换变量 let replaced = script_code .replace("{{apiKey}}", api_key) .replace("{{baseUrl}}", base_url); // 2. 在独立作用域中提取 request 配置(确保 Runtime/Context 在 await 前释放) let request_config = { let runtime = Runtime::new() .map_err(|e| AppError::Message(format!("创建 JS 运行时失败: {}", e)))?; let context = Context::full(&runtime) .map_err(|e| AppError::Message(format!("创建 JS 上下文失败: {}", e)))?; context.with(|ctx| { // 执行用户代码,获取配置对象 let config: rquickjs::Object = ctx .eval(replaced.clone()) .map_err(|e| AppError::Message(format!("解析配置失败: {}", e)))?; // 提取 request 配置 let request: rquickjs::Object = config .get("request") .map_err(|e| AppError::Message(format!("缺少 request 配置: {}", e)))?; // 将 request 转换为 JSON 字符串 let request_json: String = ctx .json_stringify(request) .map_err(|e| AppError::Message(format!("序列化 request 失败: {}", e)))? .ok_or_else(|| AppError::Message("序列化返回 None".into()))? .get() .map_err(|e| AppError::Message(format!("获取字符串失败: {}", e)))?; Ok::<_, AppError>(request_json) })? }; // Runtime 和 Context 在这里被 drop // 3. 解析 request 配置 let request: RequestConfig = serde_json::from_str(&request_config) .map_err(|e| AppError::Message(format!("request 配置格式错误: {}", e)))?; // 4. 发送 HTTP 请求 let response_data = send_http_request(&request, timeout_secs).await?; // 5. 在独立作用域中执行 extractor(确保 Runtime/Context 在函数结束前释放) let result: Value = { let runtime = Runtime::new() .map_err(|e| AppError::Message(format!("创建 JS 运行时失败: {}", e)))?; let context = Context::full(&runtime) .map_err(|e| AppError::Message(format!("创建 JS 上下文失败: {}", e)))?; context.with(|ctx| { // 重新 eval 获取配置对象 let config: rquickjs::Object = ctx .eval(replaced.clone()) .map_err(|e| AppError::Message(format!("重新解析配置失败: {}", e)))?; // 提取 extractor 函数 let extractor: Function = config .get("extractor") .map_err(|e| AppError::Message(format!("缺少 extractor 函数: {}", e)))?; // 将响应数据转换为 JS 值 let response_js: rquickjs::Value = ctx .json_parse(response_data.as_str()) .map_err(|e| AppError::Message(format!("解析响应 JSON 失败: {}", e)))?; // 调用 extractor(response) let result_js: rquickjs::Value = extractor .call((response_js,)) .map_err(|e| AppError::Message(format!("执行 extractor 失败: {}", e)))?; // 转换为 JSON 字符串 let result_json: String = ctx .json_stringify(result_js) .map_err(|e| AppError::Message(format!("序列化结果失败: {}", e)))? .ok_or_else(|| AppError::Message("序列化返回 None".into()))? .get() .map_err(|e| AppError::Message(format!("获取字符串失败: {}", e)))?; // 解析为 serde_json::Value serde_json::from_str(&result_json) .map_err(|e| AppError::Message(format!("JSON 解析失败: {}", e))) })? }; // Runtime 和 Context 在这里被 drop // 6. 验证返回值格式 validate_result(&result)?; Ok(result) } /// 请求配置结构 #[derive(Debug, serde::Deserialize)] struct RequestConfig { url: String, method: String, #[serde(default)] headers: HashMap, #[serde(default)] body: Option, } /// 发送 HTTP 请求 async fn send_http_request(config: &RequestConfig, timeout_secs: u64) -> Result { let client = Client::builder() .timeout(Duration::from_secs(timeout_secs)) .build() .map_err(|e| AppError::Message(format!("创建客户端失败: {}", e)))?; let method = config .method .parse() .unwrap_or(reqwest::Method::GET); let mut req = client.request(method.clone(), &config.url); // 添加请求头 for (k, v) in &config.headers { req = req.header(k, v); } // 添加请求体 if let Some(body) = &config.body { req = req.body(body.clone()); } // 发送请求 let resp = req .send() .await .map_err(|e| AppError::Message(format!("请求失败: {}", e)))?; let status = resp.status(); let text = resp .text() .await .map_err(|e| AppError::Message(format!("读取响应失败: {}", e)))?; if !status.is_success() { let preview = if text.len() > 200 { format!("{}...", &text[..200]) } else { text.clone() }; return Err(AppError::Message(format!("HTTP {} : {}", status, preview))); } Ok(text) } /// 验证脚本返回值(支持单对象或数组) fn validate_result(result: &Value) -> Result<(), AppError> { // 如果是数组,验证每个元素 if let Some(arr) = result.as_array() { if arr.is_empty() { return Err(AppError::InvalidInput("脚本返回的数组不能为空".into())); } for (idx, item) in arr.iter().enumerate() { validate_single_usage(item) .map_err(|e| { AppError::InvalidInput(format!("数组索引[{}]验证失败: {}", idx, e)) })?; } return Ok(()); } // 如果是单对象,直接验证(向后兼容) validate_single_usage(result) } /// 验证单个用量数据对象 fn validate_single_usage(result: &Value) -> Result<(), AppError> { let obj = result.as_object().ok_or_else(|| { AppError::InvalidInput("脚本必须返回对象或对象数组".into()) })?; // 所有字段均为可选,只进行类型检查 if obj.contains_key("isValid") && !result["isValid"].is_null() && !result["isValid"].is_boolean() { return Err(AppError::InvalidInput( "isValid 必须是布尔值或 null".into(), )); } if obj.contains_key("invalidMessage") && !result["invalidMessage"].is_null() && !result["invalidMessage"].is_string() { return Err(AppError::InvalidInput( "invalidMessage 必须是字符串或 null".into(), )); } if obj.contains_key("remaining") && !result["remaining"].is_null() && !result["remaining"].is_number() { return Err(AppError::InvalidInput( "remaining 必须是数字或 null".into(), )); } if obj.contains_key("unit") && !result["unit"].is_null() && !result["unit"].is_string() { return Err(AppError::InvalidInput( "unit 必须是字符串或 null".into(), )); } if obj.contains_key("total") && !result["total"].is_null() && !result["total"].is_number() { return Err(AppError::InvalidInput( "total 必须是数字或 null".into(), )); } if obj.contains_key("used") && !result["used"].is_null() && !result["used"].is_number() { return Err(AppError::InvalidInput( "used 必须是数字或 null".into(), )); } if obj.contains_key("planName") && !result["planName"].is_null() && !result["planName"].is_string() { return Err(AppError::InvalidInput( "planName 必须是字符串或 null".into(), )); } if obj.contains_key("extra") && !result["extra"].is_null() && !result["extra"].is_string() { return Err(AppError::InvalidInput( "extra 必须是字符串或 null".into(), )); } Ok(()) }