// Copyright 2013 The Prometheus Authors // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package model import ( "encoding/json" "strconv" "testing" "time" ) func TestComparators(t *testing.T) { t1a := TimeFromUnix(0) t1b := TimeFromUnix(0) t2 := TimeFromUnix(2*second - 1) if !t1a.Equal(t1b) { t.Fatalf("Expected %s to be equal to %s", t1a, t1b) } if t1a.Equal(t2) { t.Fatalf("Expected %s to not be equal to %s", t1a, t2) } if !t1a.Before(t2) { t.Fatalf("Expected %s to be before %s", t1a, t2) } if t1a.Before(t1b) { t.Fatalf("Expected %s to not be before %s", t1a, t1b) } if !t2.After(t1a) { t.Fatalf("Expected %s to be after %s", t2, t1a) } if t1b.After(t1a) { t.Fatalf("Expected %s to not be after %s", t1b, t1a) } } func TestTimeConversions(t *testing.T) { unixSecs := int64(1136239445) unixNsecs := int64(123456789) unixNano := unixSecs*1e9 + unixNsecs t1 := time.Unix(unixSecs, unixNsecs-unixNsecs%nanosPerTick) t2 := time.Unix(unixSecs, unixNsecs) ts := TimeFromUnixNano(unixNano) if !ts.Time().Equal(t1) { t.Fatalf("Expected %s, got %s", t1, ts.Time()) } // Test available precision. ts = TimeFromUnixNano(t2.UnixNano()) if !ts.Time().Equal(t1) { t.Fatalf("Expected %s, got %s", t1, ts.Time()) } if ts.UnixNano() != unixNano-unixNano%nanosPerTick { t.Fatalf("Expected %d, got %d", unixNano, ts.UnixNano()) } } func TestDuration(t *testing.T) { duration := time.Second + time.Minute + time.Hour goTime := time.Unix(1136239445, 0) ts := TimeFromUnix(goTime.Unix()) if !goTime.Add(duration).Equal(ts.Add(duration).Time()) { t.Fatalf("Expected %s to be equal to %s", goTime.Add(duration), ts.Add(duration)) } earlier := ts.Add(-duration) delta := ts.Sub(earlier) if delta != duration { t.Fatalf("Expected %s to be equal to %s", delta, duration) } } func TestParseDuration(t *testing.T) { var cases = []struct { in string out time.Duration expectedString string }{ { in: "0", out: 0, expectedString: "0s", }, { in: "0w", out: 0, expectedString: "0s", }, { in: "0s", out: 0, }, { in: "324ms", out: 324 * time.Millisecond, }, { in: "3s", out: 3 * time.Second, }, { in: "5m", out: 5 * time.Minute, }, { in: "1h", out: time.Hour, }, { in: "4d", out: 4 * 24 * time.Hour, }, { in: "4d1h", out: 4*24*time.Hour + time.Hour, }, { in: "14d", out: 14 * 24 * time.Hour, expectedString: "2w", }, { in: "3w", out: 3 * 7 * 24 * time.Hour, }, { in: "3w2d1h", out: 3*7*24*time.Hour + 2*24*time.Hour + time.Hour, expectedString: "23d1h", }, { in: "10y", out: 10 * 365 * 24 * time.Hour, }, } for _, c := range cases { d, err := ParseDuration(c.in) if err != nil { t.Errorf("Unexpected error on input %q", c.in) } if time.Duration(d) != c.out { t.Errorf("Expected %v but got %v", c.out, d) } expectedString := c.expectedString if c.expectedString == "" { expectedString = c.in } if d.String() != expectedString { t.Errorf("Expected duration string %q but got %q", c.in, d.String()) } } } func TestDuration_UnmarshalText(t *testing.T) { var cases = []struct { in string out time.Duration expectedString string }{ { in: "0", out: 0, expectedString: "0s", }, { in: "0w", out: 0, expectedString: "0s", }, { in: "0s", out: 0, }, { in: "324ms", out: 324 * time.Millisecond, }, { in: "3s", out: 3 * time.Second, }, { in: "5m", out: 5 * time.Minute, }, { in: "1h", out: time.Hour, }, { in: "4d", out: 4 * 24 * time.Hour, }, { in: "4d1h", out: 4*24*time.Hour + time.Hour, }, { in: "14d", out: 14 * 24 * time.Hour, expectedString: "2w", }, { in: "3w", out: 3 * 7 * 24 * time.Hour, }, { in: "3w2d1h", out: 3*7*24*time.Hour + 2*24*time.Hour + time.Hour, expectedString: "23d1h", }, { in: "10y", out: 10 * 365 * 24 * time.Hour, }, } for _, c := range cases { var d Duration err := d.UnmarshalText([]byte(c.in)) if err != nil { t.Errorf("Unexpected error on input %q", c.in) } if time.Duration(d) != c.out { t.Errorf("Expected %v but got %v", c.out, d) } expectedString := c.expectedString if c.expectedString == "" { expectedString = c.in } text, _ := d.MarshalText() // MarshalText returns hardcoded nil if string(text) != expectedString { t.Errorf("Expected duration string %q but got %q", c.in, d.String()) } } } func TestDuration_UnmarshalJSON(t *testing.T) { var cases = []struct { in string out time.Duration expectedString string }{ { in: `"0"`, out: 0, expectedString: `"0s"`, }, { in: `"0w"`, out: 0, expectedString: `"0s"`, }, { in: `"0s"`, out: 0, }, { in: `"324ms"`, out: 324 * time.Millisecond, }, { in: `"3s"`, out: 3 * time.Second, }, { in: `"5m"`, out: 5 * time.Minute, }, { in: `"1h"`, out: time.Hour, }, { in: `"4d"`, out: 4 * 24 * time.Hour, }, { in: `"4d1h"`, out: 4*24*time.Hour + time.Hour, }, { in: `"14d"`, out: 14 * 24 * time.Hour, expectedString: `"2w"`, }, { in: `"3w"`, out: 3 * 7 * 24 * time.Hour, }, { in: `"3w2d1h"`, out: 3*7*24*time.Hour + 2*24*time.Hour + time.Hour, expectedString: `"23d1h"`, }, { in: `"10y"`, out: 10 * 365 * 24 * time.Hour, }, { in: `"289y"`, out: 289 * 365 * 24 * time.Hour, }, } for _, c := range cases { var d Duration err := json.Unmarshal([]byte(c.in), &d) if err != nil { t.Errorf("Unexpected error on input %q", c.in) } if time.Duration(d) != c.out { t.Errorf("Expected %v but got %v", c.out, d) } expectedString := c.expectedString if c.expectedString == "" { expectedString = c.in } bytes, err := json.Marshal(d) if err != nil { t.Errorf("Unexpected error on marshal of %v: %s", d, err) } if string(bytes) != expectedString { t.Errorf("Expected duration string %q but got %q", c.in, d.String()) } } } func TestParseBadDuration(t *testing.T) { var cases = []string{ "1", "1y1m1d", "-1w", "1.5d", "d", "294y", "200y10400w", "107675d", "2584200h", "", } for _, c := range cases { _, err := ParseDuration(c) if err == nil { t.Errorf("Expected error on input %s", c) } } } func TestTimeJSON(t *testing.T) { tests := []struct { in Time out string }{ {Time(1), `0.001`}, {Time(-1), `-0.001`}, } for i, test := range tests { t.Run(strconv.Itoa(i), func(t *testing.T) { b, err := test.in.MarshalJSON() if err != nil { t.Fatalf("Error marshaling time: %v", err) } if string(b) != test.out { t.Errorf("Mismatch in marshal expected=%s actual=%s", test.out, b) } var tm Time if err := tm.UnmarshalJSON(b); err != nil { t.Fatalf("Error Unmarshaling time: %v", err) } if !test.in.Equal(tm) { t.Fatalf("Mismatch after Unmarshal expected=%v actual=%v", test.in, tm) } }) } } func BenchmarkParseDuration(b *testing.B) { const data = "30s" for i := 0; i < b.N; i++ { _, err := ParseDuration(data) if err != nil { b.Fatal(err) } } }