watchtower/blob/type: introduces blob type flags
Adds flags for reward outputs and commitment outputs. The fixed-size encoding for commitment outputs is treated as a flag, so that the blob format can be modified, extended, or replaced in future iterations.
This commit is contained in:
parent
9f7d2f6cbc
commit
bc09c6dbae
134
watchtower/blob/type.go
Normal file
134
watchtower/blob/type.go
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
package blob
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Flag represents a specify option that can be present in a Type.
|
||||||
|
type Flag uint16
|
||||||
|
|
||||||
|
const (
|
||||||
|
// FlagReward signals that the justice transaction should contain an
|
||||||
|
// additional output for itself. Signatures sent by the client should
|
||||||
|
// include the reward script negotiated during session creation. Without
|
||||||
|
// the flag, there is only one output sweeping clients funds back to
|
||||||
|
// them solely.
|
||||||
|
FlagReward Flag = 1 << iota
|
||||||
|
|
||||||
|
// FlagCommitOutputs signals that the blob contains the information
|
||||||
|
// required to sweep commitment outputs.
|
||||||
|
FlagCommitOutputs
|
||||||
|
)
|
||||||
|
|
||||||
|
// Type returns a Type consisting solely of this flag enabled.
|
||||||
|
func (f Flag) Type() Type {
|
||||||
|
return Type(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns the name of the flag.
|
||||||
|
func (f Flag) String() string {
|
||||||
|
switch f {
|
||||||
|
case FlagReward:
|
||||||
|
return "FlagReward"
|
||||||
|
case FlagCommitOutputs:
|
||||||
|
return "FlagCommitOutputs"
|
||||||
|
default:
|
||||||
|
return "FlagUnknown"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type is a bit vector composed of Flags that govern various aspects of
|
||||||
|
// reconstructing the justice transaction from an encrypted blob. The flags can
|
||||||
|
// be used to signal behaviors such as which inputs are being swept, which
|
||||||
|
// outputs should be added to the justice transaction, or modify serialization
|
||||||
|
// of the blob itself.
|
||||||
|
type Type uint16
|
||||||
|
|
||||||
|
// TypeDefault sweeps only commitment outputs to a sweep address controlled by
|
||||||
|
// the user, and does not give the tower a reward.
|
||||||
|
const TypeDefault = Type(FlagCommitOutputs)
|
||||||
|
|
||||||
|
// Has returns true if the Type has the passed flag enabled.
|
||||||
|
func (t Type) Has(flag Flag) bool {
|
||||||
|
return Flag(t)&flag == flag
|
||||||
|
}
|
||||||
|
|
||||||
|
// TypeFromFlags creates a single Type from an arbitrary list of flags.
|
||||||
|
func TypeFromFlags(flags ...Flag) Type {
|
||||||
|
var typ Type
|
||||||
|
for _, flag := range flags {
|
||||||
|
typ |= Type(flag)
|
||||||
|
}
|
||||||
|
|
||||||
|
return typ
|
||||||
|
}
|
||||||
|
|
||||||
|
// knownFlags maps the supported flags to their name.
|
||||||
|
var knownFlags = map[Flag]struct{}{
|
||||||
|
FlagReward: {},
|
||||||
|
FlagCommitOutputs: {},
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns a human readable description of a Type.
|
||||||
|
func (t Type) String() string {
|
||||||
|
var (
|
||||||
|
hrPieces []string
|
||||||
|
hasUnknownFlags bool
|
||||||
|
)
|
||||||
|
|
||||||
|
// Iterate through the possible flags from highest to lowest. This will
|
||||||
|
// ensure that the human readable names will be in the same order as the
|
||||||
|
// bits (left to right) if the type were to be printed in big-endian
|
||||||
|
// byte order.
|
||||||
|
for f := Flag(1 << 15); f != 0; f >>= 1 {
|
||||||
|
// If this flag is known, we'll add a human-readable name or its
|
||||||
|
// inverse depending on whether the type has this flag set.
|
||||||
|
if _, ok := knownFlags[f]; ok {
|
||||||
|
if t.Has(f) {
|
||||||
|
hrPieces = append(hrPieces, f.String())
|
||||||
|
} else {
|
||||||
|
hrPieces = append(hrPieces, "No-"+f.String())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Make note of any unknown flags that this type has
|
||||||
|
// set. If any are present, we'll prepend the bit-wise
|
||||||
|
// representation of the type in the final string.
|
||||||
|
if t.Has(f) {
|
||||||
|
hasUnknownFlags = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If there were no unknown flags, we'll simply return the list of human
|
||||||
|
// readable pieces.
|
||||||
|
if !hasUnknownFlags {
|
||||||
|
return fmt.Sprintf("[%s]", strings.Join(hrPieces, "|"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, we'll prepend the bit-wise representation to the human
|
||||||
|
// readable names.
|
||||||
|
return fmt.Sprintf("%016b[%s]", t, strings.Join(hrPieces, "|"))
|
||||||
|
}
|
||||||
|
|
||||||
|
// supportedTypes is the set of all configurations known to be supported by the
|
||||||
|
// package.
|
||||||
|
var supportedTypes = map[Type]struct{}{
|
||||||
|
FlagCommitOutputs.Type(): {},
|
||||||
|
(FlagCommitOutputs | FlagReward).Type(): {},
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsSupportedType returns true if the given type is supported by the package.
|
||||||
|
func IsSupportedType(blobType Type) bool {
|
||||||
|
_, ok := supportedTypes[blobType]
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
|
// SupportedTypes returns a list of all supported blob types.
|
||||||
|
func SupportedTypes() []Type {
|
||||||
|
supported := make([]Type, 0, len(supportedTypes))
|
||||||
|
for t := range supportedTypes {
|
||||||
|
supported = append(supported, t)
|
||||||
|
}
|
||||||
|
return supported
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user