// Copyright 2023 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 tracev2_test import ( "internal/trace/tracev2" "iter" "regexp" "slices" "strings" "testing" ) var argNameRegexp = regexp.MustCompile(`((?P[A-Za-z]+)_)?(?P[A-Za-z]+)`) func TestSpecs(t *testing.T) { if tracev2.NumEvents <= 0 { t.Fatalf("no trace events?") } if tracev2.MaxExperimentalEvent < tracev2.MaxEvent { t.Fatalf("max experimental event (%d) is < max event (%d)", tracev2.MaxExperimentalEvent, tracev2.MaxEvent) } specs := tracev2.Specs() for ev := range allEvents() { spec := &specs[ev] if spec.Name == "" { t.Errorf("expected event %d to be defined in specs", ev) continue } if spec.IsTimedEvent && spec.Args[0] != "dt" { t.Errorf("%s is a timed event, but its first argument is not 'dt'", spec.Name) } if spec.HasData && spec.Name != "String" && spec.Name != "ExperimentalBatch" { t.Errorf("%s has data, but is not a special kind of event (unsupported, but could be)", spec.Name) } if spec.IsStack && spec.Name != "Stack" { t.Errorf("%s listed as being a stack, but is not the Stack event (unsupported)", spec.Name) } if spec.IsTimedEvent && len(spec.Args) > tracev2.MaxTimedEventArgs { t.Errorf("%s has too many timed event args: have %d, want %d at most", spec.Name, len(spec.Args), tracev2.MaxTimedEventArgs) } if ev.Experimental() && spec.Experiment == tracev2.NoExperiment { t.Errorf("experimental event %s must have an experiment", spec.Name) } // Check arg types. for _, arg := range spec.Args { matches := argNameRegexp.FindStringSubmatch(arg) if len(matches) == 0 { t.Errorf("malformed argument %s for event %s", arg, spec.Name) } } // Check stacks. for _, i := range spec.StackIDs { if !strings.HasSuffix(spec.Args[i], "stack") { t.Errorf("stack argument listed at %d in %s, but argument name %s does not imply stack type", i, spec.Name, spec.Args[i]) } } for i, arg := range spec.Args { if !strings.HasSuffix(spec.Args[i], "stack") { continue } if !slices.Contains(spec.StackIDs, i) { t.Errorf("found stack argument %s in %s at index %d not listed in StackIDs", arg, spec.Name, i) } } // Check strings. for _, i := range spec.StringIDs { if !strings.HasSuffix(spec.Args[i], "string") { t.Errorf("string argument listed at %d in %s, but argument name %s does not imply string type", i, spec.Name, spec.Args[i]) } } for i, arg := range spec.Args { if !strings.HasSuffix(spec.Args[i], "string") { continue } if !slices.Contains(spec.StringIDs, i) { t.Errorf("found string argument %s in %s at index %d not listed in StringIDs", arg, spec.Name, i) } } } } func allEvents() iter.Seq[tracev2.EventType] { return func(yield func(tracev2.EventType) bool) { for ev := tracev2.EvNone + 1; ev < tracev2.NumEvents; ev++ { if !yield(ev) { return } } for ev := tracev2.MaxEvent + 1; ev < tracev2.NumExperimentalEvents; ev++ { if !yield(ev) { return } } } }