database.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. package database
  2. import (
  3. "context"
  4. "database/sql"
  5. "fmt"
  6. "os"
  7. "path/filepath"
  8. "time"
  9. _ "modernc.org/sqlite"
  10. "ai-status-light/internal/logger"
  11. )
  12. type DB struct {
  13. db *sql.DB
  14. conn *sql.Conn
  15. }
  16. type MQTTConfig struct {
  17. ID int `json:"id"`
  18. Broker string `json:"broker"`
  19. ClientID string `json:"client_id"`
  20. Username string `json:"username,omitempty"`
  21. Password string `json:"password,omitempty"`
  22. Topic string `json:"topic"`
  23. Enabled bool `json:"enabled"`
  24. }
  25. type BLEConfig struct {
  26. ID int `json:"id"`
  27. DeviceName string `json:"device_name"`
  28. ServiceUUID string `json:"service_uuid"`
  29. ModeCharUUID string `json:"mode_char_uuid"`
  30. ConfigCharUUID string `json:"config_char_uuid"`
  31. Enabled bool `json:"enabled"`
  32. }
  33. type DeviceConfig struct {
  34. ID int `json:"id"`
  35. DeviceName string `json:"device_name"`
  36. ConfigTopic string `json:"config_topic"`
  37. WifiSSID string `json:"wifi_ssid,omitempty"`
  38. WifiPass string `json:"wifi_pass,omitempty"`
  39. MqttBroker string `json:"mqtt_broker,omitempty"`
  40. MqttPort int `json:"mqtt_port,omitempty"`
  41. MqttUser string `json:"mqtt_user,omitempty"`
  42. MqttPass string `json:"mqtt_pass,omitempty"`
  43. MqttClient string `json:"mqtt_client,omitempty"`
  44. MqttTopic string `json:"mqtt_topic,omitempty"`
  45. MqttStatus string `json:"mqtt_status,omitempty"`
  46. PinRed int `json:"pin_red,omitempty"`
  47. PinGreen int `json:"pin_green,omitempty"`
  48. PinYellow int `json:"pin_yellow,omitempty"`
  49. Enabled bool `json:"enabled"`
  50. }
  51. type StatusRecord struct {
  52. ID int `json:"id"`
  53. Code string `json:"code"`
  54. Timestamp string `json:"timestamp"`
  55. }
  56. type WorkDuration struct {
  57. DurationMinutes int `json:"duration_minutes"`
  58. }
  59. func New(dbPath string) (*DB, error) {
  60. dir := filepath.Dir(dbPath)
  61. if err := os.MkdirAll(dir, 0755); err != nil {
  62. logger.Error("创建数据库目录失败: %v", err)
  63. return nil, fmt.Errorf("创建目录失败: %w", err)
  64. }
  65. db, err := sql.Open("sqlite", dbPath)
  66. if err != nil {
  67. logger.Error("打开数据库失败: %v", err)
  68. return nil, fmt.Errorf("打开数据库失败: %w", err)
  69. }
  70. conn, err := db.Conn(context.Background())
  71. if err != nil {
  72. logger.Error("获取数据库连接失败: %v", err)
  73. return nil, fmt.Errorf("获取连接失败: %w", err)
  74. }
  75. d := &DB{db: db, conn: conn}
  76. if err := d.init(); err != nil {
  77. logger.Error("初始化数据库失败: %v", err)
  78. return nil, err
  79. }
  80. logger.Debug("数据库已打开: %s", dbPath)
  81. return d, nil
  82. }
  83. func (d *DB) init() error {
  84. query := `
  85. CREATE TABLE IF NOT EXISTS mqtt_config (
  86. id INTEGER PRIMARY KEY AUTOINCREMENT,
  87. broker TEXT NOT NULL,
  88. client_id TEXT NOT NULL,
  89. username TEXT DEFAULT '',
  90. password TEXT DEFAULT '',
  91. topic TEXT NOT NULL,
  92. enabled BOOLEAN DEFAULT 1
  93. );
  94. `
  95. _, err := d.conn.ExecContext(context.Background(), query)
  96. if err != nil {
  97. return err
  98. }
  99. for _, col := range []string{"username", "password"} {
  100. alterQuery := fmt.Sprintf("ALTER TABLE mqtt_config ADD COLUMN %s TEXT DEFAULT ''", col)
  101. d.conn.ExecContext(context.Background(), alterQuery)
  102. }
  103. bleQuery := `
  104. CREATE TABLE IF NOT EXISTS ble_config (
  105. id INTEGER PRIMARY KEY AUTOINCREMENT,
  106. device_name TEXT NOT NULL,
  107. service_uuid TEXT NOT NULL,
  108. mode_char_uuid TEXT NOT NULL DEFAULT '',
  109. config_char_uuid TEXT NOT NULL DEFAULT '',
  110. enabled BOOLEAN DEFAULT 1
  111. );
  112. `
  113. _, err = d.conn.ExecContext(context.Background(), bleQuery)
  114. if err != nil {
  115. return err
  116. }
  117. for _, col := range []string{"service_uuid", "mode_char_uuid", "config_char_uuid"} {
  118. alterQuery := fmt.Sprintf("ALTER TABLE ble_config ADD COLUMN %s TEXT NOT NULL DEFAULT ''", col)
  119. d.conn.ExecContext(context.Background(), alterQuery)
  120. }
  121. deviceQuery := `
  122. CREATE TABLE IF NOT EXISTS device_config (
  123. id INTEGER PRIMARY KEY AUTOINCREMENT,
  124. device_name TEXT NOT NULL,
  125. config_topic TEXT NOT NULL DEFAULT 'agent/status/config',
  126. wifi_ssid TEXT DEFAULT '',
  127. wifi_pass TEXT DEFAULT '',
  128. mqtt_broker TEXT DEFAULT '',
  129. mqtt_port INTEGER DEFAULT 1883,
  130. mqtt_user TEXT DEFAULT '',
  131. mqtt_pass TEXT DEFAULT '',
  132. mqtt_client TEXT DEFAULT '',
  133. mqtt_topic TEXT DEFAULT '',
  134. mqtt_status TEXT DEFAULT '',
  135. pin_red INTEGER DEFAULT 4,
  136. pin_green INTEGER DEFAULT 3,
  137. pin_yellow INTEGER DEFAULT 2,
  138. enabled BOOLEAN DEFAULT 1
  139. );
  140. `
  141. _, err = d.conn.ExecContext(context.Background(), deviceQuery)
  142. if err != nil {
  143. return err
  144. }
  145. historyQuery := `
  146. CREATE TABLE IF NOT EXISTS status_history (
  147. id INTEGER PRIMARY KEY AUTOINCREMENT,
  148. code TEXT NOT NULL,
  149. timestamp DATETIME DEFAULT CURRENT_TIMESTAMP
  150. );
  151. `
  152. _, err = d.conn.ExecContext(context.Background(), historyQuery)
  153. if err != nil {
  154. return err
  155. }
  156. _, err = d.conn.ExecContext(context.Background(), "CREATE INDEX IF NOT EXISTS idx_status_history_timestamp ON status_history(timestamp)")
  157. if err != nil {
  158. return err
  159. }
  160. return nil
  161. }
  162. func (d *DB) GetMQTTConfig() (*MQTTConfig, error) {
  163. query := "SELECT id, broker, client_id, username, password, topic, enabled FROM mqtt_config WHERE enabled = 1 LIMIT 1"
  164. row := d.conn.QueryRowContext(context.Background(), query)
  165. var cfg MQTTConfig
  166. err := row.Scan(&cfg.ID, &cfg.Broker, &cfg.ClientID, &cfg.Username, &cfg.Password, &cfg.Topic, &cfg.Enabled)
  167. if err == sql.ErrNoRows {
  168. logger.Debug("未找到启用的 MQTT 配置")
  169. return nil, nil
  170. }
  171. if err != nil {
  172. logger.Error("查询 MQTT 配置失败: %v", err)
  173. return nil, fmt.Errorf("查询配置失败: %w", err)
  174. }
  175. logger.Debug("获取到 MQTT 配置: id=%d, broker=%s", cfg.ID, cfg.Broker)
  176. return &cfg, nil
  177. }
  178. func (d *DB) SaveMQTTConfig(cfg *MQTTConfig) error {
  179. if cfg.ID == 0 {
  180. query := "INSERT INTO mqtt_config (broker, client_id, username, password, topic, enabled) VALUES (?, ?, ?, ?, ?, ?)"
  181. _, err := d.conn.ExecContext(context.Background(), query, cfg.Broker, cfg.ClientID, cfg.Username, cfg.Password, cfg.Topic, cfg.Enabled)
  182. if err != nil {
  183. logger.Error("插入 MQTT 配置失败: %v", err)
  184. }
  185. return err
  186. }
  187. query := "UPDATE mqtt_config SET broker = ?, client_id = ?, username = ?, password = ?, topic = ?, enabled = ? WHERE id = ?"
  188. _, err := d.conn.ExecContext(context.Background(), query, cfg.Broker, cfg.ClientID, cfg.Username, cfg.Password, cfg.Topic, cfg.Enabled, cfg.ID)
  189. if err != nil {
  190. logger.Error("更新 MQTT 配置失败: id=%d, %v", cfg.ID, err)
  191. }
  192. return err
  193. }
  194. func (d *DB) DeleteMQTTConfig(id int) error {
  195. query := "DELETE FROM mqtt_config WHERE id = ?"
  196. _, err := d.conn.ExecContext(context.Background(), query, id)
  197. if err != nil {
  198. logger.Error("删除 MQTT 配置失败: id=%d, %v", id, err)
  199. } else {
  200. logger.Debug("MQTT 配置已删除: id=%d", id)
  201. }
  202. return err
  203. }
  204. func (d *DB) ListMQTTConfigs() ([]MQTTConfig, error) {
  205. query := "SELECT id, broker, client_id, username, password, topic, enabled FROM mqtt_config ORDER BY id"
  206. rows, err := d.conn.QueryContext(context.Background(), query)
  207. if err != nil {
  208. logger.Error("查询 MQTT 配置列表失败: %v", err)
  209. return nil, err
  210. }
  211. defer rows.Close()
  212. var configs []MQTTConfig
  213. for rows.Next() {
  214. var cfg MQTTConfig
  215. if err := rows.Scan(&cfg.ID, &cfg.Broker, &cfg.ClientID, &cfg.Username, &cfg.Password, &cfg.Topic, &cfg.Enabled); err != nil {
  216. logger.Warn("扫描 MQTT 配置行失败: %v", err)
  217. continue
  218. }
  219. configs = append(configs, cfg)
  220. }
  221. logger.Debug("查询到 %d 条 MQTT 配置", len(configs))
  222. return configs, nil
  223. }
  224. func (d *DB) Close() error {
  225. logger.Debug("数据库连接已关闭")
  226. if err := d.conn.Close(); err != nil {
  227. return err
  228. }
  229. return d.db.Close()
  230. }
  231. func (d *DB) GetBLEConfig() (*BLEConfig, error) {
  232. query := "SELECT id, device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled FROM ble_config WHERE enabled = 1 LIMIT 1"
  233. row := d.conn.QueryRowContext(context.Background(), query)
  234. var cfg BLEConfig
  235. err := row.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ServiceUUID, &cfg.ModeCharUUID, &cfg.ConfigCharUUID, &cfg.Enabled)
  236. if err == sql.ErrNoRows {
  237. logger.Debug("未找到启用的 BLE 配置")
  238. return nil, nil
  239. }
  240. if err != nil {
  241. logger.Error("查询 BLE 配置失败: %v", err)
  242. return nil, fmt.Errorf("查询配置失败: %w", err)
  243. }
  244. logger.Debug("获取到 BLE 配置: id=%d, device=%s", cfg.ID, cfg.DeviceName)
  245. return &cfg, nil
  246. }
  247. func (d *DB) SaveBLEConfig(cfg *BLEConfig) error {
  248. if cfg.ID == 0 {
  249. query := "INSERT INTO ble_config (device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled) VALUES (?, ?, ?, ?, ?)"
  250. _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ServiceUUID, cfg.ModeCharUUID, cfg.ConfigCharUUID, cfg.Enabled)
  251. if err != nil {
  252. logger.Error("插入 BLE 配置失败: %v", err)
  253. }
  254. return err
  255. }
  256. query := "UPDATE ble_config SET device_name = ?, service_uuid = ?, mode_char_uuid = ?, config_char_uuid = ?, enabled = ? WHERE id = ?"
  257. _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ServiceUUID, cfg.ModeCharUUID, cfg.ConfigCharUUID, cfg.Enabled, cfg.ID)
  258. if err != nil {
  259. logger.Error("更新 BLE 配置失败: id=%d, %v", cfg.ID, err)
  260. }
  261. return err
  262. }
  263. func (d *DB) DeleteBLEConfig(id int) error {
  264. query := "DELETE FROM ble_config WHERE id = ?"
  265. _, err := d.conn.ExecContext(context.Background(), query, id)
  266. if err != nil {
  267. logger.Error("删除 BLE 配置失败: id=%d, %v", id, err)
  268. } else {
  269. logger.Debug("BLE 配置已删除: id=%d", id)
  270. }
  271. return err
  272. }
  273. func (d *DB) ListBLEConfigs() ([]BLEConfig, error) {
  274. query := "SELECT id, device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled FROM ble_config ORDER BY id"
  275. rows, err := d.conn.QueryContext(context.Background(), query)
  276. if err != nil {
  277. logger.Error("查询 BLE 配置列表失败: %v", err)
  278. return nil, err
  279. }
  280. defer rows.Close()
  281. var configs []BLEConfig
  282. for rows.Next() {
  283. var cfg BLEConfig
  284. if err := rows.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ServiceUUID, &cfg.ModeCharUUID, &cfg.ConfigCharUUID, &cfg.Enabled); err != nil {
  285. logger.Warn("扫描 BLE 配置行失败: %v", err)
  286. continue
  287. }
  288. configs = append(configs, cfg)
  289. }
  290. logger.Debug("查询到 %d 条 BLE 配置", len(configs))
  291. return configs, nil
  292. }
  293. func (d *DB) GetDeviceConfig() (*DeviceConfig, error) {
  294. query := "SELECT id, device_name, config_topic, wifi_ssid, wifi_pass, mqtt_broker, mqtt_port, mqtt_user, mqtt_pass, mqtt_client, mqtt_topic, mqtt_status, pin_red, pin_green, pin_yellow, enabled FROM device_config WHERE enabled = 1 LIMIT 1"
  295. row := d.conn.QueryRowContext(context.Background(), query)
  296. var cfg DeviceConfig
  297. err := row.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ConfigTopic, &cfg.WifiSSID, &cfg.WifiPass, &cfg.MqttBroker, &cfg.MqttPort, &cfg.MqttUser, &cfg.MqttPass, &cfg.MqttClient, &cfg.MqttTopic, &cfg.MqttStatus, &cfg.PinRed, &cfg.PinGreen, &cfg.PinYellow, &cfg.Enabled)
  298. if err == sql.ErrNoRows {
  299. logger.Debug("未找到启用的设备配置")
  300. return nil, nil
  301. }
  302. if err != nil {
  303. logger.Error("查询设备配置失败: %v", err)
  304. return nil, fmt.Errorf("查询配置失败: %w", err)
  305. }
  306. logger.Debug("获取到设备配置: id=%d, device=%s", cfg.ID, cfg.DeviceName)
  307. return &cfg, nil
  308. }
  309. func (d *DB) SaveDeviceConfig(cfg *DeviceConfig) error {
  310. if cfg.ID == 0 {
  311. query := "INSERT INTO device_config (device_name, config_topic, wifi_ssid, wifi_pass, mqtt_broker, mqtt_port, mqtt_user, mqtt_pass, mqtt_client, mqtt_topic, mqtt_status, pin_red, pin_green, pin_yellow, enabled) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)"
  312. _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ConfigTopic, cfg.WifiSSID, cfg.WifiPass, cfg.MqttBroker, cfg.MqttPort, cfg.MqttUser, cfg.MqttPass, cfg.MqttClient, cfg.MqttTopic, cfg.MqttStatus, cfg.PinRed, cfg.PinGreen, cfg.PinYellow, cfg.Enabled)
  313. if err != nil {
  314. logger.Error("插入设备配置失败: %v", err)
  315. }
  316. return err
  317. }
  318. query := "UPDATE device_config SET device_name = ?, config_topic = ?, wifi_ssid = ?, wifi_pass = ?, mqtt_broker = ?, mqtt_port = ?, mqtt_user = ?, mqtt_pass = ?, mqtt_client = ?, mqtt_topic = ?, mqtt_status = ?, pin_red = ?, pin_green = ?, pin_yellow = ?, enabled = ? WHERE id = ?"
  319. _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ConfigTopic, cfg.WifiSSID, cfg.WifiPass, cfg.MqttBroker, cfg.MqttPort, cfg.MqttUser, cfg.MqttPass, cfg.MqttClient, cfg.MqttTopic, cfg.MqttStatus, cfg.PinRed, cfg.PinGreen, cfg.PinYellow, cfg.Enabled, cfg.ID)
  320. if err != nil {
  321. logger.Error("更新设备配置失败: id=%d, %v", cfg.ID, err)
  322. }
  323. return err
  324. }
  325. func (d *DB) DeleteDeviceConfig(id int) error {
  326. query := "DELETE FROM device_config WHERE id = ?"
  327. _, err := d.conn.ExecContext(context.Background(), query, id)
  328. if err != nil {
  329. logger.Error("删除设备配置失败: id=%d, %v", id, err)
  330. } else {
  331. logger.Debug("设备配置已删除: id=%d", id)
  332. }
  333. return err
  334. }
  335. func (d *DB) ListDeviceConfigs() ([]DeviceConfig, error) {
  336. query := "SELECT id, device_name, config_topic, wifi_ssid, wifi_pass, mqtt_broker, mqtt_port, mqtt_user, mqtt_pass, mqtt_client, mqtt_topic, mqtt_status, pin_red, pin_green, pin_yellow, enabled FROM device_config ORDER BY id"
  337. rows, err := d.conn.QueryContext(context.Background(), query)
  338. if err != nil {
  339. logger.Error("查询设备配置列表失败: %v", err)
  340. return nil, err
  341. }
  342. defer rows.Close()
  343. var configs []DeviceConfig
  344. for rows.Next() {
  345. var cfg DeviceConfig
  346. if err := rows.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ConfigTopic, &cfg.WifiSSID, &cfg.WifiPass, &cfg.MqttBroker, &cfg.MqttPort, &cfg.MqttUser, &cfg.MqttPass, &cfg.MqttClient, &cfg.MqttTopic, &cfg.MqttStatus, &cfg.PinRed, &cfg.PinGreen, &cfg.PinYellow, &cfg.Enabled); err != nil {
  347. logger.Warn("扫描设备配置行失败: %v", err)
  348. continue
  349. }
  350. configs = append(configs, cfg)
  351. }
  352. logger.Debug("查询到 %d 条设备配置", len(configs))
  353. return configs, nil
  354. }
  355. func (d *DB) SaveStatusRecord(code string) error {
  356. query := "INSERT INTO status_history (code, timestamp) VALUES (?, datetime('now'))"
  357. _, err := d.conn.ExecContext(context.Background(), query, code)
  358. if err != nil {
  359. logger.Error("保存状态记录失败: %v", err)
  360. }
  361. return err
  362. }
  363. func (d *DB) GetStatusHistory(limit int) ([]StatusRecord, error) {
  364. if limit <= 0 {
  365. limit = 100
  366. }
  367. query := "SELECT id, code, timestamp FROM status_history ORDER BY timestamp DESC LIMIT ?"
  368. rows, err := d.conn.QueryContext(context.Background(), query, limit)
  369. if err != nil {
  370. logger.Error("查询状态历史失败: %v", err)
  371. return nil, err
  372. }
  373. defer rows.Close()
  374. var records []StatusRecord
  375. for rows.Next() {
  376. var r StatusRecord
  377. if err := rows.Scan(&r.ID, &r.Code, &r.Timestamp); err != nil {
  378. logger.Warn("扫描状态记录行失败: %v", err)
  379. continue
  380. }
  381. records = append(records, r)
  382. }
  383. logger.Debug("查询到 %d 条状态记录", len(records))
  384. return records, nil
  385. }
  386. func (d *DB) CleanOldStatusRecords(hours int) error {
  387. if hours <= 0 {
  388. hours = 2
  389. }
  390. query := "DELETE FROM status_history WHERE timestamp < datetime('now', ?)"
  391. param := fmt.Sprintf("-%d hours", hours)
  392. result, err := d.conn.ExecContext(context.Background(), query, param)
  393. if err != nil {
  394. logger.Error("清理旧状态记录失败: %v", err)
  395. return err
  396. }
  397. rowsAffected, _ := result.RowsAffected()
  398. if rowsAffected > 0 {
  399. logger.Info("已清理 %d 条超过 %d 小时的状态记录", rowsAffected, hours)
  400. }
  401. return nil
  402. }
  403. func (d *DB) GetTodayWorkDuration() (*WorkDuration, error) {
  404. query := "SELECT code, timestamp FROM status_history WHERE timestamp >= date('now', 'localtime') ORDER BY timestamp ASC"
  405. rows, err := d.conn.QueryContext(context.Background(), query)
  406. if err != nil {
  407. logger.Error("查询今日工作时长失败: %v", err)
  408. return nil, err
  409. }
  410. defer rows.Close()
  411. type record struct {
  412. Code string
  413. Time time.Time
  414. }
  415. var records []record
  416. for rows.Next() {
  417. var code, ts string
  418. if err := rows.Scan(&code, &ts); err != nil {
  419. continue
  420. }
  421. t, err := time.ParseInLocation("2006-01-02 15:04:05", ts, time.Local)
  422. if err != nil {
  423. logger.Warn("解析时间戳失败: %s, %v", ts, err)
  424. continue
  425. }
  426. records = append(records, record{Code: code, Time: t})
  427. }
  428. active := map[string]bool{
  429. "busy": true, "reasoning": true, "using_tool": true,
  430. "running": true, "pending": true, "retry": true,
  431. }
  432. var totalMs int64
  433. now := time.Now()
  434. for i, r := range records {
  435. if !active[r.Code] {
  436. continue
  437. }
  438. if i+1 < len(records) {
  439. totalMs += records[i+1].Time.Sub(r.Time).Milliseconds()
  440. } else if active[r.Code] {
  441. totalMs += now.Sub(r.Time).Milliseconds()
  442. }
  443. }
  444. logger.Debug("今日工作时长: %d 分钟", int(totalMs/60000))
  445. return &WorkDuration{DurationMinutes: int(totalMs / 60000)}, nil
  446. }