2016-05-04 20:53:48 +02:00
|
|
|
// Copyright (C) 2016 Opsmate, Inc.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla
|
|
|
|
// Public License, v. 2.0. If a copy of the MPL was not distributed
|
|
|
|
// with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
//
|
|
|
|
// This software is distributed WITHOUT A WARRANTY OF ANY KIND.
|
|
|
|
// See the Mozilla Public License for details.
|
|
|
|
|
2016-05-04 20:49:07 +02:00
|
|
|
package certspotter
|
2016-02-17 23:54:25 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/sha256"
|
2017-01-06 00:43:26 +01:00
|
|
|
"encoding/json"
|
2017-01-05 23:48:35 +01:00
|
|
|
"errors"
|
2016-07-28 20:55:46 +02:00
|
|
|
"software.sslmate.com/src/certspotter/ct"
|
2016-02-17 23:54:25 +01:00
|
|
|
)
|
|
|
|
|
2016-07-28 20:55:46 +02:00
|
|
|
func reverseHashes(hashes []ct.MerkleTreeNode) {
|
|
|
|
for i := 0; i < len(hashes)/2; i++ {
|
2016-02-17 23:54:25 +01:00
|
|
|
j := len(hashes) - i - 1
|
|
|
|
hashes[i], hashes[j] = hashes[j], hashes[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-06 06:06:37 +01:00
|
|
|
func VerifyConsistencyProof(proof ct.ConsistencyProof, first *ct.SignedTreeHead, second *ct.SignedTreeHead) bool {
|
2017-01-05 23:32:28 +01:00
|
|
|
// TODO: make sure every hash in proof is right length? otherwise input to hashChildren is ambiguous
|
2016-02-17 23:54:25 +01:00
|
|
|
if second.TreeSize < first.TreeSize {
|
|
|
|
// Can't be consistent if tree got smaller
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
if first.TreeSize == second.TreeSize {
|
2016-11-15 21:38:49 +01:00
|
|
|
if !(bytes.Equal(first.SHA256RootHash[:], second.SHA256RootHash[:]) && len(proof) == 0) {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-11-15 21:38:49 +01:00
|
|
|
}
|
2017-01-06 06:06:37 +01:00
|
|
|
return true
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
if first.TreeSize == 0 {
|
|
|
|
// The purpose of the consistency proof is to ensure the append-only
|
|
|
|
// nature of the tree; i.e. that the first tree is a "prefix" of the
|
|
|
|
// second tree. If the first tree is empty, then it's trivially a prefix
|
|
|
|
// of the second tree, so no proof is needed.
|
2016-02-18 20:58:00 +01:00
|
|
|
if len(proof) != 0 {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-18 20:58:00 +01:00
|
|
|
}
|
2017-01-06 06:06:37 +01:00
|
|
|
return true
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
// Guaranteed that 0 < first.TreeSize < second.TreeSize
|
|
|
|
|
|
|
|
node := first.TreeSize - 1
|
|
|
|
lastNode := second.TreeSize - 1
|
|
|
|
|
|
|
|
// While we're the right child, everything is in both trees, so move one level up.
|
2016-07-28 20:55:46 +02:00
|
|
|
for node%2 == 1 {
|
2016-02-17 23:54:25 +01:00
|
|
|
node /= 2
|
|
|
|
lastNode /= 2
|
|
|
|
}
|
|
|
|
|
|
|
|
var newHash ct.MerkleTreeNode
|
|
|
|
var oldHash ct.MerkleTreeNode
|
|
|
|
if node > 0 {
|
|
|
|
if len(proof) == 0 {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
newHash = proof[0]
|
|
|
|
proof = proof[1:]
|
|
|
|
} else {
|
|
|
|
// The old tree was balanced, so we already know the first hash to use
|
|
|
|
newHash = first.SHA256RootHash[:]
|
|
|
|
}
|
|
|
|
oldHash = newHash
|
|
|
|
|
|
|
|
for node > 0 {
|
2016-07-28 20:55:46 +02:00
|
|
|
if node%2 == 1 {
|
2016-02-17 23:54:25 +01:00
|
|
|
// node is a right child; left sibling exists in both trees
|
|
|
|
if len(proof) == 0 {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
newHash = hashChildren(proof[0], newHash)
|
|
|
|
oldHash = hashChildren(proof[0], oldHash)
|
|
|
|
proof = proof[1:]
|
|
|
|
} else if node < lastNode {
|
|
|
|
// node is a left child; rigth sibling only exists in the new tree
|
|
|
|
if len(proof) == 0 {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
newHash = hashChildren(newHash, proof[0])
|
|
|
|
proof = proof[1:]
|
2016-07-29 00:52:32 +02:00
|
|
|
} // else node == lastNode: node is a left child with no sibling in either tree
|
2016-02-17 23:54:25 +01:00
|
|
|
node /= 2
|
|
|
|
lastNode /= 2
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Equal(oldHash, first.SHA256RootHash[:]) {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// If trees have different height, continue up the path to reach the new root
|
|
|
|
for lastNode > 0 {
|
|
|
|
if len(proof) == 0 {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
newHash = hashChildren(newHash, proof[0])
|
|
|
|
proof = proof[1:]
|
|
|
|
lastNode /= 2
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Equal(newHash, second.SHA256RootHash[:]) {
|
2017-01-06 06:06:37 +01:00
|
|
|
return false
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-06 06:06:37 +01:00
|
|
|
return true
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-05 23:32:22 +01:00
|
|
|
func hashNothing() ct.MerkleTreeNode {
|
|
|
|
return sha256.New().Sum(nil)
|
|
|
|
}
|
|
|
|
|
2016-07-28 20:55:46 +02:00
|
|
|
func hashLeaf(leafBytes []byte) ct.MerkleTreeNode {
|
2016-02-17 23:54:25 +01:00
|
|
|
hasher := sha256.New()
|
|
|
|
hasher.Write([]byte{0x00})
|
|
|
|
hasher.Write(leafBytes)
|
|
|
|
return hasher.Sum(nil)
|
|
|
|
}
|
|
|
|
|
2016-07-28 20:55:46 +02:00
|
|
|
func hashChildren(left ct.MerkleTreeNode, right ct.MerkleTreeNode) ct.MerkleTreeNode {
|
2016-02-17 23:54:25 +01:00
|
|
|
hasher := sha256.New()
|
|
|
|
hasher.Write([]byte{0x01})
|
|
|
|
hasher.Write(left)
|
|
|
|
hasher.Write(right)
|
|
|
|
return hasher.Sum(nil)
|
|
|
|
}
|
|
|
|
|
2016-02-18 19:15:56 +01:00
|
|
|
type MerkleTreeBuilder struct {
|
2016-07-28 20:55:46 +02:00
|
|
|
stack []ct.MerkleTreeNode
|
2017-01-05 23:41:44 +01:00
|
|
|
numLeaves uint64 // number of hashes added so far
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
|
2017-01-05 23:48:35 +01:00
|
|
|
func calculateStackSize (numLeaves uint64) int {
|
|
|
|
stackSize := 0
|
|
|
|
for numLeaves > 0 {
|
|
|
|
stackSize += int(numLeaves & 1)
|
|
|
|
numLeaves >>= 1
|
|
|
|
}
|
|
|
|
return stackSize
|
|
|
|
}
|
|
|
|
func EmptyMerkleTreeBuilder () *MerkleTreeBuilder {
|
|
|
|
return &MerkleTreeBuilder{}
|
|
|
|
}
|
|
|
|
func NewMerkleTreeBuilder (stack []ct.MerkleTreeNode, numLeaves uint64) (*MerkleTreeBuilder, error) {
|
|
|
|
if len(stack) != calculateStackSize(numLeaves) {
|
|
|
|
return nil, errors.New("NewMerkleTreeBuilder: incorrect stack size")
|
|
|
|
}
|
|
|
|
return &MerkleTreeBuilder{stack: stack, numLeaves: numLeaves}, nil
|
|
|
|
}
|
|
|
|
|
2016-07-28 20:55:46 +02:00
|
|
|
func (builder *MerkleTreeBuilder) Add(hash ct.MerkleTreeNode) {
|
2016-02-17 23:54:25 +01:00
|
|
|
builder.stack = append(builder.stack, hash)
|
2017-01-05 23:41:44 +01:00
|
|
|
builder.numLeaves++
|
|
|
|
numLeaves := builder.numLeaves
|
|
|
|
for numLeaves%2 == 0 {
|
2016-02-17 23:54:25 +01:00
|
|
|
left, right := builder.stack[len(builder.stack)-2], builder.stack[len(builder.stack)-1]
|
|
|
|
builder.stack = builder.stack[:len(builder.stack)-2]
|
|
|
|
builder.stack = append(builder.stack, hashChildren(left, right))
|
2017-01-05 23:41:44 +01:00
|
|
|
numLeaves /= 2
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-26 02:43:07 +01:00
|
|
|
func (builder *MerkleTreeBuilder) CalculateRoot() ct.MerkleTreeNode {
|
2016-02-17 23:54:25 +01:00
|
|
|
if len(builder.stack) == 0 {
|
2017-01-05 23:32:22 +01:00
|
|
|
return hashNothing()
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
2016-11-26 02:43:07 +01:00
|
|
|
i := len(builder.stack) - 1
|
|
|
|
hash := builder.stack[i]
|
|
|
|
for i > 0 {
|
|
|
|
i -= 1
|
|
|
|
hash = hashChildren(builder.stack[i], hash)
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
2016-11-26 02:43:07 +01:00
|
|
|
return hash
|
2016-02-17 23:54:25 +01:00
|
|
|
}
|
2017-01-06 00:00:56 +01:00
|
|
|
|
|
|
|
func (builder *MerkleTreeBuilder) GetNumLeaves() uint64 {
|
|
|
|
return builder.numLeaves
|
|
|
|
}
|
2017-01-06 00:43:26 +01:00
|
|
|
|
|
|
|
func (builder *MerkleTreeBuilder) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(map[string]interface{}{
|
|
|
|
"stack": builder.stack,
|
|
|
|
"num_leaves": builder.numLeaves,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (builder *MerkleTreeBuilder) UnmarshalJSON(b []byte) error {
|
|
|
|
var rawBuilder struct {
|
|
|
|
Stack []ct.MerkleTreeNode `json:"stack"`
|
|
|
|
NumLeaves uint64 `json:"num_leaves"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(b, &rawBuilder); err != nil {
|
|
|
|
return errors.New("Failed to unmarshal MerkleTreeBuilder: " + err.Error())
|
|
|
|
}
|
|
|
|
if len(rawBuilder.Stack) != calculateStackSize(rawBuilder.NumLeaves) {
|
|
|
|
return errors.New("Failed to unmarshal MerkleTreeBuilder: invalid stack size")
|
|
|
|
}
|
|
|
|
builder.numLeaves = rawBuilder.NumLeaves
|
|
|
|
builder.stack = rawBuilder.Stack
|
|
|
|
return nil
|
|
|
|
}
|