@@ -23,8 +23,24 @@ import (
23
23
"github.com/hyperledger/firefly-signer/pkg/ethtypes"
24
24
"github.com/hyperledger/firefly-signer/pkg/rlp"
25
25
"github.com/hyperledger/firefly-signer/pkg/secp256k1"
26
+ "golang.org/x/crypto/sha3"
26
27
)
27
28
29
+ type TransactionSignaturePayload struct {
30
+ rlpList rlp.List
31
+ data []byte
32
+ }
33
+
34
+ func (sp * TransactionSignaturePayload ) Bytes () []byte {
35
+ return sp .data
36
+ }
37
+
38
+ func (sp * TransactionSignaturePayload ) Hash () ethtypes.HexBytes0xPrefix {
39
+ msgHash := sha3 .NewLegacyKeccak256 ()
40
+ msgHash .Write (sp .data )
41
+ return msgHash .Sum (nil )
42
+ }
43
+
28
44
const (
29
45
TransactionTypeLegacy byte = 0x00
30
46
TransactionType2930 byte = 0x01 // unused
@@ -87,47 +103,85 @@ func (t *Transaction) Sign(signer *secp256k1.KeyPair, chainID int64) ([]byte, er
87
103
return t .SignLegacyEIP155 (signer , chainID )
88
104
}
89
105
90
- // SignLegacyOriginal uses legacy transaction structure, with legacy V value (27/28)
91
- func (t * Transaction ) SignLegacyOriginal (signer * secp256k1.KeyPair ) ([]byte , error ) {
106
+ // Returns the bytes that would be used to sign the transaction, without actually
107
+ // perform the signing. Can be used with Recover to verify a signing result.
108
+ func (t * Transaction ) SignaturePayload (chainID int64 ) (sp * TransactionSignaturePayload ) {
109
+ if t .MaxPriorityFeePerGas .BigInt ().Sign () > 0 || t .MaxFeePerGas .BigInt ().Sign () > 0 {
110
+ return t .SignaturePayloadEIP1559 (chainID )
111
+ }
112
+ return t .SignaturePayloadLegacyEIP155 (chainID )
113
+ }
114
+
115
+ // SignaturePayloadLegacyOriginal returns the rlpList of fields that are signed, and the
116
+ // bytes. Note that for legacy and EIP-155 transactions (everything prior to EIP-2718),
117
+ // there is no transaction type byte added (so the bytes are exactly rlpList.Encode())
118
+ func (t * Transaction ) SignaturePayloadLegacyOriginal () * TransactionSignaturePayload {
92
119
rlpList := t .BuildLegacy ()
120
+ return & TransactionSignaturePayload {
121
+ rlpList : rlpList ,
122
+ data : rlpList .Encode (),
123
+ }
124
+ }
93
125
94
- txData := rlpList .Encode ()
95
- sig , err := signer .Sign (txData )
126
+ // SignLegacyOriginal uses legacy transaction structure, with legacy V value (27/28)
127
+ func (t * Transaction ) SignLegacyOriginal (signer * secp256k1.KeyPair ) ([]byte , error ) {
128
+ signatureData := t .SignaturePayloadLegacyOriginal ()
129
+ sig , err := signer .Sign (signatureData .data )
96
130
if err != nil {
97
131
return nil , err
98
132
}
99
133
100
- rlpList = t .addSignature (rlpList , sig )
134
+ rlpList : = t .addSignature (signatureData . rlpList , sig )
101
135
return rlpList .Encode (), nil
102
136
}
103
137
138
+ // SignaturePayloadLegacyEIP155 returns the rlpList of fields that are signed, and the
139
+ // bytes. Note that for legacy and EIP-155 transactions (everything prior to EIP-2718),
140
+ // there is no transaction type byte added (so the bytes are exactly rlpList.Encode())
141
+ func (t * Transaction ) SignaturePayloadLegacyEIP155 (chainID int64 ) * TransactionSignaturePayload {
142
+ rlpList := t .BuildLegacy ()
143
+ rlpList = t .AddEIP155HashValues (rlpList , chainID )
144
+ return & TransactionSignaturePayload {
145
+ rlpList : rlpList ,
146
+ data : rlpList .Encode (),
147
+ }
148
+ }
149
+
104
150
// SignLegacyEIP155 uses legacy transaction structure, with EIP-155 signing V value (2*ChainID + 35 + Y-parity)
105
151
func (t * Transaction ) SignLegacyEIP155 (signer * secp256k1.KeyPair , chainID int64 ) ([]byte , error ) {
106
- rlpList := t .BuildLegacy ()
107
152
108
- rlpList = t .AddEIP155HashValues ( rlpList , chainID )
153
+ signaturePayload : = t .SignaturePayloadLegacyEIP155 ( chainID )
109
154
110
- hashData := rlpList .Encode ()
111
- sig , err := signer .Sign (hashData )
155
+ sig , err := signer .Sign (signaturePayload .data )
112
156
if err != nil {
113
157
return nil , err
114
158
}
115
159
116
160
// Use the EIP-155 V value, of (2*ChainID + 35 + Y-parity)
117
161
sig .UpdateEIP155 (chainID )
118
162
119
- rlpList = t .addSignature (rlpList [0 :6 ] /* we don't include the chainID+0+0 hash values in the payload */ , sig )
163
+ rlpList : = t .addSignature (signaturePayload . rlpList [0 :6 ] /* we don't include the chainID+0+0 hash values in the payload */ , sig )
120
164
return rlpList .Encode (), nil
121
165
}
122
166
123
- // SignEIP1559 uses EIP-1559 transaction structure (with EIP-2718 transaction type byte), with EIP-2930 V value (0 / 1 - direct parity-Y)
124
- func (t * Transaction ) SignEIP1559 (signer * secp256k1.KeyPair , chainID int64 ) ([]byte , error ) {
167
+ // SignaturePayloadEIP1559 returns the rlpList of fields that are signed, along with the full
168
+ // bytes for the signature / TX Hash - which have the transaction type prefixed
169
+ func (t * Transaction ) SignaturePayloadEIP1559 (chainID int64 ) * TransactionSignaturePayload {
125
170
rlpList := t .Build1559 (chainID )
126
171
127
- // First sign the transaction type, concattented with RLP list _excluding_ signature
172
+ // The signature payload is the transaction type, concatenated with RLP list _excluding_ signature
128
173
// keccak256(0x02 || rlp([chain_id, nonce, max_priority_fee_per_gas, max_fee_per_gas, gas_limit, destination, amount, data, access_list]))
129
- b := append ([]byte {TransactionType1559 }, rlpList .Encode ()... )
130
- sig , err := signer .Sign (b )
174
+ return & TransactionSignaturePayload {
175
+ rlpList : rlpList ,
176
+ data : append ([]byte {TransactionType1559 }, rlpList .Encode ()... ),
177
+ }
178
+ }
179
+
180
+ // SignEIP1559 uses EIP-1559 transaction structure (with EIP-2718 transaction type byte), with EIP-2930 V value (0 / 1 - direct parity-Y)
181
+ func (t * Transaction ) SignEIP1559 (signer * secp256k1.KeyPair , chainID int64 ) ([]byte , error ) {
182
+
183
+ signaturePayload := t .SignaturePayloadEIP1559 (chainID )
184
+ sig , err := signer .Sign (signaturePayload .data )
131
185
if err != nil {
132
186
return nil , err
133
187
}
@@ -137,7 +191,7 @@ func (t *Transaction) SignEIP1559(signer *secp256k1.KeyPair, chainID int64) ([]b
137
191
138
192
// Now we need a new RLP array, _including_ signature
139
193
// 0x02 || rlp([chain_id, nonce, max_priority_fee_per_gas, max_fee_per_gas, gas_limit, destination, amount, data, access_list, signature_y_parity, signature_r, signature_s])
140
- rlpList = t .addSignature (rlpList , sig )
194
+ rlpList : = t .addSignature (signaturePayload . rlpList , sig )
141
195
return append ([]byte {TransactionType1559 }, rlpList .Encode ()... ), nil
142
196
}
143
197
0 commit comments