-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathhasher.go
124 lines (101 loc) · 2.9 KB
/
hasher.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* or in the "license" file accompanying this file. This file is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
* express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
package ionhash
import (
"github.com/amzn/ion-go/ion"
"github.com/amzn/ion-hash-go/internal"
)
type hasher struct {
hasherProvider IonHasherProvider
currentHasher serializer
hasherStack internal.Stack
}
func newHasher(hasherProvider IonHasherProvider) (*hasher, error) {
newHasher, err := hasherProvider.NewHasher()
if err != nil {
return nil, err
}
currentHasher := newScalarSerializer(newHasher, 0)
var hasherStack internal.Stack
hasherStack.Push(currentHasher)
return &hasher{hasherProvider, currentHasher, hasherStack}, nil
}
func (h *hasher) scalar(ionValue hashValue) error {
return h.currentHasher.scalar(ionValue)
}
func (h *hasher) stepIn(ionValue hashValue) error {
var hashFunction IonHasher
if _, ok := h.currentHasher.(*structSerializer); ok {
newHasher, err := h.hasherProvider.NewHasher()
if err != nil {
return err
}
hashFunction = newHasher
} else {
hashFunction = h.currentHasher.(*scalarSerializer).hashFunction
}
if ionValue.Type() == ion.StructType {
newStructSerializer, err := newStructSerializer(hashFunction, h.depth(), h.hasherProvider)
if err != nil {
return err
}
h.currentHasher = newStructSerializer
} else {
h.currentHasher = newScalarSerializer(hashFunction, h.depth())
}
h.hasherStack.Push(h.currentHasher)
return h.currentHasher.stepIn(ionValue)
}
func (h *hasher) stepOut() error {
if h.depth() == 0 {
return &InvalidOperationError{"hasher", "stepOut", "Depth is zero. Hasher cannot step out any further"}
}
err := h.currentHasher.stepOut()
if err != nil {
return err
}
poppedHasher, err := h.hasherStack.Pop()
if err != nil {
return &InvalidOperationError{
"hasher",
"stepOut",
err.Error(),
}
}
peekedHasher, err := h.hasherStack.Peek()
if err != nil {
return &InvalidOperationError{
"hasher",
"stepOut",
err.Error(),
}
}
h.currentHasher = peekedHasher.(serializer)
if structHasher, ok := h.currentHasher.(*structSerializer); ok {
sum := poppedHasher.(serializer).sum(nil)
structHasher.appendFieldHash(sum)
}
return nil
}
func (h *hasher) sum(b []byte) ([]byte, error) {
if h.depth() != 0 {
return nil, &InvalidOperationError{
"hasher", "sum", "A sum may only be provided at the same depth hashing started"}
}
return h.currentHasher.sum(b), nil
}
func (h *hasher) depth() int {
return h.hasherStack.Size() - 1
}