rpc api: eth_getNatSpec
* xeth, rpc: implement eth_getNatSpec for tx confirmations * rename silly docserver -> httpclient * eth/backend: httpclient now accessible via eth.Ethereum init-d via config.DocRoot * cmd: introduce separate CLI flag for DocRoot (defaults to homedir) * common/path: delete unused assetpath func, separate HomeDir func
This commit is contained in:
parent
3b4ffacd0c
commit
4d005a2c1d
@ -30,7 +30,6 @@ import (
|
|||||||
|
|
||||||
"github.com/ethereum/go-ethereum/cmd/utils"
|
"github.com/ethereum/go-ethereum/cmd/utils"
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/docserver"
|
|
||||||
"github.com/ethereum/go-ethereum/common/natspec"
|
"github.com/ethereum/go-ethereum/common/natspec"
|
||||||
"github.com/ethereum/go-ethereum/common/registrar"
|
"github.com/ethereum/go-ethereum/common/registrar"
|
||||||
"github.com/ethereum/go-ethereum/eth"
|
"github.com/ethereum/go-ethereum/eth"
|
||||||
@ -77,8 +76,6 @@ func (r dumbterm) PasswordPrompt(p string) (string, error) {
|
|||||||
func (r dumbterm) AppendHistory(string) {}
|
func (r dumbterm) AppendHistory(string) {}
|
||||||
|
|
||||||
type jsre struct {
|
type jsre struct {
|
||||||
docRoot string
|
|
||||||
ds *docserver.DocServer
|
|
||||||
re *re.JSRE
|
re *re.JSRE
|
||||||
ethereum *eth.Ethereum
|
ethereum *eth.Ethereum
|
||||||
xeth *xeth.XEth
|
xeth *xeth.XEth
|
||||||
@ -153,7 +150,6 @@ func newLightweightJSRE(docRoot string, client comms.EthereumClient, datadir str
|
|||||||
js := &jsre{ps1: "> "}
|
js := &jsre{ps1: "> "}
|
||||||
js.wait = make(chan *big.Int)
|
js.wait = make(chan *big.Int)
|
||||||
js.client = client
|
js.client = client
|
||||||
js.ds = docserver.New(docRoot)
|
|
||||||
|
|
||||||
// update state in separare forever blocks
|
// update state in separare forever blocks
|
||||||
js.re = re.New(docRoot)
|
js.re = re.New(docRoot)
|
||||||
@ -181,18 +177,17 @@ func newLightweightJSRE(docRoot string, client comms.EthereumClient, datadir str
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newJSRE(ethereum *eth.Ethereum, docRoot, corsDomain string, client comms.EthereumClient, interactive bool, f xeth.Frontend) *jsre {
|
func newJSRE(ethereum *eth.Ethereum, docRoot, corsDomain string, client comms.EthereumClient, interactive bool, f xeth.Frontend) *jsre {
|
||||||
js := &jsre{ethereum: ethereum, ps1: "> ", docRoot: docRoot}
|
js := &jsre{ethereum: ethereum, ps1: "> "}
|
||||||
// set default cors domain used by startRpc from CLI flag
|
// set default cors domain used by startRpc from CLI flag
|
||||||
js.corsDomain = corsDomain
|
js.corsDomain = corsDomain
|
||||||
if f == nil {
|
if f == nil {
|
||||||
f = js
|
f = js
|
||||||
}
|
}
|
||||||
js.ds = docserver.New(docRoot)
|
|
||||||
js.xeth = xeth.New(ethereum, f)
|
js.xeth = xeth.New(ethereum, f)
|
||||||
js.wait = js.xeth.UpdateState()
|
js.wait = js.xeth.UpdateState()
|
||||||
js.client = client
|
js.client = client
|
||||||
if clt, ok := js.client.(*comms.InProcClient); ok {
|
if clt, ok := js.client.(*comms.InProcClient); ok {
|
||||||
if offeredApis, err := api.ParseApiString(shared.AllApis, codec.JSON, js.xeth, ethereum, docRoot); err == nil {
|
if offeredApis, err := api.ParseApiString(shared.AllApis, codec.JSON, js.xeth, ethereum); err == nil {
|
||||||
clt.Initialize(api.Merge(offeredApis...))
|
clt.Initialize(api.Merge(offeredApis...))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -281,7 +276,7 @@ func (js *jsre) apiBindings(f xeth.Frontend) error {
|
|||||||
apiNames = append(apiNames, a)
|
apiNames = append(apiNames, a)
|
||||||
}
|
}
|
||||||
|
|
||||||
apiImpl, err := api.ParseApiString(strings.Join(apiNames, ","), codec.JSON, js.xeth, js.ethereum, js.docRoot)
|
apiImpl, err := api.ParseApiString(strings.Join(apiNames, ","), codec.JSON, js.xeth, js.ethereum)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
utils.Fatalf("Unable to determine supported api's: %v", err)
|
utils.Fatalf("Unable to determine supported api's: %v", err)
|
||||||
}
|
}
|
||||||
@ -348,7 +343,7 @@ func (self *jsre) AskPassword() (string, bool) {
|
|||||||
|
|
||||||
func (self *jsre) ConfirmTransaction(tx string) bool {
|
func (self *jsre) ConfirmTransaction(tx string) bool {
|
||||||
if self.ethereum.NatSpec {
|
if self.ethereum.NatSpec {
|
||||||
notice := natspec.GetNotice(self.xeth, tx, self.ds)
|
notice := natspec.GetNotice(self.xeth, tx, self.ethereum.HTTPClient())
|
||||||
fmt.Println(notice)
|
fmt.Println(notice)
|
||||||
answer, _ := self.Prompt("Confirm Transaction [y/n]")
|
answer, _ := self.Prompt("Confirm Transaction [y/n]")
|
||||||
return strings.HasPrefix(strings.Trim(answer, " "), "y")
|
return strings.HasPrefix(strings.Trim(answer, " "), "y")
|
||||||
|
@ -31,7 +31,7 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/accounts"
|
"github.com/ethereum/go-ethereum/accounts"
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/compiler"
|
"github.com/ethereum/go-ethereum/common/compiler"
|
||||||
"github.com/ethereum/go-ethereum/common/docserver"
|
"github.com/ethereum/go-ethereum/common/httpclient"
|
||||||
"github.com/ethereum/go-ethereum/common/natspec"
|
"github.com/ethereum/go-ethereum/common/natspec"
|
||||||
"github.com/ethereum/go-ethereum/common/registrar"
|
"github.com/ethereum/go-ethereum/common/registrar"
|
||||||
"github.com/ethereum/go-ethereum/core"
|
"github.com/ethereum/go-ethereum/core"
|
||||||
@ -62,7 +62,7 @@ var (
|
|||||||
type testjethre struct {
|
type testjethre struct {
|
||||||
*jsre
|
*jsre
|
||||||
lastConfirm string
|
lastConfirm string
|
||||||
ds *docserver.DocServer
|
client *httpclient.HTTPClient
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *testjethre) UnlockAccount(acc []byte) bool {
|
func (self *testjethre) UnlockAccount(acc []byte) bool {
|
||||||
@ -75,7 +75,7 @@ func (self *testjethre) UnlockAccount(acc []byte) bool {
|
|||||||
|
|
||||||
func (self *testjethre) ConfirmTransaction(tx string) bool {
|
func (self *testjethre) ConfirmTransaction(tx string) bool {
|
||||||
if self.ethereum.NatSpec {
|
if self.ethereum.NatSpec {
|
||||||
self.lastConfirm = natspec.GetNotice(self.xeth, tx, self.ds)
|
self.lastConfirm = natspec.GetNotice(self.xeth, tx, self.client)
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
@ -101,6 +101,7 @@ func testREPL(t *testing.T, config func(*eth.Config)) (string, *testjethre, *eth
|
|||||||
AccountManager: am,
|
AccountManager: am,
|
||||||
MaxPeers: 0,
|
MaxPeers: 0,
|
||||||
Name: "test",
|
Name: "test",
|
||||||
|
DocRoot: "/",
|
||||||
SolcPath: testSolcPath,
|
SolcPath: testSolcPath,
|
||||||
PowTest: true,
|
PowTest: true,
|
||||||
NewDB: func(path string) (ethdb.Database, error) { return db, nil },
|
NewDB: func(path string) (ethdb.Database, error) { return db, nil },
|
||||||
@ -130,8 +131,7 @@ func testREPL(t *testing.T, config func(*eth.Config)) (string, *testjethre, *eth
|
|||||||
|
|
||||||
assetPath := filepath.Join(os.Getenv("GOPATH"), "src", "github.com", "ethereum", "go-ethereum", "cmd", "mist", "assets", "ext")
|
assetPath := filepath.Join(os.Getenv("GOPATH"), "src", "github.com", "ethereum", "go-ethereum", "cmd", "mist", "assets", "ext")
|
||||||
client := comms.NewInProcClient(codec.JSON)
|
client := comms.NewInProcClient(codec.JSON)
|
||||||
ds := docserver.New("/")
|
tf := &testjethre{client: ethereum.HTTPClient()}
|
||||||
tf := &testjethre{ds: ds}
|
|
||||||
repl := newJSRE(ethereum, assetPath, "", client, false, tf)
|
repl := newJSRE(ethereum, assetPath, "", client, false, tf)
|
||||||
tf.jsre = repl
|
tf.jsre = repl
|
||||||
return tmp, tf, ethereum
|
return tmp, tf, ethereum
|
||||||
|
@ -139,6 +139,11 @@ var (
|
|||||||
Name: "natspec",
|
Name: "natspec",
|
||||||
Usage: "Enable NatSpec confirmation notice",
|
Usage: "Enable NatSpec confirmation notice",
|
||||||
}
|
}
|
||||||
|
DocRootFlag = DirectoryFlag{
|
||||||
|
Name: "docroot",
|
||||||
|
Usage: "Document Root for HTTPClient file scheme",
|
||||||
|
Value: DirectoryString{common.HomeDir()},
|
||||||
|
}
|
||||||
CacheFlag = cli.IntFlag{
|
CacheFlag = cli.IntFlag{
|
||||||
Name: "cache",
|
Name: "cache",
|
||||||
Usage: "Megabytes of memory allocated to internal caching",
|
Usage: "Megabytes of memory allocated to internal caching",
|
||||||
@ -452,6 +457,7 @@ func MakeEthConfig(clientID, version string, ctx *cli.Context) *eth.Config {
|
|||||||
Olympic: ctx.GlobalBool(OlympicFlag.Name),
|
Olympic: ctx.GlobalBool(OlympicFlag.Name),
|
||||||
NAT: MakeNAT(ctx),
|
NAT: MakeNAT(ctx),
|
||||||
NatSpec: ctx.GlobalBool(NatspecEnabledFlag.Name),
|
NatSpec: ctx.GlobalBool(NatspecEnabledFlag.Name),
|
||||||
|
DocRoot: ctx.GlobalString(DocRootFlag.Name),
|
||||||
Discovery: !ctx.GlobalBool(NoDiscoverFlag.Name),
|
Discovery: !ctx.GlobalBool(NoDiscoverFlag.Name),
|
||||||
NodeKey: MakeNodeKey(ctx),
|
NodeKey: MakeNodeKey(ctx),
|
||||||
Shh: ctx.GlobalBool(WhisperEnabledFlag.Name),
|
Shh: ctx.GlobalBool(WhisperEnabledFlag.Name),
|
||||||
@ -616,7 +622,7 @@ func StartIPC(eth *eth.Ethereum, ctx *cli.Context) error {
|
|||||||
xeth := xeth.New(eth, fe)
|
xeth := xeth.New(eth, fe)
|
||||||
codec := codec.JSON
|
codec := codec.JSON
|
||||||
|
|
||||||
apis, err := api.ParseApiString(ctx.GlobalString(IPCApiFlag.Name), codec, xeth, eth, ctx.GlobalString(JSpathFlag.Name))
|
apis, err := api.ParseApiString(ctx.GlobalString(IPCApiFlag.Name), codec, xeth, eth)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -637,7 +643,7 @@ func StartRPC(eth *eth.Ethereum, ctx *cli.Context) error {
|
|||||||
xeth := xeth.New(eth, nil)
|
xeth := xeth.New(eth, nil)
|
||||||
codec := codec.JSON
|
codec := codec.JSON
|
||||||
|
|
||||||
apis, err := api.ParseApiString(ctx.GlobalString(RpcApiFlag.Name), codec, xeth, eth, ctx.GlobalString(JSpathFlag.Name))
|
apis, err := api.ParseApiString(ctx.GlobalString(RpcApiFlag.Name), codec, xeth, eth)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU Lesser General Public License
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package docserver
|
package httpclient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
@ -26,14 +26,14 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/crypto"
|
"github.com/ethereum/go-ethereum/crypto"
|
||||||
)
|
)
|
||||||
|
|
||||||
type DocServer struct {
|
type HTTPClient struct {
|
||||||
*http.Transport
|
*http.Transport
|
||||||
DocRoot string
|
DocRoot string
|
||||||
schemes []string
|
schemes []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(docRoot string) (self *DocServer) {
|
func New(docRoot string) (self *HTTPClient) {
|
||||||
self = &DocServer{
|
self = &HTTPClient{
|
||||||
Transport: &http.Transport{},
|
Transport: &http.Transport{},
|
||||||
DocRoot: docRoot,
|
DocRoot: docRoot,
|
||||||
schemes: []string{"file"},
|
schemes: []string{"file"},
|
||||||
@ -46,18 +46,18 @@ func New(docRoot string) (self *DocServer) {
|
|||||||
|
|
||||||
// A Client is higher-level than a RoundTripper (such as Transport) and additionally handles HTTP details such as cookies and redirects.
|
// A Client is higher-level than a RoundTripper (such as Transport) and additionally handles HTTP details such as cookies and redirects.
|
||||||
|
|
||||||
func (self *DocServer) Client() *http.Client {
|
func (self *HTTPClient) Client() *http.Client {
|
||||||
return &http.Client{
|
return &http.Client{
|
||||||
Transport: self,
|
Transport: self,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *DocServer) RegisterScheme(scheme string, rt http.RoundTripper) {
|
func (self *HTTPClient) RegisterScheme(scheme string, rt http.RoundTripper) {
|
||||||
self.schemes = append(self.schemes, scheme)
|
self.schemes = append(self.schemes, scheme)
|
||||||
self.RegisterProtocol(scheme, rt)
|
self.RegisterProtocol(scheme, rt)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *DocServer) HasScheme(scheme string) bool {
|
func (self *HTTPClient) HasScheme(scheme string) bool {
|
||||||
for _, s := range self.schemes {
|
for _, s := range self.schemes {
|
||||||
if s == scheme {
|
if s == scheme {
|
||||||
return true
|
return true
|
||||||
@ -66,43 +66,41 @@ func (self *DocServer) HasScheme(scheme string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *DocServer) GetAuthContent(uri string, hash common.Hash) (content []byte, err error) {
|
func (self *HTTPClient) GetAuthContent(uri string, hash common.Hash) ([]byte, error) {
|
||||||
// retrieve content
|
// retrieve content
|
||||||
content, err = self.Get(uri, "")
|
content, err := self.Get(uri, "")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// check hash to authenticate content
|
// check hash to authenticate content
|
||||||
chash := crypto.Sha3Hash(content)
|
chash := crypto.Sha3Hash(content)
|
||||||
if chash != hash {
|
if chash != hash {
|
||||||
content = nil
|
return nil, fmt.Errorf("content hash mismatch %x != %x (exp)", hash[:], chash[:])
|
||||||
err = fmt.Errorf("content hash mismatch %x != %x (exp)", hash[:], chash[:])
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
return content, nil
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get(uri, path) downloads the document at uri, if path is non-empty it
|
// Get(uri, path) downloads the document at uri, if path is non-empty it
|
||||||
// is interpreted as a filepath to which the contents are saved
|
// is interpreted as a filepath to which the contents are saved
|
||||||
func (self *DocServer) Get(uri, path string) (content []byte, err error) {
|
func (self *HTTPClient) Get(uri, path string) ([]byte, error) {
|
||||||
// retrieve content
|
// retrieve content
|
||||||
resp, err := self.Client().Get(uri)
|
resp, err := self.Client().Get(uri)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
defer func() {
|
defer func() {
|
||||||
if resp != nil {
|
if resp != nil {
|
||||||
resp.Body.Close()
|
resp.Body.Close()
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
if err != nil {
|
var content []byte
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
content, err = ioutil.ReadAll(resp.Body)
|
content, err = ioutil.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if resp.StatusCode/100 != 2 {
|
if resp.StatusCode/100 != 2 {
|
||||||
@ -112,9 +110,15 @@ func (self *DocServer) Get(uri, path string) (content []byte, err error) {
|
|||||||
if path != "" {
|
if path != "" {
|
||||||
var abspath string
|
var abspath string
|
||||||
abspath, err = filepath.Abs(path)
|
abspath, err = filepath.Abs(path)
|
||||||
ioutil.WriteFile(abspath, content, 0700)
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
err = ioutil.WriteFile(abspath, content, 0600)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
return content, nil
|
||||||
|
|
||||||
}
|
}
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU Lesser General Public License
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package docserver
|
package httpclient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
@ -28,19 +28,19 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestGetAuthContent(t *testing.T) {
|
func TestGetAuthContent(t *testing.T) {
|
||||||
dir, err := ioutil.TempDir("", "docserver-test")
|
dir, err := ioutil.TempDir("", "httpclient-test")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal("cannot create temporary directory:", err)
|
t.Fatal("cannot create temporary directory:", err)
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(dir)
|
defer os.RemoveAll(dir)
|
||||||
ds := New(dir)
|
client := New(dir)
|
||||||
|
|
||||||
text := "test"
|
text := "test"
|
||||||
hash := crypto.Sha3Hash([]byte(text))
|
hash := crypto.Sha3Hash([]byte(text))
|
||||||
if err := ioutil.WriteFile(path.Join(dir, "test.content"), []byte(text), os.ModePerm); err != nil {
|
if err := ioutil.WriteFile(path.Join(dir, "test.content"), []byte(text), os.ModePerm); err != nil {
|
||||||
t.Fatal("could not write test file", err)
|
t.Fatal("could not write test file", err)
|
||||||
}
|
}
|
||||||
content, err := ds.GetAuthContent("file:///test.content", hash)
|
content, err := client.GetAuthContent("file:///test.content", hash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("no error expected, got %v", err)
|
t.Errorf("no error expected, got %v", err)
|
||||||
}
|
}
|
||||||
@ -49,7 +49,7 @@ func TestGetAuthContent(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
hash = common.Hash{}
|
hash = common.Hash{}
|
||||||
content, err = ds.GetAuthContent("file:///test.content", hash)
|
content, err = client.GetAuthContent("file:///test.content", hash)
|
||||||
expected := "content hash mismatch 0000000000000000000000000000000000000000000000000000000000000000 != 9c22ff5f21f0b81b113e63f7db6da94fedef11b2119b4088b89664fb9a3cb658 (exp)"
|
expected := "content hash mismatch 0000000000000000000000000000000000000000000000000000000000000000 != 9c22ff5f21f0b81b113e63f7db6da94fedef11b2119b4088b89664fb9a3cb658 (exp)"
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Errorf("expected error, got nothing")
|
t.Errorf("expected error, got nothing")
|
||||||
@ -66,12 +66,12 @@ type rt struct{}
|
|||||||
func (rt) RoundTrip(req *http.Request) (resp *http.Response, err error) { return }
|
func (rt) RoundTrip(req *http.Request) (resp *http.Response, err error) { return }
|
||||||
|
|
||||||
func TestRegisterScheme(t *testing.T) {
|
func TestRegisterScheme(t *testing.T) {
|
||||||
ds := New("/tmp/")
|
client := New("/tmp/")
|
||||||
if ds.HasScheme("scheme") {
|
if client.HasScheme("scheme") {
|
||||||
t.Errorf("expected scheme not to be registered")
|
t.Errorf("expected scheme not to be registered")
|
||||||
}
|
}
|
||||||
ds.RegisterScheme("scheme", rt{})
|
client.RegisterScheme("scheme", rt{})
|
||||||
if !ds.HasScheme("scheme") {
|
if !client.HasScheme("scheme") {
|
||||||
t.Errorf("expected scheme to be registered")
|
t.Errorf("expected scheme to be registered")
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -23,7 +23,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/docserver"
|
"github.com/ethereum/go-ethereum/common/httpclient"
|
||||||
"github.com/ethereum/go-ethereum/common/registrar"
|
"github.com/ethereum/go-ethereum/common/registrar"
|
||||||
"github.com/ethereum/go-ethereum/crypto"
|
"github.com/ethereum/go-ethereum/crypto"
|
||||||
"github.com/ethereum/go-ethereum/xeth"
|
"github.com/ethereum/go-ethereum/xeth"
|
||||||
@ -43,7 +43,7 @@ type NatSpec struct {
|
|||||||
// the implementation is frontend friendly in that it always gives back
|
// the implementation is frontend friendly in that it always gives back
|
||||||
// a notice that is safe to display
|
// a notice that is safe to display
|
||||||
// :FIXME: the second return value is an error, which can be used to fine-tune bahaviour
|
// :FIXME: the second return value is an error, which can be used to fine-tune bahaviour
|
||||||
func GetNotice(xeth *xeth.XEth, tx string, http *docserver.DocServer) (notice string) {
|
func GetNotice(xeth *xeth.XEth, tx string, http *httpclient.HTTPClient) (notice string) {
|
||||||
ns, err := New(xeth, tx, http)
|
ns, err := New(xeth, tx, http)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if ns == nil {
|
if ns == nil {
|
||||||
@ -83,7 +83,7 @@ type contractInfo struct {
|
|||||||
DeveloperDoc json.RawMessage `json:"developerDoc"`
|
DeveloperDoc json.RawMessage `json:"developerDoc"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(xeth *xeth.XEth, jsontx string, http *docserver.DocServer) (self *NatSpec, err error) {
|
func New(xeth *xeth.XEth, jsontx string, http *httpclient.HTTPClient) (self *NatSpec, err error) {
|
||||||
|
|
||||||
// extract contract address from tx
|
// extract contract address from tx
|
||||||
var tx jsonTx
|
var tx jsonTx
|
||||||
@ -104,7 +104,7 @@ func New(xeth *xeth.XEth, jsontx string, http *docserver.DocServer) (self *NatSp
|
|||||||
}
|
}
|
||||||
|
|
||||||
// also called by admin.contractInfo.get
|
// also called by admin.contractInfo.get
|
||||||
func FetchDocsForContract(contractAddress string, xeth *xeth.XEth, ds *docserver.DocServer) (content []byte, err error) {
|
func FetchDocsForContract(contractAddress string, xeth *xeth.XEth, client *httpclient.HTTPClient) (content []byte, err error) {
|
||||||
// retrieve contract hash from state
|
// retrieve contract hash from state
|
||||||
codehex := xeth.CodeAt(contractAddress)
|
codehex := xeth.CodeAt(contractAddress)
|
||||||
codeb := xeth.CodeAtBytes(contractAddress)
|
codeb := xeth.CodeAtBytes(contractAddress)
|
||||||
@ -122,8 +122,8 @@ func FetchDocsForContract(contractAddress string, xeth *xeth.XEth, ds *docserver
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if ds.HasScheme("bzz") {
|
if client.HasScheme("bzz") {
|
||||||
content, err = ds.Get("bzz://"+hash.Hex()[2:], "")
|
content, err = client.Get("bzz://"+hash.Hex()[2:], "")
|
||||||
if err == nil { // non-fatal
|
if err == nil { // non-fatal
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -137,7 +137,7 @@ func FetchDocsForContract(contractAddress string, xeth *xeth.XEth, ds *docserver
|
|||||||
}
|
}
|
||||||
|
|
||||||
// get content via http client and authenticate content using hash
|
// get content via http client and authenticate content using hash
|
||||||
content, err = ds.GetAuthContent(uri, hash)
|
content, err = client.GetAuthContent(uri, hash)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -28,7 +28,7 @@ import (
|
|||||||
|
|
||||||
"github.com/ethereum/go-ethereum/accounts"
|
"github.com/ethereum/go-ethereum/accounts"
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/docserver"
|
"github.com/ethereum/go-ethereum/common/httpclient"
|
||||||
"github.com/ethereum/go-ethereum/common/registrar"
|
"github.com/ethereum/go-ethereum/common/registrar"
|
||||||
"github.com/ethereum/go-ethereum/core"
|
"github.com/ethereum/go-ethereum/core"
|
||||||
"github.com/ethereum/go-ethereum/crypto"
|
"github.com/ethereum/go-ethereum/crypto"
|
||||||
@ -113,8 +113,8 @@ func (self *testFrontend) UnlockAccount(acc []byte) bool {
|
|||||||
|
|
||||||
func (self *testFrontend) ConfirmTransaction(tx string) bool {
|
func (self *testFrontend) ConfirmTransaction(tx string) bool {
|
||||||
if self.wantNatSpec {
|
if self.wantNatSpec {
|
||||||
ds := docserver.New("/tmp/")
|
client := httpclient.New("/tmp/")
|
||||||
self.lastConfirm = GetNotice(self.xeth, tx, ds)
|
self.lastConfirm = GetNotice(self.xeth, tx, client)
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
@ -23,8 +23,6 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/kardianos/osext"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// MakeName creates a node name that follows the ethereum convention
|
// MakeName creates a node name that follows the ethereum convention
|
||||||
@ -65,48 +63,18 @@ func AbsolutePath(Datadir string, filename string) string {
|
|||||||
return filepath.Join(Datadir, filename)
|
return filepath.Join(Datadir, filename)
|
||||||
}
|
}
|
||||||
|
|
||||||
func DefaultAssetPath() string {
|
func HomeDir() (home string) {
|
||||||
var assetPath string
|
|
||||||
pwd, _ := os.Getwd()
|
|
||||||
srcdir := filepath.Join(os.Getenv("GOPATH"), "src", "github.com", "ethereum", "go-ethereum", "cmd", "mist")
|
|
||||||
|
|
||||||
// If the current working directory is the go-ethereum dir
|
|
||||||
// assume a debug build and use the source directory as
|
|
||||||
// asset directory.
|
|
||||||
if pwd == srcdir {
|
|
||||||
assetPath = filepath.Join(pwd, "assets")
|
|
||||||
} else {
|
|
||||||
switch runtime.GOOS {
|
|
||||||
case "darwin":
|
|
||||||
// Get Binary Directory
|
|
||||||
exedir, _ := osext.ExecutableFolder()
|
|
||||||
assetPath = filepath.Join(exedir, "..", "Resources")
|
|
||||||
case "linux":
|
|
||||||
assetPath = filepath.Join("usr", "share", "mist")
|
|
||||||
case "windows":
|
|
||||||
assetPath = filepath.Join(".", "assets")
|
|
||||||
default:
|
|
||||||
assetPath = "."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if the assetPath exists. If not, try the source directory
|
|
||||||
// This happens when binary is run from outside cmd/mist directory
|
|
||||||
if _, err := os.Stat(assetPath); os.IsNotExist(err) {
|
|
||||||
assetPath = filepath.Join(srcdir, "assets")
|
|
||||||
}
|
|
||||||
|
|
||||||
return assetPath
|
|
||||||
}
|
|
||||||
|
|
||||||
func DefaultDataDir() string {
|
|
||||||
// Try to place the data folder in the user's home dir
|
|
||||||
var home string
|
|
||||||
if usr, err := user.Current(); err == nil {
|
if usr, err := user.Current(); err == nil {
|
||||||
home = usr.HomeDir
|
home = usr.HomeDir
|
||||||
} else {
|
} else {
|
||||||
home = os.Getenv("HOME")
|
home = os.Getenv("HOME")
|
||||||
}
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func DefaultDataDir() string {
|
||||||
|
// Try to place the data folder in the user's home dir
|
||||||
|
home := HomeDir()
|
||||||
if home != "" {
|
if home != "" {
|
||||||
if runtime.GOOS == "darwin" {
|
if runtime.GOOS == "darwin" {
|
||||||
return filepath.Join(home, "Library", "Ethereum")
|
return filepath.Join(home, "Library", "Ethereum")
|
||||||
|
@ -35,6 +35,7 @@ import (
|
|||||||
"github.com/ethereum/go-ethereum/accounts"
|
"github.com/ethereum/go-ethereum/accounts"
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/compiler"
|
"github.com/ethereum/go-ethereum/common/compiler"
|
||||||
|
"github.com/ethereum/go-ethereum/common/httpclient"
|
||||||
"github.com/ethereum/go-ethereum/core"
|
"github.com/ethereum/go-ethereum/core"
|
||||||
"github.com/ethereum/go-ethereum/core/state"
|
"github.com/ethereum/go-ethereum/core/state"
|
||||||
"github.com/ethereum/go-ethereum/core/types"
|
"github.com/ethereum/go-ethereum/core/types"
|
||||||
@ -106,6 +107,7 @@ type Config struct {
|
|||||||
LogJSON string
|
LogJSON string
|
||||||
VmDebug bool
|
VmDebug bool
|
||||||
NatSpec bool
|
NatSpec bool
|
||||||
|
DocRoot string
|
||||||
AutoDAG bool
|
AutoDAG bool
|
||||||
PowTest bool
|
PowTest bool
|
||||||
ExtraData []byte
|
ExtraData []byte
|
||||||
@ -249,6 +251,8 @@ type Ethereum struct {
|
|||||||
GpobaseStepUp int
|
GpobaseStepUp int
|
||||||
GpobaseCorrectionFactor int
|
GpobaseCorrectionFactor int
|
||||||
|
|
||||||
|
httpclient *httpclient.HTTPClient
|
||||||
|
|
||||||
net *p2p.Server
|
net *p2p.Server
|
||||||
eventMux *event.TypeMux
|
eventMux *event.TypeMux
|
||||||
miner *miner.Miner
|
miner *miner.Miner
|
||||||
@ -400,6 +404,7 @@ func New(config *Config) (*Ethereum, error) {
|
|||||||
GpobaseStepDown: config.GpobaseStepDown,
|
GpobaseStepDown: config.GpobaseStepDown,
|
||||||
GpobaseStepUp: config.GpobaseStepUp,
|
GpobaseStepUp: config.GpobaseStepUp,
|
||||||
GpobaseCorrectionFactor: config.GpobaseCorrectionFactor,
|
GpobaseCorrectionFactor: config.GpobaseCorrectionFactor,
|
||||||
|
httpclient: httpclient.New(config.DocRoot),
|
||||||
}
|
}
|
||||||
|
|
||||||
if config.PowTest {
|
if config.PowTest {
|
||||||
@ -702,6 +707,12 @@ func (self *Ethereum) StopAutoDAG() {
|
|||||||
glog.V(logger.Info).Infof("Automatic pregeneration of ethash DAG OFF (ethash dir: %s)", ethash.DefaultDir)
|
glog.V(logger.Info).Infof("Automatic pregeneration of ethash DAG OFF (ethash dir: %s)", ethash.DefaultDir)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// HTTPClient returns the light http client used for fetching offchain docs
|
||||||
|
// (natspec, source for verification)
|
||||||
|
func (self *Ethereum) HTTPClient() *httpclient.HTTPClient {
|
||||||
|
return self.httpclient
|
||||||
|
}
|
||||||
|
|
||||||
func (self *Ethereum) Solc() (*compiler.Solidity, error) {
|
func (self *Ethereum) Solc() (*compiler.Solidity, error) {
|
||||||
var err error
|
var err error
|
||||||
if self.solc == nil {
|
if self.solc == nil {
|
||||||
|
@ -25,7 +25,6 @@ import (
|
|||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/compiler"
|
"github.com/ethereum/go-ethereum/common/compiler"
|
||||||
"github.com/ethereum/go-ethereum/common/docserver"
|
|
||||||
"github.com/ethereum/go-ethereum/common/natspec"
|
"github.com/ethereum/go-ethereum/common/natspec"
|
||||||
"github.com/ethereum/go-ethereum/common/registrar"
|
"github.com/ethereum/go-ethereum/common/registrar"
|
||||||
"github.com/ethereum/go-ethereum/core"
|
"github.com/ethereum/go-ethereum/core"
|
||||||
@ -84,19 +83,15 @@ type adminApi struct {
|
|||||||
ethereum *eth.Ethereum
|
ethereum *eth.Ethereum
|
||||||
codec codec.Codec
|
codec codec.Codec
|
||||||
coder codec.ApiCoder
|
coder codec.ApiCoder
|
||||||
docRoot string
|
|
||||||
ds *docserver.DocServer
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// create a new admin api instance
|
// create a new admin api instance
|
||||||
func NewAdminApi(xeth *xeth.XEth, ethereum *eth.Ethereum, codec codec.Codec, docRoot string) *adminApi {
|
func NewAdminApi(xeth *xeth.XEth, ethereum *eth.Ethereum, codec codec.Codec) *adminApi {
|
||||||
return &adminApi{
|
return &adminApi{
|
||||||
xeth: xeth,
|
xeth: xeth,
|
||||||
ethereum: ethereum,
|
ethereum: ethereum,
|
||||||
codec: codec,
|
codec: codec,
|
||||||
coder: codec.New(nil),
|
coder: codec.New(nil),
|
||||||
docRoot: docRoot,
|
|
||||||
ds: docserver.New(docRoot),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -258,7 +253,7 @@ func (self *adminApi) StartRPC(req *shared.Request) (interface{}, error) {
|
|||||||
CorsDomain: args.CorsDomain,
|
CorsDomain: args.CorsDomain,
|
||||||
}
|
}
|
||||||
|
|
||||||
apis, err := ParseApiString(args.Apis, self.codec, self.xeth, self.ethereum, self.docRoot)
|
apis, err := ParseApiString(args.Apis, self.codec, self.xeth, self.ethereum)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
@ -439,7 +434,7 @@ func (self *adminApi) GetContractInfo(req *shared.Request) (interface{}, error)
|
|||||||
return nil, shared.NewDecodeParamError(err.Error())
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
infoDoc, err := natspec.FetchDocsForContract(args.Contract, self.xeth, self.ds)
|
infoDoc, err := natspec.FetchDocsForContract(args.Contract, self.xeth, self.ethereum.HTTPClient())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -459,7 +454,7 @@ func (self *adminApi) HttpGet(req *shared.Request) (interface{}, error) {
|
|||||||
return nil, shared.NewDecodeParamError(err.Error())
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
resp, err := self.ds.Get(args.Uri, args.Path)
|
resp, err := self.ethereum.HTTPClient().Get(args.Uri, args.Path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func TestParseApiString(t *testing.T) {
|
func TestParseApiString(t *testing.T) {
|
||||||
apis, err := ParseApiString("", codec.JSON, nil, nil, "")
|
apis, err := ParseApiString("", codec.JSON, nil, nil)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Errorf("Expected an err from parsing empty API string but got nil")
|
t.Errorf("Expected an err from parsing empty API string but got nil")
|
||||||
}
|
}
|
||||||
@ -39,7 +39,7 @@ func TestParseApiString(t *testing.T) {
|
|||||||
t.Errorf("Expected 0 apis from empty API string")
|
t.Errorf("Expected 0 apis from empty API string")
|
||||||
}
|
}
|
||||||
|
|
||||||
apis, err = ParseApiString("eth", codec.JSON, nil, nil, "")
|
apis, err = ParseApiString("eth", codec.JSON, nil, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("Expected nil err from parsing empty API string but got %v", err)
|
t.Errorf("Expected nil err from parsing empty API string but got %v", err)
|
||||||
}
|
}
|
||||||
@ -48,7 +48,7 @@ func TestParseApiString(t *testing.T) {
|
|||||||
t.Errorf("Expected 1 apis but got %d - %v", apis, apis)
|
t.Errorf("Expected 1 apis but got %d - %v", apis, apis)
|
||||||
}
|
}
|
||||||
|
|
||||||
apis, err = ParseApiString("eth,eth", codec.JSON, nil, nil, "")
|
apis, err = ParseApiString("eth,eth", codec.JSON, nil, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("Expected nil err from parsing empty API string but got \"%v\"", err)
|
t.Errorf("Expected nil err from parsing empty API string but got \"%v\"", err)
|
||||||
}
|
}
|
||||||
@ -57,7 +57,7 @@ func TestParseApiString(t *testing.T) {
|
|||||||
t.Errorf("Expected 2 apis but got %d - %v", apis, apis)
|
t.Errorf("Expected 2 apis but got %d - %v", apis, apis)
|
||||||
}
|
}
|
||||||
|
|
||||||
apis, err = ParseApiString("eth,invalid", codec.JSON, nil, nil, "")
|
apis, err = ParseApiString("eth,invalid", codec.JSON, nil, nil)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Errorf("Expected an err but got no err")
|
t.Errorf("Expected an err but got no err")
|
||||||
}
|
}
|
||||||
|
@ -24,6 +24,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
|
"github.com/ethereum/go-ethereum/common/natspec"
|
||||||
"github.com/ethereum/go-ethereum/eth"
|
"github.com/ethereum/go-ethereum/eth"
|
||||||
"github.com/ethereum/go-ethereum/rpc/codec"
|
"github.com/ethereum/go-ethereum/rpc/codec"
|
||||||
"github.com/ethereum/go-ethereum/rpc/shared"
|
"github.com/ethereum/go-ethereum/rpc/shared"
|
||||||
@ -67,6 +68,7 @@ var (
|
|||||||
"eth_getUncleCountByBlockNumber": (*ethApi).GetUncleCountByBlockNumber,
|
"eth_getUncleCountByBlockNumber": (*ethApi).GetUncleCountByBlockNumber,
|
||||||
"eth_getData": (*ethApi).GetData,
|
"eth_getData": (*ethApi).GetData,
|
||||||
"eth_getCode": (*ethApi).GetData,
|
"eth_getCode": (*ethApi).GetData,
|
||||||
|
"eth_getNatSpec": (*ethApi).GetNatSpec,
|
||||||
"eth_sign": (*ethApi).Sign,
|
"eth_sign": (*ethApi).Sign,
|
||||||
"eth_sendRawTransaction": (*ethApi).SendRawTransaction,
|
"eth_sendRawTransaction": (*ethApi).SendRawTransaction,
|
||||||
"eth_sendTransaction": (*ethApi).SendTransaction,
|
"eth_sendTransaction": (*ethApi).SendTransaction,
|
||||||
@ -322,6 +324,18 @@ func (self *ethApi) SendTransaction(req *shared.Request) (interface{}, error) {
|
|||||||
return v, nil
|
return v, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (self *ethApi) GetNatSpec(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(NewTxArgs)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
var jsontx = fmt.Sprintf(`{"params":[{"to":"%s","data": "%s"}]}`, args.To, args.Data)
|
||||||
|
notice := natspec.GetNotice(self.xeth, jsontx, self.ethereum.HTTPClient())
|
||||||
|
|
||||||
|
return notice, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (self *ethApi) EstimateGas(req *shared.Request) (interface{}, error) {
|
func (self *ethApi) EstimateGas(req *shared.Request) (interface{}, error) {
|
||||||
_, gas, err := self.doCall(req.Params)
|
_, gas, err := self.doCall(req.Params)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -35,6 +35,12 @@ web3._extend({
|
|||||||
call: 'eth_resend',
|
call: 'eth_resend',
|
||||||
params: 3,
|
params: 3,
|
||||||
inputFormatter: [web3._extend.formatters.inputTransactionFormatter, web3._extend.utils.fromDecimal, web3._extend.utils.fromDecimal]
|
inputFormatter: [web3._extend.formatters.inputTransactionFormatter, web3._extend.utils.fromDecimal, web3._extend.utils.fromDecimal]
|
||||||
|
}),
|
||||||
|
new web3._extend.Method({
|
||||||
|
name: 'getNatSpec',
|
||||||
|
call: 'eth_getNatSpec',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3._extend.formatters.inputTransactionFormatter]
|
||||||
})
|
})
|
||||||
],
|
],
|
||||||
properties:
|
properties:
|
||||||
|
@ -89,6 +89,7 @@ var (
|
|||||||
"getBlockTransactionCount",
|
"getBlockTransactionCount",
|
||||||
"getBlockUncleCount",
|
"getBlockUncleCount",
|
||||||
"getCode",
|
"getCode",
|
||||||
|
"getNatSpec",
|
||||||
"getCompilers",
|
"getCompilers",
|
||||||
"gasPrice",
|
"gasPrice",
|
||||||
"getStorageAt",
|
"getStorageAt",
|
||||||
@ -153,7 +154,7 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// Parse a comma separated API string to individual api's
|
// Parse a comma separated API string to individual api's
|
||||||
func ParseApiString(apistr string, codec codec.Codec, xeth *xeth.XEth, eth *eth.Ethereum, docRoot string) ([]shared.EthereumApi, error) {
|
func ParseApiString(apistr string, codec codec.Codec, xeth *xeth.XEth, eth *eth.Ethereum) ([]shared.EthereumApi, error) {
|
||||||
if len(strings.TrimSpace(apistr)) == 0 {
|
if len(strings.TrimSpace(apistr)) == 0 {
|
||||||
return nil, fmt.Errorf("Empty apistr provided")
|
return nil, fmt.Errorf("Empty apistr provided")
|
||||||
}
|
}
|
||||||
@ -164,7 +165,7 @@ func ParseApiString(apistr string, codec codec.Codec, xeth *xeth.XEth, eth *eth.
|
|||||||
for i, name := range names {
|
for i, name := range names {
|
||||||
switch strings.ToLower(strings.TrimSpace(name)) {
|
switch strings.ToLower(strings.TrimSpace(name)) {
|
||||||
case shared.AdminApiName:
|
case shared.AdminApiName:
|
||||||
apis[i] = NewAdminApi(xeth, eth, codec, docRoot)
|
apis[i] = NewAdminApi(xeth, eth, codec)
|
||||||
case shared.DebugApiName:
|
case shared.DebugApiName:
|
||||||
apis[i] = NewDebugApi(xeth, eth, codec)
|
apis[i] = NewDebugApi(xeth, eth, codec)
|
||||||
case shared.DbApiName:
|
case shared.DbApiName:
|
||||||
|
Loading…
Reference in New Issue
Block a user