package systems import ( "database/sql" "errors" "strings" "time" "github.com/henrygd/beszel/internal/entities/smart" "github.com/pocketbase/pocketbase/core" ) type smartFetchState struct { LastAttempt int64 Successful bool } // FetchAndSaveSmartDevices fetches SMART data from the agent and saves it to the database func (sys *System) FetchAndSaveSmartDevices() error { smartData, err := sys.FetchSmartDataFromAgent() if err != nil { sys.recordSmartFetchResult(err, 0) return err } err = sys.saveSmartDevices(smartData) sys.recordSmartFetchResult(err, len(smartData)) return err } // recordSmartFetchResult stores a cooldown entry for the SMART interval and marks // whether the last fetch produced any devices, so failed setup can retry on reconnect. func (sys *System) recordSmartFetchResult(err error, deviceCount int) { if sys.manager == nil { return } interval := sys.smartFetchInterval() success := err == nil && deviceCount > 0 if sys.manager.hub != nil { sys.manager.hub.Logger().Info("SMART fetch result", "system", sys.Id, "success", success, "devices", deviceCount, "interval", interval.String(), "err", err) } sys.manager.smartFetchMap.Set(sys.Id, smartFetchState{LastAttempt: time.Now().UnixMilli(), Successful: success}, interval+time.Minute) } // shouldFetchSmart returns true when there is no active SMART cooldown entry for this system. func (sys *System) shouldFetchSmart() bool { if sys.manager == nil { return true } state, ok := sys.manager.smartFetchMap.GetOk(sys.Id) if !ok { return true } return !time.UnixMilli(state.LastAttempt).Add(sys.smartFetchInterval()).After(time.Now()) } // smartFetchInterval returns the agent-provided SMART interval or the default when unset. func (sys *System) smartFetchInterval() time.Duration { if sys.smartInterval > 0 { return sys.smartInterval } return time.Hour } // saveSmartDevices saves SMART device data to the smart_devices collection func (sys *System) saveSmartDevices(smartData map[string]smart.SmartData) error { if len(smartData) == 0 { return nil } hub := sys.manager.hub collection, err := hub.FindCachedCollectionByNameOrId("smart_devices") if err != nil { return err } for deviceKey, device := range smartData { if err := sys.upsertSmartDeviceRecord(collection, deviceKey, device); err != nil { return err } } return nil } func (sys *System) upsertSmartDeviceRecord(collection *core.Collection, deviceKey string, device smart.SmartData) error { hub := sys.manager.hub recordID := makeStableHashId(sys.Id, deviceKey) record, err := hub.FindRecordById(collection, recordID) if err != nil { if !errors.Is(err, sql.ErrNoRows) { return err } record = core.NewRecord(collection) record.Set("id", recordID) } name := device.DiskName if name == "" { name = deviceKey } powerOnHours, powerCycles := extractPowerMetrics(device.Attributes) record.Set("system", sys.Id) record.Set("name", name) record.Set("model", device.ModelName) record.Set("state", device.SmartStatus) record.Set("capacity", device.Capacity) record.Set("temp", device.Temperature) record.Set("firmware", device.FirmwareVersion) record.Set("serial", device.SerialNumber) record.Set("type", device.DiskType) record.Set("hours", powerOnHours) record.Set("cycles", powerCycles) record.Set("attributes", device.Attributes) return hub.SaveNoValidate(record) } // extractPowerMetrics extracts power on hours and power cycles from SMART attributes func extractPowerMetrics(attributes []*smart.SmartAttribute) (powerOnHours, powerCycles uint64) { for _, attr := range attributes { nameLower := strings.ToLower(attr.Name) if powerOnHours == 0 && (strings.Contains(nameLower, "poweronhours") || strings.Contains(nameLower, "power_on_hours")) { powerOnHours = attr.RawValue } if powerCycles == 0 && ((strings.Contains(nameLower, "power") && strings.Contains(nameLower, "cycle")) || strings.Contains(nameLower, "startstopcycles")) { powerCycles = attr.RawValue } if powerOnHours > 0 && powerCycles > 0 { break } } return }