06-08-2018: Minor updates
27-07-2018: Update draft to use amino encoding
11-07-2018: Initial Draft
5-26-2021: Multisigs were moved into the Cosmos-sdk
Multisignatures, or technically Accountable Subgroup Multisignatures (ASM), are signature schemes which enable any subgroup of a set of signers to sign any message, and reveal to the verifier exactly who the signers were. This allows for complex conditionals of when to validate a signature.
Suppose the set of signers is of size n. If we validate a signature if any subgroup of size k signs a message, this becomes what is commonly referred to as a k of n multisig in Bitcoin.
This ADR specifies the encoding standard for general accountable subgroup multisignatures, k of n accountable subgroup multisignatures, and its weighted variant.
In the future, we can also allow for more complex conditionals on the accountable subgroup.
Every ASM will then have its own struct, implementing the crypto.Pubkey interface.
This ADR assumes that replacing crypto.Signature with []bytes has been accepted.
The pubkey is the following struct:
type ThresholdMultiSignaturePubKey struct { // K of N threshold multisig
K uint `json:"threshold"`
Pubkeys []crypto.Pubkey `json:"pubkeys"`
}
We will derive N from the length of pubkeys. (For spatial efficiency in encoding)
Verify
will expect an []byte
encoded version of the Multisignature.
(Multisignature is described in the next section)
The multisignature will be rejected if the bitmap has less than k indices,
or if any signature at any of the k indices is not a valid signature from
the kth public key on the message.
(If more than k signatures are included, all must be valid)
Bytes
will be the amino encoded version of the pubkey.
Address will be Hash(amino_encoded_pubkey)
The reason this doesn't use log_8(n)
bytes per signer is because that heavily optimizes for the case where a very small number of signers are required.
e.g. for n
of size 24
, that would only be more space efficient for k < 3
.
This seems less likely, and that it should not be the case optimized for.
The pubkey is the following struct:
type WeightedThresholdMultiSignaturePubKey struct {
Weights []uint `json:"weights"`
Threshold uint `json:"threshold"`
Pubkeys []crypto.Pubkey `json:"pubkeys"`
}
Weights and Pubkeys must be of the same length. Everything else proceeds identically to the K of N multisig, except the multisig fails if the sum of the weights is less than the threshold.
The inter-mediate phase of the signatures (as it accrues more signatures) will be the following struct:
type Multisignature struct {
BitArray CryptoBitArray // Documented later
Sigs [][]byte
It is important to recall that each private key will output a signature on the provided message itself.
So no signing algorithm ever outputs the multisignature.
The UI will take a signature, cast into a multisignature, and then keep adding
new signatures into it, and when done marshal into []byte
.
This will require the following helper methods:
func SigToMultisig(sig []byte, n int)
func GetIndex(pk crypto.Pubkey, []crypto.Pubkey)
func AddSignature(sig Signature, index int, multiSig *Multisignature)
The multisignature will be converted to an []byte
using amino.MarshalBinaryBare. *
We would be using a new implementation of a bitarray. The struct it would be encoded/decoded from is
type CryptoBitArray struct {
ExtraBitsStored byte `json:"extra_bits"` // The number of extra bits in elems.
Elems []byte `json:"elems"`
}
The reason for not using the BitArray currently implemented in libs/common/bit_array.go
is that it is less space efficient, due to a space / time trade-off.
Evidence for this is outlined in this issue.
In the multisig, we will not be performing arithmetic operations,
so there is no performance increase with the current implementation,
and just loss of spatial efficiency.
Implementing this new bit array with []byte
should be simple, as no
arithmetic operations between bit arrays are required, and save a couple of bytes.
(Explained in that same issue)
When this bit array encoded, the number of elements is encoded due to amino.
However we may be encoding a full byte for what we actually only need 1-7 bits for.
We store that difference in ExtraBitsStored.
This allows for us to have an unbounded number of signers, and is more space efficient than what is currently used in libs/common
.
Again the implementation of this space saving feature is straight forward.
We will use straight forward amino encoding. This is chosen for ease of compatibility in other languages.
If desired, we can use ed25519 batch verification for all ed25519 keys. This is a future point of discussion, but would be backwards compatible as this information won't need to be marshalled. (There may even be cofactor concerns without ristretto) Aggregation of pubkeys / sigs in Schnorr sigs / BLS sigs is not backwards compatible, and would need to be a new ASM type.
Implemented (moved to cosmos-sdk)
- Supports multisignatures, in a way that won't require any special cases in our downstream verification code.
- Easy to serialize / deserialize
- Unbounded number of signers
- Larger codebase, however this should reside in a subfolder of tendermint/crypto, as it provides no new interfaces. (Ref #tendermint/go-crypto#136)
- Space inefficient due to utilization of amino encoding
- Suggested implementation requires a new struct for every ASM.