1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
|
package flags
import (
"flag"
"io"
"os"
"strings"
"sync"
"testing"
"time"
)
func parseForTest(t *testing.T, args ...string) (Flags, error) {
t.Helper()
oldCommandLine := flag.CommandLine
oldArgs := os.Args
oldSingleton := singleton
oldOnce := once
oldParseErr := parseErr
oldPID := pidFilter.Load()
oldTID := tidFilter.Load()
oldTUIExport := tuiExportEnable.Load()
fs := flag.NewFlagSet("ior-test", flag.ContinueOnError)
fs.SetOutput(io.Discard)
flag.CommandLine = fs
os.Args = append([]string{"ior"}, args...)
singleton = Flags{TUIExportEnable: true}
once = sync.Once{}
parseErr = nil
pidFilter.Store(-1)
tidFilter.Store(-1)
tuiExportEnable.Store(true)
err := parse()
cfg := singleton
t.Cleanup(func() {
flag.CommandLine = oldCommandLine
os.Args = oldArgs
singleton = oldSingleton
once = oldOnce
parseErr = oldParseErr
pidFilter.Store(oldPID)
tidFilter.Store(oldTID)
tuiExportEnable.Store(oldTUIExport)
})
return cfg, err
}
func TestParseLiveFlagsAndInterval(t *testing.T) {
cfg, err := parseForTest(t, "-live", "-live-interval", "200ms", "-pid", "1234")
if err != nil {
t.Fatalf("parse returned error: %v", err)
}
if !cfg.LiveFlamegraph {
t.Fatalf("expected -live to enable live mode")
}
if cfg.LiveInterval != 200*time.Millisecond {
t.Fatalf("live interval = %v, want %v", cfg.LiveInterval, 200*time.Millisecond)
}
if cfg.PidFilter != 1234 {
t.Fatalf("pid filter = %d, want 1234", cfg.PidFilter)
}
if got := int(pidFilter.Load()); got != 1234 {
t.Fatalf("global pid filter = %d, want 1234", got)
}
}
func TestParseLiveDefaults(t *testing.T) {
cfg, err := parseForTest(t)
if err != nil {
t.Fatalf("parse returned error: %v", err)
}
if cfg.LiveFlamegraph {
t.Fatalf("expected live mode disabled by default")
}
if cfg.LiveInterval != 200*time.Millisecond {
t.Fatalf("default live interval = %v, want %v", cfg.LiveInterval, 200*time.Millisecond)
}
}
func TestParseDefaultCollapsedFieldsOrder(t *testing.T) {
cfg, err := parseForTest(t)
if err != nil {
t.Fatalf("parse returned error: %v", err)
}
want := []string{"comm", "path", "tracepoint"}
if len(cfg.CollapsedFields) != len(want) {
t.Fatalf("default collapsed fields len = %d, want %d", len(cfg.CollapsedFields), len(want))
}
for i := range want {
if cfg.CollapsedFields[i] != want[i] {
t.Fatalf("default collapsed fields[%d] = %q, want %q", i, cfg.CollapsedFields[i], want[i])
}
}
}
func TestParseInvalidCollapsedFieldReturnsError(t *testing.T) {
_, err := parseForTest(t, "-fields", "comm,invalid")
if err == nil {
t.Fatalf("expected parse error for invalid collapsed field")
}
if !strings.Contains(err.Error(), "invalid field for collapse: invalid") {
t.Fatalf("unexpected error: %v", err)
}
}
func TestParseInvalidCountFieldReturnsError(t *testing.T) {
_, err := parseForTest(t, "-count", "invalid")
if err == nil {
t.Fatalf("expected parse error for invalid count field")
}
if !strings.Contains(err.Error(), "invalid count field: invalid") {
t.Fatalf("unexpected error: %v", err)
}
}
func TestParseInvalidTracepointRegexReturnsError(t *testing.T) {
_, err := parseForTest(t, "-tps", "[")
if err == nil {
t.Fatalf("expected parse error for invalid tracepoint regex")
}
if !strings.Contains(err.Error(), "unable to compile regex") {
t.Fatalf("unexpected error: %v", err)
}
}
|