| // Copyright 2015 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package trace |
| |
| import ( |
| "bytes" |
| "os" |
| "path/filepath" |
| "strings" |
| "testing" |
| ) |
| |
| func TestCorruptedInputs(t *testing.T) { |
| // These inputs crashed parser previously. |
| tests := []string{ |
| "gotrace\x00\x020", |
| "gotrace\x00Q00\x020", |
| "gotrace\x00T00\x020", |
| "gotrace\x00\xc3\x0200", |
| "go 1.5 trace\x00\x00\x00\x00\x020", |
| "go 1.5 trace\x00\x00\x00\x00Q00\x020", |
| "go 1.5 trace\x00\x00\x00\x00T00\x020", |
| "go 1.5 trace\x00\x00\x00\x00\xc3\x0200", |
| } |
| for _, data := range tests { |
| res, err := Parse(strings.NewReader(data), "") |
| if err == nil || res.Events != nil || res.Stacks != nil { |
| t.Fatalf("no error on input: %q", data) |
| } |
| } |
| } |
| |
| func TestParseCanned(t *testing.T) { |
| files, err := os.ReadDir("./testdata") |
| if err != nil { |
| t.Fatalf("failed to read ./testdata: %v", err) |
| } |
| for _, f := range files { |
| info, err := f.Info() |
| if err != nil { |
| t.Fatal(err) |
| } |
| if testing.Short() && info.Size() > 10000 { |
| continue |
| } |
| name := filepath.Join("./testdata", f.Name()) |
| data, err := os.ReadFile(name) |
| if err != nil { |
| t.Fatal(err) |
| } |
| // Instead of Parse that requires a proper binary name for old traces, |
| // we use 'parse' that omits symbol lookup if an empty string is given. |
| _, _, err = parse(bytes.NewReader(data), "") |
| switch { |
| case strings.HasSuffix(f.Name(), "_good"): |
| if err != nil { |
| t.Errorf("failed to parse good trace %v: %v", f.Name(), err) |
| } |
| case strings.HasSuffix(f.Name(), "_unordered"): |
| if err != ErrTimeOrder { |
| t.Errorf("unordered trace is not detected %v: %v", f.Name(), err) |
| } |
| default: |
| t.Errorf("unknown input file suffix: %v", f.Name()) |
| } |
| } |
| } |
| |
| func TestParseVersion(t *testing.T) { |
| tests := map[string]int{ |
| "go 1.5 trace\x00\x00\x00\x00": 1005, |
| "go 1.7 trace\x00\x00\x00\x00": 1007, |
| "go 1.10 trace\x00\x00\x00": 1010, |
| "go 1.25 trace\x00\x00\x00": 1025, |
| "go 1.234 trace\x00\x00": 1234, |
| "go 1.2345 trace\x00": -1, |
| "go 0.0 trace\x00\x00\x00\x00": -1, |
| "go a.b trace\x00\x00\x00\x00": -1, |
| } |
| for header, ver := range tests { |
| ver1, err := parseHeader([]byte(header)) |
| if ver == -1 { |
| if err == nil { |
| t.Fatalf("no error on input: %q, version %v", header, ver1) |
| } |
| } else { |
| if err != nil { |
| t.Fatalf("failed to parse: %q (%v)", header, err) |
| } |
| if ver != ver1 { |
| t.Fatalf("wrong version: %v, want %v, input: %q", ver1, ver, header) |
| } |
| } |
| } |
| } |
| |
| func TestTimestampOverflow(t *testing.T) { |
| // Test that parser correctly handles large timestamps (long tracing). |
| w := NewWriter() |
| w.Emit(EvBatch, 0, 0) |
| w.Emit(EvFrequency, 1e9) |
| for ts := uint64(1); ts < 1e16; ts *= 2 { |
| w.Emit(EvGoCreate, ts, ts, 0, 0) |
| } |
| if _, err := Parse(w, ""); err != nil { |
| t.Fatalf("failed to parse: %v", err) |
| } |
| } |