forked from dcb9/janus
-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy patheth_estimateGas.go
70 lines (58 loc) · 2.15 KB
/
eth_estimateGas.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
package transformer
import (
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/labstack/echo"
"github.com/pkg/errors"
"github.com/qtumproject/janus/pkg/eth"
"github.com/qtumproject/janus/pkg/qtum"
)
// 22000
var NonContractVMGasLimit = "0x55f0"
var ErrExecutionReverted = errors.New("execution reverted")
// 10% isn't enough in some cases, neither is 15%, 20% works
var GAS_BUFFER = 1.20
// ProxyETHEstimateGas implements ETHProxy
type ProxyETHEstimateGas struct {
*ProxyETHCall
}
func (p *ProxyETHEstimateGas) Method() string {
return "eth_estimateGas"
}
func (p *ProxyETHEstimateGas) Request(rawreq *eth.JSONRPCRequest, c echo.Context) (interface{}, eth.JSONRPCError) {
var ethreq eth.CallRequest
if jsonErr := unmarshalRequest(rawreq.Params, ðreq); jsonErr != nil {
// TODO: Correct error code?
return nil, eth.NewInvalidParamsError(jsonErr.Error())
}
if ethreq.Data == "" {
response := eth.EstimateGasResponse(NonContractVMGasLimit)
return &response, nil
}
// when supplying this parameter to callcontract to estimate gas in the qtum api
// if there isn't enough gas specified here, the result will be an exception
// Excepted = "OutOfGasIntrinsic"
// Gas = "the supplied value"
// this is different from geth's behavior
// which will return a used gas value that is higher than the incoming gas parameter
// so we set this to nil so that callcontract will return the actual gas estimate
ethreq.Gas = nil
// eth req -> qtum req
qtumreq, jsonErr := p.ToRequest(ðreq)
if jsonErr != nil {
return nil, jsonErr
}
// qtum [code: -5] Incorrect address occurs here
qtumresp, err := p.CallContract(c.Request().Context(), qtumreq)
if err != nil {
return nil, eth.NewCallbackError(err.Error())
}
return p.toResp(qtumresp)
}
func (p *ProxyETHEstimateGas) toResp(qtumresp *qtum.CallContractResponse) (*eth.EstimateGasResponse, eth.JSONRPCError) {
if qtumresp.ExecutionResult.Excepted != "None" {
return nil, eth.NewCallbackError(ErrExecutionReverted.Error())
}
gas := eth.EstimateGasResponse(hexutil.EncodeUint64(uint64(float64(qtumresp.ExecutionResult.GasUsed) * GAS_BUFFER)))
p.GetDebugLogger().Log(p.Method(), gas)
return &gas, nil
}