| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366 |
- package database
- import (
- "context"
- "database/sql"
- "fmt"
- "os"
- "path/filepath"
- _ "modernc.org/sqlite"
- "ai-status-light/internal/logger"
- )
- type DB struct {
- db *sql.DB
- conn *sql.Conn
- }
- type MQTTConfig struct {
- ID int `json:"id"`
- Broker string `json:"broker"`
- ClientID string `json:"client_id"`
- Username string `json:"username,omitempty"`
- Password string `json:"password,omitempty"`
- Topic string `json:"topic"`
- Enabled bool `json:"enabled"`
- }
- type BLEConfig struct {
- ID int `json:"id"`
- DeviceName string `json:"device_name"`
- ServiceUUID string `json:"service_uuid"`
- ModeCharUUID string `json:"mode_char_uuid"`
- ConfigCharUUID string `json:"config_char_uuid"`
- Enabled bool `json:"enabled"`
- }
- type DeviceConfig struct {
- ID int `json:"id"`
- DeviceName string `json:"device_name"`
- ConfigTopic string `json:"config_topic"`
- WifiSSID string `json:"wifi_ssid,omitempty"`
- WifiPass string `json:"wifi_pass,omitempty"`
- MqttBroker string `json:"mqtt_broker,omitempty"`
- MqttPort int `json:"mqtt_port,omitempty"`
- MqttUser string `json:"mqtt_user,omitempty"`
- MqttPass string `json:"mqtt_pass,omitempty"`
- MqttClient string `json:"mqtt_client,omitempty"`
- MqttTopic string `json:"mqtt_topic,omitempty"`
- MqttStatus string `json:"mqtt_status,omitempty"`
- PinRed int `json:"pin_red,omitempty"`
- PinGreen int `json:"pin_green,omitempty"`
- PinYellow int `json:"pin_yellow,omitempty"`
- Enabled bool `json:"enabled"`
- }
- func New(dbPath string) (*DB, error) {
- dir := filepath.Dir(dbPath)
- if err := os.MkdirAll(dir, 0755); err != nil {
- logger.Error("创建数据库目录失败: %v", err)
- return nil, fmt.Errorf("创建目录失败: %w", err)
- }
- db, err := sql.Open("sqlite", dbPath)
- if err != nil {
- logger.Error("打开数据库失败: %v", err)
- return nil, fmt.Errorf("打开数据库失败: %w", err)
- }
- conn, err := db.Conn(context.Background())
- if err != nil {
- logger.Error("获取数据库连接失败: %v", err)
- return nil, fmt.Errorf("获取连接失败: %w", err)
- }
- d := &DB{db: db, conn: conn}
- if err := d.init(); err != nil {
- logger.Error("初始化数据库失败: %v", err)
- return nil, err
- }
- logger.Debug("数据库已打开: %s", dbPath)
- return d, nil
- }
- func (d *DB) init() error {
- query := `
- CREATE TABLE IF NOT EXISTS mqtt_config (
- id INTEGER PRIMARY KEY AUTOINCREMENT,
- broker TEXT NOT NULL,
- client_id TEXT NOT NULL,
- username TEXT DEFAULT '',
- password TEXT DEFAULT '',
- topic TEXT NOT NULL,
- enabled BOOLEAN DEFAULT 1
- );
- `
- _, err := d.conn.ExecContext(context.Background(), query)
- if err != nil {
- return err
- }
- for _, col := range []string{"username", "password"} {
- alterQuery := fmt.Sprintf("ALTER TABLE mqtt_config ADD COLUMN %s TEXT DEFAULT ''", col)
- d.conn.ExecContext(context.Background(), alterQuery)
- }
- bleQuery := `
- CREATE TABLE IF NOT EXISTS ble_config (
- id INTEGER PRIMARY KEY AUTOINCREMENT,
- device_name TEXT NOT NULL,
- service_uuid TEXT NOT NULL,
- mode_char_uuid TEXT NOT NULL DEFAULT '',
- config_char_uuid TEXT NOT NULL DEFAULT '',
- enabled BOOLEAN DEFAULT 1
- );
- `
- _, err = d.conn.ExecContext(context.Background(), bleQuery)
- if err != nil {
- return err
- }
- for _, col := range []string{"service_uuid", "mode_char_uuid", "config_char_uuid"} {
- alterQuery := fmt.Sprintf("ALTER TABLE ble_config ADD COLUMN %s TEXT NOT NULL DEFAULT ''", col)
- d.conn.ExecContext(context.Background(), alterQuery)
- }
- deviceQuery := `
- CREATE TABLE IF NOT EXISTS device_config (
- id INTEGER PRIMARY KEY AUTOINCREMENT,
- device_name TEXT NOT NULL,
- config_topic TEXT NOT NULL DEFAULT 'agent/status/config',
- wifi_ssid TEXT DEFAULT '',
- wifi_pass TEXT DEFAULT '',
- mqtt_broker TEXT DEFAULT '',
- mqtt_port INTEGER DEFAULT 1883,
- mqtt_user TEXT DEFAULT '',
- mqtt_pass TEXT DEFAULT '',
- mqtt_client TEXT DEFAULT '',
- mqtt_topic TEXT DEFAULT '',
- mqtt_status TEXT DEFAULT '',
- pin_red INTEGER DEFAULT 4,
- pin_green INTEGER DEFAULT 3,
- pin_yellow INTEGER DEFAULT 2,
- enabled BOOLEAN DEFAULT 1
- );
- `
- _, err = d.conn.ExecContext(context.Background(), deviceQuery)
- if err != nil {
- return err
- }
- return nil
- }
- func (d *DB) GetMQTTConfig() (*MQTTConfig, error) {
- query := "SELECT id, broker, client_id, username, password, topic, enabled FROM mqtt_config WHERE enabled = 1 LIMIT 1"
- row := d.conn.QueryRowContext(context.Background(), query)
- var cfg MQTTConfig
- err := row.Scan(&cfg.ID, &cfg.Broker, &cfg.ClientID, &cfg.Username, &cfg.Password, &cfg.Topic, &cfg.Enabled)
- if err == sql.ErrNoRows {
- logger.Debug("未找到启用的 MQTT 配置")
- return nil, nil
- }
- if err != nil {
- logger.Error("查询 MQTT 配置失败: %v", err)
- return nil, fmt.Errorf("查询配置失败: %w", err)
- }
- logger.Debug("获取到 MQTT 配置: id=%d, broker=%s", cfg.ID, cfg.Broker)
- return &cfg, nil
- }
- func (d *DB) SaveMQTTConfig(cfg *MQTTConfig) error {
- if cfg.ID == 0 {
- query := "INSERT INTO mqtt_config (broker, client_id, username, password, topic, enabled) VALUES (?, ?, ?, ?, ?, ?)"
- _, err := d.conn.ExecContext(context.Background(), query, cfg.Broker, cfg.ClientID, cfg.Username, cfg.Password, cfg.Topic, cfg.Enabled)
- if err != nil {
- logger.Error("插入 MQTT 配置失败: %v", err)
- }
- return err
- }
- query := "UPDATE mqtt_config SET broker = ?, client_id = ?, username = ?, password = ?, topic = ?, enabled = ? WHERE id = ?"
- _, err := d.conn.ExecContext(context.Background(), query, cfg.Broker, cfg.ClientID, cfg.Username, cfg.Password, cfg.Topic, cfg.Enabled, cfg.ID)
- if err != nil {
- logger.Error("更新 MQTT 配置失败: id=%d, %v", cfg.ID, err)
- }
- return err
- }
- func (d *DB) DeleteMQTTConfig(id int) error {
- query := "DELETE FROM mqtt_config WHERE id = ?"
- _, err := d.conn.ExecContext(context.Background(), query, id)
- if err != nil {
- logger.Error("删除 MQTT 配置失败: id=%d, %v", id, err)
- } else {
- logger.Debug("MQTT 配置已删除: id=%d", id)
- }
- return err
- }
- func (d *DB) ListMQTTConfigs() ([]MQTTConfig, error) {
- query := "SELECT id, broker, client_id, username, password, topic, enabled FROM mqtt_config ORDER BY id"
- rows, err := d.conn.QueryContext(context.Background(), query)
- if err != nil {
- logger.Error("查询 MQTT 配置列表失败: %v", err)
- return nil, err
- }
- defer rows.Close()
- var configs []MQTTConfig
- for rows.Next() {
- var cfg MQTTConfig
- if err := rows.Scan(&cfg.ID, &cfg.Broker, &cfg.ClientID, &cfg.Username, &cfg.Password, &cfg.Topic, &cfg.Enabled); err != nil {
- logger.Warn("扫描 MQTT 配置行失败: %v", err)
- continue
- }
- configs = append(configs, cfg)
- }
- logger.Debug("查询到 %d 条 MQTT 配置", len(configs))
- return configs, nil
- }
- func (d *DB) Close() error {
- logger.Debug("数据库连接已关闭")
- if err := d.conn.Close(); err != nil {
- return err
- }
- return d.db.Close()
- }
- func (d *DB) GetBLEConfig() (*BLEConfig, error) {
- query := "SELECT id, device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled FROM ble_config WHERE enabled = 1 LIMIT 1"
- row := d.conn.QueryRowContext(context.Background(), query)
- var cfg BLEConfig
- err := row.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ServiceUUID, &cfg.ModeCharUUID, &cfg.ConfigCharUUID, &cfg.Enabled)
- if err == sql.ErrNoRows {
- logger.Debug("未找到启用的 BLE 配置")
- return nil, nil
- }
- if err != nil {
- logger.Error("查询 BLE 配置失败: %v", err)
- return nil, fmt.Errorf("查询配置失败: %w", err)
- }
- logger.Debug("获取到 BLE 配置: id=%d, device=%s", cfg.ID, cfg.DeviceName)
- return &cfg, nil
- }
- func (d *DB) SaveBLEConfig(cfg *BLEConfig) error {
- if cfg.ID == 0 {
- query := "INSERT INTO ble_config (device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled) VALUES (?, ?, ?, ?, ?)"
- _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ServiceUUID, cfg.ModeCharUUID, cfg.ConfigCharUUID, cfg.Enabled)
- if err != nil {
- logger.Error("插入 BLE 配置失败: %v", err)
- }
- return err
- }
- query := "UPDATE ble_config SET device_name = ?, service_uuid = ?, mode_char_uuid = ?, config_char_uuid = ?, enabled = ? WHERE id = ?"
- _, err := d.conn.ExecContext(context.Background(), query, cfg.DeviceName, cfg.ServiceUUID, cfg.ModeCharUUID, cfg.ConfigCharUUID, cfg.Enabled, cfg.ID)
- if err != nil {
- logger.Error("更新 BLE 配置失败: id=%d, %v", cfg.ID, err)
- }
- return err
- }
- func (d *DB) DeleteBLEConfig(id int) error {
- query := "DELETE FROM ble_config WHERE id = ?"
- _, err := d.conn.ExecContext(context.Background(), query, id)
- if err != nil {
- logger.Error("删除 BLE 配置失败: id=%d, %v", id, err)
- } else {
- logger.Debug("BLE 配置已删除: id=%d", id)
- }
- return err
- }
- func (d *DB) ListBLEConfigs() ([]BLEConfig, error) {
- query := "SELECT id, device_name, service_uuid, mode_char_uuid, config_char_uuid, enabled FROM ble_config ORDER BY id"
- rows, err := d.conn.QueryContext(context.Background(), query)
- if err != nil {
- logger.Error("查询 BLE 配置列表失败: %v", err)
- return nil, err
- }
- defer rows.Close()
- var configs []BLEConfig
- for rows.Next() {
- var cfg BLEConfig
- if err := rows.Scan(&cfg.ID, &cfg.DeviceName, &cfg.ServiceUUID, &cfg.ModeCharUUID, &cfg.ConfigCharUUID, &cfg.Enabled); err != nil {
- logger.Warn("扫描 BLE 配置行失败: %v", err)
- continue
- }
- configs = append(configs, cfg)
- }
- logger.Debug("查询到 %d 条 BLE 配置", len(configs))
- return configs, nil
- }
- func (d *DB) GetDeviceConfig() (*DeviceConfig, error) {
- 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"
- row := d.conn.QueryRowContext(context.Background(), query)
- var cfg DeviceConfig
- 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)
- if err == sql.ErrNoRows {
- logger.Debug("未找到启用的设备配置")
- return nil, nil
- }
- if err != nil {
- logger.Error("查询设备配置失败: %v", err)
- return nil, fmt.Errorf("查询配置失败: %w", err)
- }
- logger.Debug("获取到设备配置: id=%d, device=%s", cfg.ID, cfg.DeviceName)
- return &cfg, nil
- }
- func (d *DB) SaveDeviceConfig(cfg *DeviceConfig) error {
- if cfg.ID == 0 {
- 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 (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)"
- _, 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)
- if err != nil {
- logger.Error("插入设备配置失败: %v", err)
- }
- return err
- }
- 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 = ?"
- _, 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)
- if err != nil {
- logger.Error("更新设备配置失败: id=%d, %v", cfg.ID, err)
- }
- return err
- }
- func (d *DB) DeleteDeviceConfig(id int) error {
- query := "DELETE FROM device_config WHERE id = ?"
- _, err := d.conn.ExecContext(context.Background(), query, id)
- if err != nil {
- logger.Error("删除设备配置失败: id=%d, %v", id, err)
- } else {
- logger.Debug("设备配置已删除: id=%d", id)
- }
- return err
- }
- func (d *DB) ListDeviceConfigs() ([]DeviceConfig, error) {
- 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"
- rows, err := d.conn.QueryContext(context.Background(), query)
- if err != nil {
- logger.Error("查询设备配置列表失败: %v", err)
- return nil, err
- }
- defer rows.Close()
- var configs []DeviceConfig
- for rows.Next() {
- var cfg DeviceConfig
- 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 {
- logger.Warn("扫描设备配置行失败: %v", err)
- continue
- }
- configs = append(configs, cfg)
- }
- logger.Debug("查询到 %d 条设备配置", len(configs))
- return configs, nil
- }
|