1
2
3
4
5 package pprof
6
7 import (
8 "bytes"
9 "fmt"
10 "internal/profile"
11 "io"
12 "net/http"
13 "net/http/httptest"
14 "runtime"
15 "runtime/pprof"
16 "strings"
17 "sync"
18 "sync/atomic"
19 "testing"
20 "time"
21 )
22
23
24
25 func TestDescriptions(t *testing.T) {
26 for _, p := range pprof.Profiles() {
27 _, ok := profileDescriptions[p.Name()]
28 if ok != true {
29 t.Errorf("%s does not exist in profileDescriptions map\n", p.Name())
30 }
31 }
32 }
33
34 func TestHandlers(t *testing.T) {
35 testCases := []struct {
36 path string
37 handler http.HandlerFunc
38 statusCode int
39 contentType string
40 contentDisposition string
41 resp []byte
42 }{
43 {"/debug/pprof/<script>scripty<script>", Index, http.StatusNotFound, "text/plain; charset=utf-8", "", []byte("Unknown profile\n")},
44 {"/debug/pprof/heap", Index, http.StatusOK, "application/octet-stream", `attachment; filename="heap"`, nil},
45 {"/debug/pprof/heap?debug=1", Index, http.StatusOK, "text/plain; charset=utf-8", "", nil},
46 {"/debug/pprof/cmdline", Cmdline, http.StatusOK, "text/plain; charset=utf-8", "", nil},
47 {"/debug/pprof/profile?seconds=1", Profile, http.StatusOK, "application/octet-stream", `attachment; filename="profile"`, nil},
48 {"/debug/pprof/symbol", Symbol, http.StatusOK, "text/plain; charset=utf-8", "", nil},
49 {"/debug/pprof/trace", Trace, http.StatusOK, "application/octet-stream", `attachment; filename="trace"`, nil},
50 {"/debug/pprof/mutex", Index, http.StatusOK, "application/octet-stream", `attachment; filename="mutex"`, nil},
51 {"/debug/pprof/block?seconds=1", Index, http.StatusOK, "application/octet-stream", `attachment; filename="block-delta"`, nil},
52 {"/debug/pprof/goroutine?seconds=1", Index, http.StatusOK, "application/octet-stream", `attachment; filename="goroutine-delta"`, nil},
53 {"/debug/pprof/", Index, http.StatusOK, "text/html; charset=utf-8", "", []byte("Types of profiles available:")},
54 }
55 for _, tc := range testCases {
56 t.Run(tc.path, func(t *testing.T) {
57 req := httptest.NewRequest("GET", "http://example.com"+tc.path, nil)
58 w := httptest.NewRecorder()
59 tc.handler(w, req)
60
61 resp := w.Result()
62 if got, want := resp.StatusCode, tc.statusCode; got != want {
63 t.Errorf("status code: got %d; want %d", got, want)
64 }
65
66 body, err := io.ReadAll(resp.Body)
67 if err != nil {
68 t.Errorf("when reading response body, expected non-nil err; got %v", err)
69 }
70 if got, want := resp.Header.Get("X-Content-Type-Options"), "nosniff"; got != want {
71 t.Errorf("X-Content-Type-Options: got %q; want %q", got, want)
72 }
73 if got, want := resp.Header.Get("Content-Type"), tc.contentType; got != want {
74 t.Errorf("Content-Type: got %q; want %q", got, want)
75 }
76 if got, want := resp.Header.Get("Content-Disposition"), tc.contentDisposition; got != want {
77 t.Errorf("Content-Disposition: got %q; want %q", got, want)
78 }
79
80 if resp.StatusCode == http.StatusOK {
81 return
82 }
83 if got, want := resp.Header.Get("X-Go-Pprof"), "1"; got != want {
84 t.Errorf("X-Go-Pprof: got %q; want %q", got, want)
85 }
86 if !bytes.Equal(body, tc.resp) {
87 t.Errorf("response: got %q; want %q", body, tc.resp)
88 }
89 })
90 }
91 }
92
93 var Sink uint32
94
95 func mutexHog1(mu1, mu2 *sync.Mutex, start time.Time, dt time.Duration) {
96 atomic.AddUint32(&Sink, 1)
97 for time.Since(start) < dt {
98
99
100
101
102
103 t1 := time.Now()
104 for time.Since(start) < dt && time.Since(t1) < 10*time.Millisecond {
105 mu1.Lock()
106 mu2.Lock()
107 mu1.Unlock()
108 mu2.Unlock()
109 }
110 if runtime.Compiler == "gccgo" {
111 runtime.Gosched()
112 }
113 }
114 }
115
116
117
118
119
120 func mutexHog2(mu1, mu2 *sync.Mutex, start time.Time, dt time.Duration) {
121 atomic.AddUint32(&Sink, 2)
122 for time.Since(start) < dt {
123
124 t1 := time.Now()
125 for time.Since(start) < dt && time.Since(t1) < 10*time.Millisecond {
126 mu1.Lock()
127 mu2.Lock()
128 mu1.Unlock()
129 mu2.Unlock()
130 }
131 if runtime.Compiler == "gccgo" {
132 runtime.Gosched()
133 }
134 }
135 }
136
137
138
139 func mutexHog(duration time.Duration, hogger func(mu1, mu2 *sync.Mutex, start time.Time, dt time.Duration)) {
140 start := time.Now()
141 mu1 := new(sync.Mutex)
142 mu2 := new(sync.Mutex)
143 var wg sync.WaitGroup
144 wg.Add(10)
145 for i := 0; i < 10; i++ {
146 go func() {
147 defer wg.Done()
148 hogger(mu1, mu2, start, duration)
149 }()
150 }
151 wg.Wait()
152 }
153
154 func TestDeltaProfile(t *testing.T) {
155 rate := runtime.SetMutexProfileFraction(1)
156 defer func() {
157 runtime.SetMutexProfileFraction(rate)
158 }()
159
160
161
162 mutexHog(20*time.Millisecond, mutexHog1)
163
164
165
166
167
168 p, err := query("/debug/pprof/mutex")
169 if err != nil {
170 t.Skipf("mutex profile is unsupported: %v", err)
171 }
172
173 if !seen(p, "mutexHog1") {
174 t.Skipf("mutex profile is not working: %v", p)
175 }
176
177
178 done := make(chan bool)
179 go func() {
180 for {
181 mutexHog(20*time.Millisecond, mutexHog2)
182 select {
183 case <-done:
184 done <- true
185 return
186 default:
187 time.Sleep(10 * time.Millisecond)
188 }
189 }
190 }()
191 defer func() {
192 done <- true
193 <-done
194 }()
195
196 for _, d := range []int{1, 4, 16, 32} {
197 endpoint := fmt.Sprintf("/debug/pprof/mutex?seconds=%d", d)
198 p, err := query(endpoint)
199 if err != nil {
200 t.Fatalf("failed to query %q: %v", endpoint, err)
201 }
202 if !seen(p, "mutexHog1") && seen(p, "mutexHog2") && p.DurationNanos > 0 {
203 break
204 }
205 if d == 32 {
206 t.Errorf("want mutexHog2 but no mutexHog1 in the profile, and non-zero p.DurationNanos, got %v", p)
207 }
208 }
209 p, err = query("/debug/pprof/mutex")
210 if err != nil {
211 t.Fatalf("failed to query mutex profile: %v", err)
212 }
213 if !seen(p, "mutexHog1") || !seen(p, "mutexHog2") {
214 t.Errorf("want both mutexHog1 and mutexHog2 in the profile, got %v", p)
215 }
216 }
217
218 var srv = httptest.NewServer(nil)
219
220 func query(endpoint string) (*profile.Profile, error) {
221 url := srv.URL + endpoint
222 r, err := http.Get(url)
223 if err != nil {
224 return nil, fmt.Errorf("failed to fetch %q: %v", url, err)
225 }
226 if r.StatusCode != http.StatusOK {
227 return nil, fmt.Errorf("failed to fetch %q: %v", url, r.Status)
228 }
229
230 b, err := io.ReadAll(r.Body)
231 r.Body.Close()
232 if err != nil {
233 return nil, fmt.Errorf("failed to read and parse the result from %q: %v", url, err)
234 }
235 return profile.Parse(bytes.NewBuffer(b))
236 }
237
238
239
240 func seen(p *profile.Profile, fname string) bool {
241 locIDs := map[*profile.Location]bool{}
242 for _, loc := range p.Location {
243 for _, l := range loc.Line {
244 if strings.Contains(l.Function.Name, fname) {
245 locIDs[loc] = true
246 break
247 }
248 }
249 }
250 for _, sample := range p.Sample {
251 for _, loc := range sample.Location {
252 if locIDs[loc] {
253 return true
254 }
255 }
256 }
257 return false
258 }
259
View as plain text