cmd/lncli: use .svg for graph rendering add weighted edge thickness
This commit is contained in:
parent
160317c9fa
commit
eb52f7f695
@ -7,6 +7,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"math"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"strconv"
|
"strconv"
|
||||||
@ -771,7 +772,7 @@ var DescribeGraphCommand = cli.Command{
|
|||||||
Flags: []cli.Flag{
|
Flags: []cli.Flag{
|
||||||
cli.BoolFlag{
|
cli.BoolFlag{
|
||||||
Name: "draw",
|
Name: "draw",
|
||||||
Usage: "If true, then an image of graph will be generated and displayed. The generated image is stored within the current directory with a file name of 'graph.png'",
|
Usage: "If true, then an image of graph will be generated and displayed. The generated image is stored within the current directory with a file name of 'graph.svg'",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
Action: describeGraph,
|
Action: describeGraph,
|
||||||
@ -797,6 +798,36 @@ func describeGraph(ctx *cli.Context) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// normalizeFunc is a factory function which returns a function that normalizes
|
||||||
|
// the capacity of of edges within the graph. The value of the returned
|
||||||
|
// function can be used to either plot the capacities, or to use a weight in a
|
||||||
|
// rendering of the graph.
|
||||||
|
func normalizeFunc(edges []*lnrpc.ChannelEdge, scaleFactor float64) func(int64) float64 {
|
||||||
|
var (
|
||||||
|
min float64 = math.MaxInt64
|
||||||
|
max float64
|
||||||
|
)
|
||||||
|
|
||||||
|
for _, edge := range edges {
|
||||||
|
// In order to obtain saner values, we reduce the capacity of a
|
||||||
|
// channel to it's base 2 logarithm.
|
||||||
|
z := math.Log2(float64(edge.Capacity))
|
||||||
|
|
||||||
|
if z < min {
|
||||||
|
min = z
|
||||||
|
}
|
||||||
|
if z > max {
|
||||||
|
max = z
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return func(x int64) float64 {
|
||||||
|
y := math.Log2(float64(x))
|
||||||
|
|
||||||
|
return float64(y-min) / float64(max-min) * scaleFactor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
||||||
// First we'll create a temporary file that we'll write the compiled
|
// First we'll create a temporary file that we'll write the compiled
|
||||||
// string that describes our graph in the dot format to.
|
// string that describes our graph in the dot format to.
|
||||||
@ -808,7 +839,7 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
|
|
||||||
// Next, we'll create (or re-create) the file that the final graph
|
// Next, we'll create (or re-create) the file that the final graph
|
||||||
// image will be written to.
|
// image will be written to.
|
||||||
imageFile, err := os.Create("graph.png")
|
imageFile, err := os.Create("graph.svg")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -818,7 +849,7 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
graphName := "LightningNetwork"
|
graphName := "LightningNetwork"
|
||||||
graphCanvas := gographviz.NewGraph()
|
graphCanvas := gographviz.NewGraph()
|
||||||
graphCanvas.SetName(graphName)
|
graphCanvas.SetName(graphName)
|
||||||
graphCanvas.SetDir(true)
|
graphCanvas.SetDir(false)
|
||||||
|
|
||||||
// For each node within the graph, we'll add a new vertex to the graph.
|
// For each node within the graph, we'll add a new vertex to the graph.
|
||||||
for _, node := range graph.Nodes {
|
for _, node := range graph.Nodes {
|
||||||
@ -833,6 +864,8 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
graphCanvas.AddNode(graphName, nodeID, gographviz.Attrs{})
|
graphCanvas.AddNode(graphName, nodeID, gographviz.Attrs{})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
normalize := normalizeFunc(graph.Edges, 3)
|
||||||
|
|
||||||
// Similarly, for each edge we'll add an edge between the corresponding
|
// Similarly, for each edge we'll add an edge between the corresponding
|
||||||
// nodes added to the graph above.
|
// nodes added to the graph above.
|
||||||
for _, edge := range graph.Edges {
|
for _, edge := range graph.Edges {
|
||||||
@ -843,6 +876,8 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
|
|
||||||
// The weight for our edge will be the total capacity of the
|
// The weight for our edge will be the total capacity of the
|
||||||
// channel, in BTC.
|
// channel, in BTC.
|
||||||
|
// TODO(roasbeef): can also factor in the edges time-lock delta
|
||||||
|
// and fee information
|
||||||
amt := btcutil.Amount(edge.Capacity).ToBTC()
|
amt := btcutil.Amount(edge.Capacity).ToBTC()
|
||||||
edgeWeight := strconv.FormatFloat(amt, 'f', -1, 64)
|
edgeWeight := strconv.FormatFloat(amt, 'f', -1, 64)
|
||||||
|
|
||||||
@ -850,22 +885,18 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
// of it's channel ID.
|
// of it's channel ID.
|
||||||
edgeLabel := strconv.FormatUint(edge.ChannelId, 10)[:10]
|
edgeLabel := strconv.FormatUint(edge.ChannelId, 10)[:10]
|
||||||
|
|
||||||
// We'll only draw an edge from a source to a destination if
|
// We'll also use a normalized version of the channels'
|
||||||
// the source is currently advertising a route in that
|
// capacity in satoshis in order to modulate the "thickness" of
|
||||||
// direction.
|
// the line that creates the edge within the graph.
|
||||||
if edge.Node1Policy.TimeLockDelta != 0 {
|
normalizedCapacity := normalize(edge.Capacity)
|
||||||
graphCanvas.AddEdge(src, dest, true, gographviz.Attrs{
|
edgeThickness := strconv.FormatFloat(normalizedCapacity, 'f', -1, 64)
|
||||||
|
|
||||||
|
graphCanvas.AddEdge(src, dest, false, gographviz.Attrs{
|
||||||
|
"penwidth": edgeThickness,
|
||||||
"weight": edgeWeight,
|
"weight": edgeWeight,
|
||||||
"label": edgeLabel,
|
"label": edgeLabel,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if edge.Node2Policy.TimeLockDelta != 0 {
|
|
||||||
graphCanvas.AddEdge(dest, src, true, gographviz.Attrs{
|
|
||||||
"weight": edgeWeight,
|
|
||||||
"label": edgeLabel,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// With the declarative generation of the graph complete, we now write
|
// With the declarative generation of the graph complete, we now write
|
||||||
// the dot-string description of the graph
|
// the dot-string description of the graph
|
||||||
@ -880,7 +911,7 @@ func drawChannelGraph(graph *lnrpc.ChannelGraph) error {
|
|||||||
// Once our dot file has been written to disk, we can use the dot
|
// Once our dot file has been written to disk, we can use the dot
|
||||||
// command itself to generate the drawn rendering of the graph
|
// command itself to generate the drawn rendering of the graph
|
||||||
// described.
|
// described.
|
||||||
drawCmd := exec.Command("dot", "-T"+"png", "-o"+imageFile.Name(),
|
drawCmd := exec.Command("dot", "-T"+"svg", "-o"+imageFile.Name(),
|
||||||
tempDotFile.Name())
|
tempDotFile.Name())
|
||||||
if err := drawCmd.Run(); err != nil {
|
if err := drawCmd.Run(); err != nil {
|
||||||
return err
|
return err
|
||||||
|
Loading…
Reference in New Issue
Block a user