a058b0ab8e
- 修复记忆管理数据库连接不可用 (ai-core重编译+Unicode修复) - 修复IoT子会话工具调用链路日志缺失 - 新增最终审查子会话(review_provider) 支持消息格式解析拆分 - 实现历史消息持久化(后端存储+前端分页加载) - 前端新增动作消息(ActionMessage)类型和渲染 - 优化对话链路速度(非阻塞子会话+快速问候通道) - JWT密钥环境变量化(无默认值启动panic) - Token自动刷新机制(401拦截器+refresh接口) - WebSocket指数退避重连(jitter+最大10次) - localStorage清理一致性(cyrene_前缀+版本检查) - IoT环境变量统一为IOT_SERVICE_URL
208 lines
6.1 KiB
Go
208 lines
6.1 KiB
Go
package tools
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"net/http"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// IoTClient IoT 调试服务 HTTP 客户端
|
|
type IoTClient struct {
|
|
baseURL string
|
|
client *http.Client
|
|
|
|
// 缓存控制
|
|
mu sync.RWMutex
|
|
cache []IoTDevice
|
|
cacheTime time.Time
|
|
cacheTTL time.Duration
|
|
}
|
|
|
|
// NewIoTClient 创建 IoT 客户端
|
|
func NewIoTClient(baseURL string) *IoTClient {
|
|
return &IoTClient{
|
|
baseURL: baseURL,
|
|
client: &http.Client{
|
|
Timeout: 5 * time.Second,
|
|
},
|
|
cacheTTL: 60 * time.Second,
|
|
}
|
|
}
|
|
|
|
// GetAllDevices 获取所有设备列表(带缓存)
|
|
func (c *IoTClient) GetAllDevices() ([]IoTDevice, error) {
|
|
// 检查缓存
|
|
c.mu.RLock()
|
|
if c.cache != nil && time.Since(c.cacheTime) < c.cacheTTL {
|
|
devices := make([]IoTDevice, len(c.cache))
|
|
copy(devices, c.cache)
|
|
c.mu.RUnlock()
|
|
return devices, nil
|
|
}
|
|
c.mu.RUnlock()
|
|
|
|
// 请求 API
|
|
resp, err := c.client.Get(c.baseURL + "/api/v1/devices")
|
|
if err != nil {
|
|
log.Printf("[IoT客户端] 请求失败: %v", err)
|
|
return nil, fmt.Errorf("获取设备列表失败: %w", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
return nil, fmt.Errorf("获取设备列表返回状态码 %d", resp.StatusCode)
|
|
}
|
|
|
|
var result struct {
|
|
Devices []IoTDevice `json:"devices"`
|
|
Total int `json:"total"`
|
|
}
|
|
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
|
|
return nil, fmt.Errorf("解析设备列表失败: %w", err)
|
|
}
|
|
|
|
// 更新缓存
|
|
c.mu.Lock()
|
|
c.cache = result.Devices
|
|
c.cacheTime = time.Now()
|
|
c.mu.Unlock()
|
|
|
|
return result.Devices, nil
|
|
}
|
|
|
|
// GetDevice 获取单个设备详情
|
|
func (c *IoTClient) GetDevice(id string) (*IoTDevice, error) {
|
|
resp, err := c.client.Get(c.baseURL + "/api/v1/devices/" + id)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("获取设备 %s 失败: %w", id, err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
return nil, fmt.Errorf("设备 %s 不存在", id)
|
|
}
|
|
if resp.StatusCode != http.StatusOK {
|
|
return nil, fmt.Errorf("获取设备 %s 返回状态码 %d", id, resp.StatusCode)
|
|
}
|
|
|
|
var result struct {
|
|
Device IoTDevice `json:"device"`
|
|
}
|
|
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
|
|
return nil, fmt.Errorf("解析设备信息失败: %w", err)
|
|
}
|
|
|
|
return &result.Device, nil
|
|
}
|
|
|
|
// ToggleDevice 切换设备开关状态
|
|
func (c *IoTClient) ToggleDevice(id string) error {
|
|
log.Printf("[tool-engine:IoT-client] 🔄 切换设备: id=%s, url=%s", id, c.baseURL+"/api/v1/devices/"+id+"/toggle")
|
|
|
|
req, err := http.NewRequest(http.MethodPost, c.baseURL+"/api/v1/devices/"+id+"/toggle", nil)
|
|
if err != nil {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 创建切换请求失败: device=%s, err=%v", id, err)
|
|
return fmt.Errorf("创建切换请求失败: %w", err)
|
|
}
|
|
|
|
resp, err := c.client.Do(req)
|
|
if err != nil {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 切换设备 HTTP 失败: device=%s, err=%v", id, err)
|
|
return fmt.Errorf("切换设备 %s 失败: %w", id, err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 设备不存在: %s", id)
|
|
return fmt.Errorf("设备 %s 不存在", id)
|
|
}
|
|
if resp.StatusCode != http.StatusOK {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 切换设备返回非200: device=%s, status=%d", id, resp.StatusCode)
|
|
return fmt.Errorf("切换设备 %s 返回状态码 %d", id, resp.StatusCode)
|
|
}
|
|
|
|
// 切换后清除缓存,确保下次查询获取最新状态
|
|
c.mu.Lock()
|
|
c.cache = nil
|
|
c.mu.Unlock()
|
|
|
|
log.Printf("[tool-engine:IoT-client] ✅ 切换设备成功: %s", id)
|
|
return nil
|
|
}
|
|
|
|
// SetDeviceProperty 设置设备属性(温度、亮度、位置、模式、颜色等)
|
|
func (c *IoTClient) SetDeviceProperty(id string, field string, value interface{}) error {
|
|
log.Printf("[tool-engine:IoT-client] 🔧 设置设备属性: device=%s, field=%s, value=%v, url=%s", id, field, value, c.baseURL+"/api/v1/devices/"+id+"/set")
|
|
|
|
body, err := json.Marshal(map[string]interface{}{
|
|
"field": field,
|
|
"value": value,
|
|
})
|
|
if err != nil {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 序列化请求失败: device=%s, err=%v", id, err)
|
|
return fmt.Errorf("序列化请求失败: %w", err)
|
|
}
|
|
|
|
req, err := http.NewRequest(http.MethodPost, c.baseURL+"/api/v1/devices/"+id+"/set", nil)
|
|
if err != nil {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 创建设置请求失败: device=%s, err=%v", id, err)
|
|
return fmt.Errorf("创建设置请求失败: %w", err)
|
|
}
|
|
req.Header.Set("Content-Type", "application/json")
|
|
req.Body = io.NopCloser(bytes.NewReader(body))
|
|
|
|
resp, err := c.client.Do(req)
|
|
if err != nil {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 设置设备属性 HTTP 失败: device=%s, field=%s, err=%v", id, field, err)
|
|
return fmt.Errorf("设置设备 %s 属性失败: %w", id, err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode == http.StatusNotFound {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 设备不存在: %s", id)
|
|
return fmt.Errorf("设备 %s 不存在", id)
|
|
}
|
|
if resp.StatusCode != http.StatusOK {
|
|
var errResp struct {
|
|
Error string `json:"error"`
|
|
}
|
|
json.NewDecoder(resp.Body).Decode(&errResp)
|
|
if errResp.Error != "" {
|
|
log.Printf("[tool-engine:IoT-client] ❌ 设置设备属性失败: device=%s, err=%s", id, errResp.Error)
|
|
return fmt.Errorf("设置设备 %s 属性失败: %s", id, errResp.Error)
|
|
}
|
|
log.Printf("[tool-engine:IoT-client] ❌ 设置设备属性返回非200: device=%s, status=%d", id, resp.StatusCode)
|
|
return fmt.Errorf("设置设备 %s 属性返回状态码 %d", id, resp.StatusCode)
|
|
}
|
|
|
|
// 修改后清除缓存
|
|
c.mu.Lock()
|
|
c.cache = nil
|
|
c.mu.Unlock()
|
|
|
|
log.Printf("[tool-engine:IoT-client] ✅ 设置设备属性成功: device=%s, field=%s, value=%v", id, field, value)
|
|
return nil
|
|
}
|
|
|
|
// GetDevicesForContext 获取设备状态摘要(供上下文注入使用,失败不报错)
|
|
func (c *IoTClient) GetDevicesForContext() []IoTDevice {
|
|
devices, err := c.GetAllDevices()
|
|
if err != nil {
|
|
log.Printf("[IoT客户端] 获取设备状态摘要失败: %v", err)
|
|
return nil
|
|
}
|
|
return devices
|
|
}
|
|
|
|
// InvalidateCache 使缓存失效
|
|
func (c *IoTClient) InvalidateCache() {
|
|
c.mu.Lock()
|
|
c.cache = nil
|
|
c.mu.Unlock()
|
|
}
|