mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-11 12:10:05 +00:00
456ac5f9a3
* Done * Fixed lock * Fixed all the tests * Comments * Fixed more tests * Merge branch 'master' into better-head-obj * Fixed more tests * Merge branch 'better-head-obj' of git+ssh://github.com/prysmaticlabs/prysm into better-head-obj * Prestons feedback & fixed test * Nishant's feedback * Participation edge case * Gaz * Merge branch 'master' into better-head-obj * Merge branch 'master' of git+ssh://github.com/prysmaticlabs/prysm into better-head-obj * Raul's feedback * Merge branch 'better-head-obj' of git+ssh://github.com/prysmaticlabs/prysm into better-head-obj
84 lines
2.5 KiB
Go
84 lines
2.5 KiB
Go
package blockchain
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/emicklei/dot"
|
|
)
|
|
|
|
const template = `<html>
|
|
<head>
|
|
<script src="//cdnjs.cloudflare.com/ajax/libs/viz.js/2.1.2/viz.js"></script>
|
|
<script src="//cdnjs.cloudflare.com/ajax/libs/viz.js/2.1.2/full.render.js"></script>
|
|
<body>
|
|
<script type="application/javascript">
|
|
var graph = ` + "`%s`;" + `
|
|
var viz = new Viz();
|
|
viz.renderSVGElement(graph) // reading the graph.
|
|
.then(function(element) {
|
|
document.body.appendChild(element); // appends to document.
|
|
})
|
|
.catch(error => {
|
|
// Create a new Viz instance (@see Caveats page for more info)
|
|
viz = new Viz();
|
|
// Possibly display the error
|
|
console.error(error);
|
|
});
|
|
</script>
|
|
</head>
|
|
</body>
|
|
</html>`
|
|
|
|
// TreeHandler is a handler to serve /tree page in metrics.
|
|
func (s *Service) TreeHandler(w http.ResponseWriter, _ *http.Request) {
|
|
if s.headState() == nil {
|
|
if _, err := w.Write([]byte("Unavailable during initial syncing")); err != nil {
|
|
log.WithError(err).Error("Failed to render p2p info page")
|
|
}
|
|
}
|
|
|
|
nodes := s.forkChoiceStore.Nodes()
|
|
|
|
graph := dot.NewGraph(dot.Directed)
|
|
graph.Attr("rankdir", "RL")
|
|
graph.Attr("labeljust", "l")
|
|
|
|
dotNodes := make([]*dot.Node, len(nodes))
|
|
avgBalance := uint64(averageBalance(s.headState().Balances()))
|
|
|
|
for i := len(nodes) - 1; i >= 0; i-- {
|
|
// Construct label for each node.
|
|
slot := strconv.Itoa(int(nodes[i].Slot))
|
|
weight := strconv.Itoa(int(nodes[i].Weight / 1e9)) // Convert unit Gwei to unit ETH.
|
|
votes := strconv.Itoa(int(nodes[i].Weight / 1e9 / avgBalance))
|
|
bestDescendent := strconv.Itoa(int(nodes[i].BestDescendent))
|
|
index := strconv.Itoa(int(i))
|
|
label := "slot: " + slot + "\n index: " + index + "\n bestDescendent: " + bestDescendent + "\n votes: " + votes + "\n weight: " + weight
|
|
var dotN dot.Node
|
|
if nodes[i].Parent != ^uint64(0) {
|
|
dotN = graph.Node(index).Box().Attr("label", label)
|
|
}
|
|
|
|
if nodes[i].Slot == s.headSlot() &&
|
|
nodes[i].BestDescendent == ^uint64(0) {
|
|
dotN = dotN.Attr("color", "green")
|
|
}
|
|
|
|
dotNodes[i] = &dotN
|
|
}
|
|
|
|
for i := len(nodes) - 1; i >= 0; i-- {
|
|
if nodes[i].Parent != ^uint64(0) && nodes[i].Parent < uint64(len(dotNodes)) {
|
|
graph.Edge(*dotNodes[i], *dotNodes[nodes[i].Parent])
|
|
}
|
|
}
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
w.Header().Set("Content-Type", "text/html")
|
|
if _, err := fmt.Fprintf(w, template, graph.String()); err != nil {
|
|
log.WithError(err).Error("Failed to render p2p info page")
|
|
}
|
|
}
|