-
Notifications
You must be signed in to change notification settings - Fork 39
/
client_api_eth.go
265 lines (230 loc) · 10.6 KB
/
client_api_eth.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
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
package clientapi
import (
"context"
"errors"
"fmt"
"math/big"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/ethereum/go-ethereum/params"
"github.com/ten-protocol/go-ten/go/common"
"github.com/ten-protocol/go-ten/go/common/host"
"github.com/ten-protocol/go-ten/go/common/log"
"github.com/ten-protocol/go-ten/go/responses"
"github.com/ten-protocol/go-ten/lib/gethfork/rpc"
gethcommon "github.com/ethereum/go-ethereum/common"
gethlog "github.com/ethereum/go-ethereum/log"
)
// EthereumAPI implements a subset of the Ethereum JSON RPC operations. All the method signatures are copied from the
// corresponding Geth implementations.
type EthereumAPI struct {
host host.Host
logger gethlog.Logger
}
func NewEthereumAPI(host host.Host, logger gethlog.Logger) *EthereumAPI {
return &EthereumAPI{
host: host,
logger: logger,
}
}
// ChainId returns the Obscuro chain ID.
func (api *EthereumAPI) ChainId() (*hexutil.Big, error) { //nolint:stylecheck,revive
return (*hexutil.Big)(big.NewInt(api.host.Config().ObscuroChainID)), nil
}
// BlockNumber returns the height of the current head batch.
func (api *EthereumAPI) BlockNumber() hexutil.Uint64 {
header, err := api.host.Storage().FetchHeadBatchHeader()
if err != nil {
// This error may be nefarious, but unfortunately the Eth API doesn't allow us to return an error.
api.logger.Error("could not retrieve head batch header", log.ErrKey, err)
return 0
}
return hexutil.Uint64(header.Number.Uint64())
}
// GetBlockByNumber returns the header of the batch with the given height.
func (api *EthereumAPI) GetBlockByNumber(ctx context.Context, number rpc.BlockNumber, _ bool) (*common.BatchHeader, error) {
batchHash, err := api.batchNumberToBatchHash(number)
if err != nil {
return nil, fmt.Errorf("could not find batch with height %d. Cause: %w", number, err)
}
return api.GetBlockByHash(ctx, *batchHash, true)
}
// GetBlockByHash returns the header of the batch with the given hash.
func (api *EthereumAPI) GetBlockByHash(_ context.Context, hash gethcommon.Hash, _ bool) (*common.BatchHeader, error) {
batchHeader, err := api.host.Storage().FetchBatchHeaderByHash(hash)
if err != nil {
return nil, err
}
return batchHeader, nil
}
// GasPrice is a placeholder for an RPC method required by MetaMask/Remix.
func (api *EthereumAPI) GasPrice(context.Context) (*hexutil.Big, error) {
header, err := api.host.Storage().FetchHeadBatchHeader()
if err != nil {
return nil, err
}
if header.BaseFee == nil || header.BaseFee.Cmp(gethcommon.Big0) == 0 {
return (*hexutil.Big)(big.NewInt(params.InitialBaseFee)), nil
}
return (*hexutil.Big)(big.NewInt(0).Set(header.BaseFee)), nil
}
// GetBalance returns the address's balance on the Obscuro network, encrypted with the viewing key corresponding to the
// `address` field and encoded as hex.
func (api *EthereumAPI) GetBalance(ctx context.Context, encryptedParams common.EncryptedParamsGetBalance) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().GetBalance(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("GetBalance", sysError)
}
return *enclaveResponse, nil
}
// Call returns the result of executing the smart contract as a user, encrypted with the viewing key corresponding to
// the `from` field and encoded as hex.
func (api *EthereumAPI) Call(ctx context.Context, encryptedParams common.EncryptedParamsCall) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().ObsCall(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("Call", sysError)
}
return *enclaveResponse, nil
}
// GetTransactionReceipt returns the transaction receipt for the given transaction hash, encrypted with the viewing key
// corresponding to the original transaction submitter and encoded as hex, or nil if no matching transaction exists.
func (api *EthereumAPI) GetTransactionReceipt(ctx context.Context, encryptedParams common.EncryptedParamsGetTxReceipt) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().GetTransactionReceipt(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("GetTransactionReceipt", sysError)
}
return *enclaveResponse, nil
}
// EstimateGas requests the enclave the gas estimation based on the callMsg supplied params (encrypted)
func (api *EthereumAPI) EstimateGas(ctx context.Context, encryptedParams common.EncryptedParamsEstimateGas) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().EstimateGas(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("EstimateGas", sysError)
}
return *enclaveResponse, nil
}
// SendRawTransaction sends the encrypted transaction.
func (api *EthereumAPI) SendRawTransaction(ctx context.Context, encryptedParams common.EncryptedParamsSendRawTx) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.SubmitAndBroadcastTx(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("SendRawTransaction", sysError)
}
return *enclaveResponse, nil
}
// GetCode returns the code stored at the given address in the state for the given batch height or batch hash.
// todo (#1620) - instead of converting the block number of hash client-side, do it on the enclave
func (api *EthereumAPI) GetCode(ctx context.Context, address gethcommon.Address, blockNrOrHash rpc.BlockNumberOrHash) (hexutil.Bytes, error) {
var batchHash *gethcommon.Hash
// requested a number
if batchNumber, ok := blockNrOrHash.Number(); ok {
hash, err := api.batchNumberToBatchHash(batchNumber)
if err != nil {
return nil, fmt.Errorf("could not retrieve batch with height %d. Cause: %w", batchNumber, err)
}
batchHash = hash
}
// requested a hash
if hash, ok := blockNrOrHash.Hash(); ok {
batchHash = &hash
}
if batchHash == nil {
return nil, errors.New("invalid arguments; neither batch height nor batch hash specified")
}
code, sysError := api.host.EnclaveClient().GetCode(ctx, address, batchHash)
if sysError != nil {
api.logger.Error(fmt.Sprintf("Enclave System Error. Function %s", "GetCode"), log.ErrKey, sysError)
return nil, fmt.Errorf(responses.InternalErrMsg)
}
return code, nil
}
func (api *EthereumAPI) GetTransactionCount(ctx context.Context, encryptedParams common.EncryptedParamsGetTxCount) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().GetTransactionCount(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("GetTransactionCount", sysError)
}
return *enclaveResponse, nil
}
// GetTransactionByHash returns the transaction with the given hash, encrypted with the viewing key corresponding to the
// `from` field and encoded as hex, or nil if no matching transaction exists.
func (api *EthereumAPI) GetTransactionByHash(ctx context.Context, encryptedParams common.EncryptedParamsGetTxByHash) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().GetTransaction(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("GetTransactionByHash", sysError)
}
return *enclaveResponse, nil
}
// GetStorageAt is not currently supported (some narrow version of it may be supported in the future for proxy contracts).
func (api *EthereumAPI) GetStorageAt(ctx context.Context, encryptedParams common.EncryptedParamsGetStorageSlot) (responses.EnclaveResponse, error) {
enclaveResponse, sysError := api.host.EnclaveClient().GetStorageSlot(ctx, encryptedParams)
if sysError != nil {
return api.handleSysError("GetStorageAt", sysError)
}
return *enclaveResponse, nil
}
func (api *EthereumAPI) MaxPriorityFeePerGas(_ context.Context) (*hexutil.Big, error) {
// todo - implement with the gas mechanics
header, err := api.host.Storage().FetchHeadBatchHeader()
if err != nil {
api.logger.Error("Unable to retrieve header for fee history.", log.ErrKey, err)
return nil, fmt.Errorf("unable to retrieve MaxPriorityFeePerGas")
}
// just return the base fee?
return (*hexutil.Big)(header.BaseFee), err
}
// FeeHistory is a placeholder for an RPC method required by MetaMask/Remix.
// rpc.DecimalOrHex -> []byte
func (api *EthereumAPI) FeeHistory(context.Context, string, rpc.BlockNumber, []float64) (*FeeHistoryResult, error) {
// todo (#1621) - return a non-dummy fee history
header, err := api.host.Storage().FetchHeadBatchHeader()
if err != nil {
api.logger.Error("Unable to retrieve header for fee history.", log.ErrKey, err)
return nil, fmt.Errorf("unable to retrieve fee history")
}
batches := make([]*common.BatchHeader, 0)
batches = append(batches, header)
feeHist := &FeeHistoryResult{
OldestBlock: (*hexutil.Big)(header.Number),
Reward: [][]*hexutil.Big{},
BaseFee: []*hexutil.Big{},
GasUsedRatio: []float64{},
}
for _, header := range batches {
// 0.9 - This number represents how full the block is. As we dont have a dynamic base fee, we tell whomever is requesting that
// we expect the baseFee to increase, rather than decrease in order to avoid underpriced transactions.
feeHist.GasUsedRatio = append(feeHist.GasUsedRatio, 0.9)
feeHist.BaseFee = append(feeHist.BaseFee, (*hexutil.Big)(header.BaseFee))
}
return feeHist, nil
}
// FeeHistoryResult is the structure returned by Geth `eth_feeHistory` API.
type FeeHistoryResult struct {
OldestBlock *hexutil.Big `json:"oldestBlock"`
Reward [][]*hexutil.Big `json:"reward,omitempty"`
BaseFee []*hexutil.Big `json:"baseFeePerGas,omitempty"`
GasUsedRatio []float64 `json:"gasUsedRatio"`
}
// Given a batch number, returns the hash of the batch with that number.
func (api *EthereumAPI) batchNumberToBatchHash(batchNumber rpc.BlockNumber) (*gethcommon.Hash, error) {
// Handling the special cases first. No special handling is required for rpc.EarliestBlockNumber.
// note: our API currently treats all these block statuses the same for obscuro batches
if batchNumber == rpc.LatestBlockNumber || batchNumber == rpc.PendingBlockNumber ||
batchNumber == rpc.FinalizedBlockNumber || batchNumber == rpc.SafeBlockNumber {
batchHeader, err := api.host.Storage().FetchHeadBatchHeader()
if err != nil {
return nil, err
}
batchHash := batchHeader.Hash()
return &batchHash, nil
}
batchNumberBig := big.NewInt(batchNumber.Int64())
batchHash, err := api.host.Storage().FetchBatchHashByHeight(batchNumberBig)
if err != nil {
return nil, err
}
return batchHash, nil
}
func (api *EthereumAPI) handleSysError(function string, sysError common.SystemError) (responses.EnclaveResponse, error) {
api.logger.Error(fmt.Sprintf("Enclave System Error. Function %s", function), log.ErrKey, sysError)
return responses.EnclaveResponse{
Err: &responses.InternalErrMsg,
}, nil
}