-
Notifications
You must be signed in to change notification settings - Fork 136
/
Copy pathgroup_operations.go
175 lines (148 loc) · 5.35 KB
/
group_operations.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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
// Copyright 2024 Coinbase, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License 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 parser
import (
"sort"
"github.com/coinbase/rosetta-sdk-go/asserter"
"github.com/coinbase/rosetta-sdk-go/types"
)
// OperationGroup is a group of related operations
// If all operations in a group have the same operation.Type,
// the Type is also populated.
type OperationGroup struct {
Type string
Operations []*types.Operation
Currencies []*types.Currency
NilAmountPresent bool
}
func containsInt(valid []int, value int) bool {
for _, v := range valid {
if v == value {
return true
}
}
return false
}
// addOperationToGroup appends a *types.Operation to an *OperationGroup.
func addOperationToGroup(
destination *OperationGroup,
destinationIndex int,
assignments *[]int,
op *types.Operation,
) {
// Remove group type if different
if op.Type != destination.Type && destination.Type != "" {
destination.Type = ""
}
// Update op assignment
destination.Operations = append(destination.Operations, op)
(*assignments)[op.OperationIdentifier.Index] = destinationIndex
// Handle nil currency
if op.Amount == nil {
destination.NilAmountPresent = true
return
}
// Add op to currency if amount is not nil
if !asserter.ContainsCurrency(destination.Currencies, op.Amount.Currency) {
destination.Currencies = append(destination.Currencies, op.Amount.Currency)
}
}
// sortOperationGroups returns a slice of OperationGroups sorted by the lowest
// OperationIdentifier.Index in each group. This function also sorts all
// operations in each OperationGroup by OperationIdentifier.Index. It can be
// useful to consumers to have a deterministic ordering of groups and ops within
// each group.
func sortOperationGroups(opLen int, opGroups map[int]*OperationGroup) []*OperationGroup {
sliceGroups := []*OperationGroup{}
// Golang map ordering is non-deterministic.
// Return groups sorted by lowest op in group
for i := 0; i < opLen; i++ {
v, ok := opGroups[i]
if !ok {
continue
}
// Sort all operations by index in a group
sort.Slice(v.Operations, func(i, j int) bool {
return v.Operations[i].OperationIdentifier.Index < v.Operations[j].OperationIdentifier.Index
})
sliceGroups = append(sliceGroups, v)
}
return sliceGroups
}
// GroupOperations parses all of a transaction's opertations and returns a slice
// of each group of related operations (assuming transitive relatedness). This
// should ONLY be called on operations that have already been asserted for
// correctness. Assertion ensures there are no duplicate operation indexes,
// operations are sorted, and that operations only reference operations with
// an index less than theirs.
//
// OperationGroups are returned in ascending order based on the lowest
// OperationIdentifier.Index in the group. The operations in each OperationGroup
// are also sorted.
func GroupOperations(transaction *types.Transaction) []*OperationGroup {
ops := transaction.Operations
// We use a map of ints to keep track of *OperationGroup instead of a slice
// because merging groups involves removing and combing many items. While we
// could manipulate a slice (leaving holes where groups were merged), it
// seemed less complex to manipulate a map.
//
// Nonetheless, either solution avoids modifying up to `n` opAssignments
// whenever 2 groups merge (this occurs when merging groups in a slice without
// leaving holes).
opGroups := map[int]*OperationGroup{}
opAssignments := make([]int, len(ops))
for i, op := range ops {
// Create new group
if len(op.RelatedOperations) == 0 {
key := len(opGroups)
opGroups[key] = &OperationGroup{
Type: op.Type,
Operations: []*types.Operation{op},
}
if op.Amount != nil {
opGroups[key].Currencies = []*types.Currency{op.Amount.Currency}
} else {
opGroups[key].Currencies = []*types.Currency{}
opGroups[key].NilAmountPresent = true
}
opAssignments[i] = key
continue
}
// Find groups to merge
groupsToMerge := []int{}
for _, relatedOp := range op.RelatedOperations {
if !containsInt(groupsToMerge, opAssignments[relatedOp.Index]) {
groupsToMerge = append(groupsToMerge, opAssignments[relatedOp.Index])
}
}
// Ensure first index is lowest because all other groups
// will be merged into it.
sort.Ints(groupsToMerge)
mergedGroupIndex := groupsToMerge[0]
mergedGroup := opGroups[mergedGroupIndex]
// Add op to unified group
addOperationToGroup(mergedGroup, mergedGroupIndex, &opAssignments, op)
// Merge Groups
for _, otherGroupIndex := range groupsToMerge[1:] {
otherGroup := opGroups[otherGroupIndex]
// Add otherGroup ops to mergedGroup
for _, otherOp := range otherGroup.Operations {
addOperationToGroup(mergedGroup, mergedGroupIndex, &opAssignments, otherOp)
}
// Delete otherGroup
delete(opGroups, otherGroupIndex)
}
}
return sortOperationGroups(len(ops), opGroups)
}