Merge pull request #467 from bcongdon/feature/customize-alias-and-color

cmd/lncli+lnd: Add alias and color customization
This commit is contained in:
Olaoluwa Osuntokun 2018-01-24 16:48:36 -08:00 committed by GitHub
commit 4f91b66c51
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 78 additions and 11 deletions

@ -57,6 +57,9 @@ const (
defaultLitecoinBaseFeeMSat = 1000 defaultLitecoinBaseFeeMSat = 1000
defaultLitecoinFeeRate = 1 defaultLitecoinFeeRate = 1
defaultLitecoinTimeLockDelta = 576 defaultLitecoinTimeLockDelta = 576
defaultAlias = ""
defaultColor = "#3399FF"
) )
var ( var (
@ -174,6 +177,9 @@ type config struct {
NoEncryptWallet bool `long:"noencryptwallet" description:"If set, wallet will be encrypted using the default passphrase."` NoEncryptWallet bool `long:"noencryptwallet" description:"If set, wallet will be encrypted using the default passphrase."`
TrickleDelay int `long:"trickledelay" description:"Time in milliseconds between each release of announcements to the network"` TrickleDelay int `long:"trickledelay" description:"Time in milliseconds between each release of announcements to the network"`
Alias string `long:"alias" description:"The node alias. Used as a moniker by peers and intelligence services"`
Color string `long:"color" description:"The color of the node in hex format (i.e. '#3399FF'). Used to customize node appearance in intelligence services"`
} }
// loadConfig initializes and parses the config using a config file and command // loadConfig initializes and parses the config using a config file and command
@ -226,6 +232,8 @@ func loadConfig() (*config, error) {
Allocation: 0.6, Allocation: 0.6,
}, },
TrickleDelay: defaultTrickleDelay, TrickleDelay: defaultTrickleDelay,
Alias: defaultAlias,
Color: defaultColor,
} }
// Pre-parse the command line options to pick up an alternative config // Pre-parse the command line options to pick up an alternative config

@ -40,7 +40,8 @@ func NewNodeAlias(s string) (NodeAlias, error) {
// String returns a utf8 string representation of the alias bytes. // String returns a utf8 string representation of the alias bytes.
func (n NodeAlias) String() string { func (n NodeAlias) String() string {
return string(n[:]) // Trim trailing zero-bytes for presentation
return string(bytes.Trim(n[:], "\x00"))
} }
// NodeAnnouncement message is used to announce the presence of a Lightning // NodeAnnouncement message is used to announce the presence of a Lightning

@ -1060,6 +1060,7 @@ func (r *rpcServer) GetInfo(ctx context.Context,
Testnet: activeNetParams.Params == &chaincfg.TestNet3Params, Testnet: activeNetParams.Params == &chaincfg.TestNet3Params,
Chains: activeChains, Chains: activeChains,
Uris: uris, Uris: uris,
Alias: nodeAnn.Alias.String(),
}, nil }, nil
} }

@ -234,14 +234,19 @@ func newServer(listenAddrs []string, chanDB *channeldb.DB, cc *chainControl,
chanGraph := chanDB.ChannelGraph() chanGraph := chanDB.ChannelGraph()
defaultColor := color.RGBA{ // #3399FF // Parse node color from configuration.
R: 51, color, err := parseHexColor(cfg.Color)
G: 153, if err != nil {
B: 255, srvrLog.Errorf("unable to parse color: %v\n", err)
return nil, err
} }
// TODO(roasbeef): make alias configurable // If no alias is provided, default to first 10 characters of public key
alias, err := lnwire.NewNodeAlias(hex.EncodeToString(serializedPubKey[:10])) alias := cfg.Alias
if alias == "" {
alias = hex.EncodeToString(serializedPubKey[:10])
}
nodeAlias, err := lnwire.NewNodeAlias(alias)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -250,9 +255,9 @@ func newServer(listenAddrs []string, chanDB *channeldb.DB, cc *chainControl,
LastUpdate: time.Now(), LastUpdate: time.Now(),
Addresses: selfAddrs, Addresses: selfAddrs,
PubKey: privKey.PubKey(), PubKey: privKey.PubKey(),
Alias: alias.String(), Alias: nodeAlias.String(),
Features: s.globalFeatures, Features: s.globalFeatures,
Color: defaultColor, Color: color,
} }
// If our information has changed since our last boot, then we'll // If our information has changed since our last boot, then we'll
@ -264,9 +269,9 @@ func newServer(listenAddrs []string, chanDB *channeldb.DB, cc *chainControl,
Timestamp: uint32(selfNode.LastUpdate.Unix()), Timestamp: uint32(selfNode.LastUpdate.Unix()),
Addresses: selfNode.Addresses, Addresses: selfNode.Addresses,
NodeID: selfNode.PubKey, NodeID: selfNode.PubKey,
Alias: alias, Alias: nodeAlias,
Features: selfNode.Features.RawFeatureVector, Features: selfNode.Features.RawFeatureVector,
RGBColor: defaultColor, RGBColor: color,
} }
selfNode.AuthSig, err = discovery.SignAnnouncement(s.nodeSigner, selfNode.AuthSig, err = discovery.SignAnnouncement(s.nodeSigner,
s.identityPriv.PubKey(), nodeAnn, s.identityPriv.PubKey(), nodeAnn,
@ -1780,3 +1785,21 @@ func (s *server) Peers() []*peer {
return peers return peers
} }
// parseHexColor takes a hex string representation of a color in the
// form "#RRGGBB", parses the hex color values, and returns a color.RGBA
// struct of the same color.
func parseHexColor(colorStr string) (color.RGBA, error) {
if len(colorStr) != 7 || colorStr[0] != '#' {
return color.RGBA{}, errors.New("Color must be in format #RRGGBB")
}
// Decode the hex color string to bytes.
// The resulting byte array is in the form [R, G, B].
colorBytes, err := hex.DecodeString(colorStr[1:])
if err != nil {
return color.RGBA{}, err
}
return color.RGBA{R: colorBytes[0], G: colorBytes[1], B: colorBytes[2]}, nil
}

34
server_test.go Normal file

@ -0,0 +1,34 @@
package main
import "testing"
func TestParseHexColor(t *testing.T) {
empty := ""
color, err := parseHexColor(empty)
if err == nil {
t.Fatalf("Empty color string should return error, but did not")
}
tooLong := "#1234567"
color, err = parseHexColor(tooLong)
if err == nil {
t.Fatalf("Invalid color string %s should return error, but did not",
tooLong)
}
invalidFormat := "$123456"
color, err = parseHexColor(invalidFormat)
if err == nil {
t.Fatalf("Invalid color string %s should return error, but did not",
invalidFormat)
}
valid := "#C0FfeE"
color, err = parseHexColor(valid)
if err != nil {
t.Fatalf("Color %s valid to parse: %s", valid, err)
}
if color.R != 0xc0 || color.G != 0xff || color.B != 0xee {
t.Fatalf("Color %s incorrectly parsed as %v", valid, color)
}
}