// Copyright 2017 Google Inc. All Rights Reserved. // // 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 driver import ( "encoding/json" "html/template" "net/http" "strings" "github.com/google/pprof/internal/graph" "github.com/google/pprof/internal/measurement" "github.com/google/pprof/internal/report" ) type treeNode struct { Name string `json:"n"` FullName string `json:"f"` Cum int64 `json:"v"` CumFormat string `json:"l"` Percent string `json:"p"` Children []*treeNode `json:"c"` } // flamegraph generates a web page containing a flamegraph. func (ui *webInterface) flamegraph(w http.ResponseWriter, req *http.Request) { // Force the call tree so that the graph is a tree. // Also do not trim the tree so that the flame graph contains all functions. rpt, errList := ui.makeReport(w, req, []string{"svg"}, func(cfg *config) { cfg.CallTree = true cfg.Trim = false }) if rpt == nil { return // error already reported } // Generate dot graph. g, config := report.GetDOT(rpt) var nodes []*treeNode nroots := 0 rootValue := int64(0) nodeArr := []string{} nodeMap := map[*graph.Node]*treeNode{} // Make all nodes and the map, collect the roots. for _, n := range g.Nodes { v := n.CumValue() fullName := n.Info.PrintableName() node := &treeNode{ Name: graph.ShortenFunctionName(fullName), FullName: fullName, Cum: v, CumFormat: config.FormatValue(v), Percent: strings.TrimSpace(measurement.Percentage(v, config.Total)), } nodes = append(nodes, node) if len(n.In) == 0 { nodes[nroots], nodes[len(nodes)-1] = nodes[len(nodes)-1], nodes[nroots] nroots++ rootValue += v } nodeMap[n] = node // Get all node names into an array. nodeArr = append(nodeArr, n.Info.Name) } // Populate the child links. for _, n := range g.Nodes { node := nodeMap[n] for child := range n.Out { node.Children = append(node.Children, nodeMap[child]) } } rootNode := &treeNode{ Name: "root", FullName: "root", Cum: rootValue, CumFormat: config.FormatValue(rootValue), Percent: strings.TrimSpace(measurement.Percentage(rootValue, config.Total)), Children: nodes[0:nroots], } // JSON marshalling flame graph b, err := json.Marshal(rootNode) if err != nil { http.Error(w, "error serializing flame graph", http.StatusInternalServerError) ui.options.UI.PrintErr(err) return } ui.render(w, req, "flamegraph", rpt, errList, config.Labels, webArgs{ FlameGraph: template.JS(b), Nodes: nodeArr, }) }