package measurements_test import ( "encoding/json" "github.com/analogj/scrutiny/webapp/backend/pkg" "github.com/analogj/scrutiny/webapp/backend/pkg/models/collector" "github.com/analogj/scrutiny/webapp/backend/pkg/models/measurements" "github.com/stretchr/testify/require" "io/ioutil" "os" "testing" "time" ) func TestSmart_Flatten(t *testing.T) { //setup timeNow := time.Now() smart := measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: pkg.DeviceProtocolAta, Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Attributes: nil, Status: 0, } //test tags, fields := smart.Flatten() //assert require.Equal(t, map[string]string{"device_protocol": "ATA", "device_wwn": "test-wwn"}, tags) require.Equal(t, map[string]interface{}{"power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50)}, fields) } func TestSmart_Flatten_ATA(t *testing.T) { //setup timeNow := time.Now() smart := measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: pkg.DeviceProtocolAta, Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Status: 0, Attributes: map[string]measurements.SmartAttribute{ "1": &measurements.SmartAtaAttribute{ AttributeId: 1, Value: 100, Threshold: 1, Worst: 100, RawValue: 0, RawString: "0", WhenFailed: "", }, "2": &measurements.SmartAtaAttribute{ AttributeId: 2, Value: 135, Threshold: 54, Worst: 135, RawValue: 108, RawString: "108", WhenFailed: "", }, }, } //test tags, fields := smart.Flatten() //assert require.Equal(t, map[string]string{"device_protocol": "ATA", "device_wwn": "test-wwn"}, tags) require.Equal(t, map[string]interface{}{ "attr.1.attribute_id": "1", "attr.1.failure_rate": float64(0), "attr.1.raw_string": "0", "attr.1.raw_value": int64(0), "attr.1.status": int64(0), "attr.1.status_reason": "", "attr.1.thresh": int64(1), "attr.1.transformed_value": int64(0), "attr.1.value": int64(100), "attr.1.when_failed": "", "attr.1.worst": int64(100), "attr.2.attribute_id": "2", "attr.2.failure_rate": float64(0), "attr.2.raw_string": "108", "attr.2.raw_value": int64(108), "attr.2.status": int64(0), "attr.2.status_reason": "", "attr.2.thresh": int64(54), "attr.2.transformed_value": int64(0), "attr.2.value": int64(135), "attr.2.when_failed": "", "attr.2.worst": int64(135), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50), }, fields) } func TestSmart_Flatten_SCSI(t *testing.T) { //setup timeNow := time.Now() smart := measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: pkg.DeviceProtocolScsi, Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Status: 0, Attributes: map[string]measurements.SmartAttribute{ "read_errors_corrected_by_eccfast": &measurements.SmartScsiAttribute{ AttributeId: "read_errors_corrected_by_eccfast", Value: int64(300357663), }, }, } //test tags, fields := smart.Flatten() //assert require.Equal(t, map[string]string{"device_protocol": "SCSI", "device_wwn": "test-wwn"}, tags) require.Equal(t, map[string]interface{}{ "attr.read_errors_corrected_by_eccfast.attribute_id": "read_errors_corrected_by_eccfast", "attr.read_errors_corrected_by_eccfast.failure_rate": float64(0), "attr.read_errors_corrected_by_eccfast.status": int64(0), "attr.read_errors_corrected_by_eccfast.status_reason": "", "attr.read_errors_corrected_by_eccfast.thresh": int64(0), "attr.read_errors_corrected_by_eccfast.transformed_value": int64(0), "attr.read_errors_corrected_by_eccfast.value": int64(300357663), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50)}, fields) } func TestSmart_Flatten_NVMe(t *testing.T) { //setup timeNow := time.Now() smart := measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: pkg.DeviceProtocolNvme, Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Status: 0, Attributes: map[string]measurements.SmartAttribute{ "available_spare": &measurements.SmartNvmeAttribute{ AttributeId: "available_spare", Value: int64(100), }, }, } //test tags, fields := smart.Flatten() //assert require.Equal(t, map[string]string{"device_protocol": "NVMe", "device_wwn": "test-wwn"}, tags) require.Equal(t, map[string]interface{}{ "attr.available_spare.attribute_id": "available_spare", "attr.available_spare.failure_rate": float64(0), "attr.available_spare.status": int64(0), "attr.available_spare.status_reason": "", "attr.available_spare.thresh": int64(0), "attr.available_spare.transformed_value": int64(0), "attr.available_spare.value": int64(100), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50)}, fields) } func TestNewSmartFromInfluxDB_ATA(t *testing.T) { //setup timeNow := time.Now() attrs := map[string]interface{}{ "_time": timeNow, "device_wwn": "test-wwn", "device_protocol": pkg.DeviceProtocolAta, "attr.1.attribute_id": "1", "attr.1.failure_rate": float64(0), "attr.1.raw_string": "108", "attr.1.raw_value": int64(108), "attr.1.status": int64(0), "attr.1.status_reason": "", "attr.1.thresh": int64(54), "attr.1.transformed_value": int64(0), "attr.1.value": int64(135), "attr.1.when_failed": "", "attr.1.worst": int64(135), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50), } //test smart, err := measurements.NewSmartFromInfluxDB(attrs) //assert require.NoError(t, err) require.Equal(t, &measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: "ATA", Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Attributes: map[string]measurements.SmartAttribute{ "1": &measurements.SmartAtaAttribute{ AttributeId: 1, Value: 135, Threshold: 54, Worst: 135, RawValue: 108, RawString: "108", WhenFailed: "", }, }, Status: 0}, smart) } func TestNewSmartFromInfluxDB_NVMe(t *testing.T) { //setup timeNow := time.Now() attrs := map[string]interface{}{ "_time": timeNow, "device_wwn": "test-wwn", "device_protocol": pkg.DeviceProtocolNvme, "attr.available_spare.attribute_id": "available_spare", "attr.available_spare.failure_rate": float64(0), "attr.available_spare.status": int64(0), "attr.available_spare.status_reason": "", "attr.available_spare.thresh": int64(0), "attr.available_spare.transformed_value": int64(0), "attr.available_spare.value": int64(100), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50), } //test smart, err := measurements.NewSmartFromInfluxDB(attrs) //assert require.NoError(t, err) require.Equal(t, &measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: "NVMe", Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Attributes: map[string]measurements.SmartAttribute{ "available_spare": &measurements.SmartNvmeAttribute{ AttributeId: "available_spare", Value: int64(100), }, }, Status: 0}, smart) } func TestNewSmartFromInfluxDB_SCSI(t *testing.T) { //setup timeNow := time.Now() attrs := map[string]interface{}{ "_time": timeNow, "device_wwn": "test-wwn", "device_protocol": pkg.DeviceProtocolScsi, "attr.read_errors_corrected_by_eccfast.attribute_id": "read_errors_corrected_by_eccfast", "attr.read_errors_corrected_by_eccfast.failure_rate": float64(0), "attr.read_errors_corrected_by_eccfast.status": int64(0), "attr.read_errors_corrected_by_eccfast.status_reason": "", "attr.read_errors_corrected_by_eccfast.thresh": int64(0), "attr.read_errors_corrected_by_eccfast.transformed_value": int64(0), "attr.read_errors_corrected_by_eccfast.value": int64(300357663), "power_cycle_count": int64(10), "power_on_hours": int64(10), "temp": int64(50), } //test smart, err := measurements.NewSmartFromInfluxDB(attrs) //assert require.NoError(t, err) require.Equal(t, &measurements.Smart{ Date: timeNow, DeviceWWN: "test-wwn", DeviceProtocol: "SCSI", Temp: 50, PowerOnHours: 10, PowerCycleCount: 10, Attributes: map[string]measurements.SmartAttribute{ "read_errors_corrected_by_eccfast": &measurements.SmartScsiAttribute{ AttributeId: "read_errors_corrected_by_eccfast", Value: int64(300357663), }, }, Status: 0}, smart) } func TestFromCollectorSmartInfo(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-ata.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusPassed, smartMdl.Status) require.Equal(t, 18, len(smartMdl.Attributes)) //check that temperature was correctly parsed require.Equal(t, int64(163210330144), smartMdl.Attributes["194"].(*measurements.SmartAtaAttribute).RawValue) require.Equal(t, int64(32), smartMdl.Attributes["194"].(*measurements.SmartAtaAttribute).TransformedValue) //ensure that Scrutiny warning for a non critical attribute does not set device status to failed. require.Equal(t, pkg.AttributeStatusWarningScrutiny, smartMdl.Attributes["3"].GetStatus()) } func TestFromCollectorSmartInfo_Fail_Smart(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-fail.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusFailedSmart, smartMdl.Status) require.Equal(t, 0, len(smartMdl.Attributes)) } func TestFromCollectorSmartInfo_Fail_ScrutinySmart(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-fail2.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusFailedScrutiny|pkg.DeviceStatusFailedSmart, smartMdl.Status) require.Equal(t, 17, len(smartMdl.Attributes)) } func TestFromCollectorSmartInfo_Fail_ScrutinyNonCriticalFailed(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-ata-failed-scrutiny.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusFailedScrutiny, smartMdl.Status) require.Equal(t, pkg.AttributeStatusFailedScrutiny, smartMdl.Attributes["199"].GetStatus(), "scrutiny should detect that %d failed (status: %d, %s)", smartMdl.Attributes["199"].(*measurements.SmartAtaAttribute).AttributeId, smartMdl.Attributes["199"].GetStatus(), smartMdl.Attributes["199"].(*measurements.SmartAtaAttribute).StatusReason, ) require.Equal(t, 14, len(smartMdl.Attributes)) } //TODO: Scrutiny Warn //TODO: Smart + Scrutiny Warn func TestFromCollectorSmartInfo_NVMe_Fail_Scrutiny(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-nvme-failed.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusFailedScrutiny, smartMdl.Status) require.Equal(t, pkg.AttributeStatusFailedScrutiny, smartMdl.Attributes["media_errors"].GetStatus(), "scrutiny should detect that %s failed (status: %d, %s)", smartMdl.Attributes["media_errors"].(*measurements.SmartNvmeAttribute).AttributeId, smartMdl.Attributes["media_errors"].GetStatus(), smartMdl.Attributes["media_errors"].(*measurements.SmartNvmeAttribute).StatusReason, ) require.Equal(t, 16, len(smartMdl.Attributes)) } func TestFromCollectorSmartInfo_Nvme(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-nvme.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusPassed, smartMdl.Status) require.Equal(t, 16, len(smartMdl.Attributes)) require.Equal(t, int64(111303174), smartMdl.Attributes["host_reads"].(*measurements.SmartNvmeAttribute).Value) require.Equal(t, int64(83170961), smartMdl.Attributes["host_writes"].(*measurements.SmartNvmeAttribute).Value) } func TestFromCollectorSmartInfo_Scsi(t *testing.T) { //setup smartDataFile, err := os.Open("../testdata/smart-scsi.json") require.NoError(t, err) defer smartDataFile.Close() var smartJson collector.SmartInfo smartDataBytes, err := ioutil.ReadAll(smartDataFile) require.NoError(t, err) err = json.Unmarshal(smartDataBytes, &smartJson) require.NoError(t, err) //test smartMdl := measurements.Smart{} err = smartMdl.FromCollectorSmartInfo("WWN-test", smartJson) //assert require.NoError(t, err) require.Equal(t, "WWN-test", smartMdl.DeviceWWN) require.Equal(t, pkg.DeviceStatusPassed, smartMdl.Status) require.Equal(t, 13, len(smartMdl.Attributes)) require.Equal(t, int64(56), smartMdl.Attributes["scsi_grown_defect_list"].(*measurements.SmartScsiAttribute).Value) require.Equal(t, int64(300357663), smartMdl.Attributes["read_errors_corrected_by_eccfast"].(*measurements.SmartScsiAttribute).Value) //total_errors_corrected }