From 35e86bd1f18b6c4a14f7cce45641f1767b7fe277 Mon Sep 17 00:00:00 2001 From: Jeff Thompson Date: Thu, 23 Nov 2023 09:35:38 +0100 Subject: [PATCH 1/3] chore: In gnoclient.Client, add methods Render and QEval. Signed-off-by: Jeff Thompson --- gno.land/pkg/gnoclient/client_queries.go | 41 +++++++++++++++++++++++- 1 file changed, 40 insertions(+), 1 deletion(-) diff --git a/gno.land/pkg/gnoclient/client_queries.go b/gno.land/pkg/gnoclient/client_queries.go index b7c51208238..b20000fdd19 100644 --- a/gno.land/pkg/gnoclient/client_queries.go +++ b/gno.land/pkg/gnoclient/client_queries.go @@ -67,9 +67,48 @@ func (c Client) QueryAppVersion() (string, *ctypes.ResultABCIQuery, error) { qres, err := c.RPCClient.ABCIQuery(path, data) if err != nil { - return "", nil, errors.Wrap(err, "query account") + return "", nil, errors.Wrap(err, "query app version") } version := string(qres.Response.Value) return version, qres, nil } + +// Render calls the Render function for pkgPath with optional args. The pkgPath should +// include the prefix like "gno.land/". This is similar to using a browser URL +// /: where doesn't have the prefix like "gno.land/". +func (c Client) Render(pkgPath string, args string) (string, *ctypes.ResultABCIQuery, error) { + if err := c.validateRPCClient(); err != nil { + return "", nil, err + } + + path := "vm/qrender" + data := []byte(fmt.Sprintf("%s\n%s", pkgPath, args)) + + qres, err := c.RPCClient.ABCIQuery(path, data) + if err != nil { + return "", nil, errors.Wrap(err, "query render") + } + + return string(qres.Response.Data), qres, nil +} + +// QEval evaluates the given expression with the realm code at pkgPath. The pkgPath should +// include the prefix like "gno.land/". The expression is usually a function call like +// "GetBoardIDFromName(\"testboard\")". The return value is a typed expression like +// "(1 gno.land/r/demo/boards.BoardID)\n(true bool)". +func (c Client) QEval(pkgPath string, expression string) (string, *ctypes.ResultABCIQuery, error) { + if err := c.validateRPCClient(); err != nil { + return "", nil, err + } + + path := "vm/qeval" + data := []byte(fmt.Sprintf("%s\n%s", pkgPath, expression)) + + qres, err := c.RPCClient.ABCIQuery(path, data) + if err != nil { + return "", nil, errors.Wrap(err, "query qeval") + } + + return string(qres.Response.Data), qres, nil +} From dcb7d4398f57453f42fd4594e75274338791eacc Mon Sep 17 00:00:00 2001 From: Jeff Thompson Date: Thu, 23 Nov 2023 09:37:39 +0100 Subject: [PATCH 2/3] chore: In gnoclient.Signer.Validate, sign a blank transaction to verify if the password unlocks the account Signed-off-by: Jeff Thompson --- gno.land/pkg/gnoclient/signer.go | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/gno.land/pkg/gnoclient/signer.go b/gno.land/pkg/gnoclient/signer.go index 098af742e4d..5eba124f2a8 100644 --- a/gno.land/pkg/gnoclient/signer.go +++ b/gno.land/pkg/gnoclient/signer.go @@ -3,6 +3,7 @@ package gnoclient import ( "fmt" + "github.com/gnolang/gno/gno.land/pkg/sdk/vm" "github.com/gnolang/gno/tm2/pkg/crypto/keys" "github.com/gnolang/gno/tm2/pkg/errors" "github.com/gnolang/gno/tm2/pkg/std" @@ -33,7 +34,20 @@ func (s SignerFromKeybase) Validate() error { return err } - // TODO: Also verify if the password unlocks the account. + // To verify if the password unlocks the account, sign a blank transaction. + msg := vm.MsgCall{ + Caller: s.Info().GetAddress(), + } + signCfg := SignCfg{ + UnsignedTX: std.Tx{ + Msgs: []std.Msg{msg}, + Fee: std.NewFee(0, std.NewCoin("ugnot", 1000000)), + }, + } + if _, err = s.Sign(signCfg); err != nil { + return err + } + return nil } From 8abebbe2b20523b1891e9ef872daecb930085080 Mon Sep 17 00:00:00 2001 From: Jeff Thompson Date: Thu, 23 Nov 2023 09:39:46 +0100 Subject: [PATCH 3/3] chore: In gnoclient.Client.QueryAccount, param addr should be crypto.Address, not string Signed-off-by: Jeff Thompson --- gno.land/pkg/gnoclient/client_queries.go | 8 ++++++-- gno.land/pkg/gnoclient/client_txs.go | 2 +- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/gno.land/pkg/gnoclient/client_queries.go b/gno.land/pkg/gnoclient/client_queries.go index b20000fdd19..ce03305d503 100644 --- a/gno.land/pkg/gnoclient/client_queries.go +++ b/gno.land/pkg/gnoclient/client_queries.go @@ -6,6 +6,7 @@ import ( "github.com/gnolang/gno/tm2/pkg/amino" rpcclient "github.com/gnolang/gno/tm2/pkg/bft/rpc/client" ctypes "github.com/gnolang/gno/tm2/pkg/bft/rpc/core/types" + "github.com/gnolang/gno/tm2/pkg/crypto" "github.com/gnolang/gno/tm2/pkg/errors" "github.com/gnolang/gno/tm2/pkg/std" ) @@ -35,18 +36,21 @@ func (c Client) Query(cfg QueryCfg) (*ctypes.ResultABCIQuery, error) { } // QueryAccount retrieves account information for a given address. -func (c Client) QueryAccount(addr string) (*std.BaseAccount, *ctypes.ResultABCIQuery, error) { +func (c Client) QueryAccount(addr crypto.Address) (*std.BaseAccount, *ctypes.ResultABCIQuery, error) { if err := c.validateRPCClient(); err != nil { return nil, nil, err } - path := fmt.Sprintf("auth/accounts/%s", addr) + path := fmt.Sprintf("auth/accounts/%s", crypto.AddressToBech32(addr)) data := []byte{} qres, err := c.RPCClient.ABCIQuery(path, data) if err != nil { return nil, nil, errors.Wrap(err, "query account") } + if qres.Response.Data == nil || len(qres.Response.Data) == 0 || string(qres.Response.Data) == "null" { + return nil, nil, std.ErrUnknownAddress("unknown address: " + crypto.AddressToBech32(addr)) + } var qret struct{ BaseAccount std.BaseAccount } err = amino.UnmarshalJSON(qres.Response.Data, &qret) diff --git a/gno.land/pkg/gnoclient/client_txs.go b/gno.land/pkg/gnoclient/client_txs.go index cb779891e16..ab475154fad 100644 --- a/gno.land/pkg/gnoclient/client_txs.go +++ b/gno.land/pkg/gnoclient/client_txs.go @@ -86,7 +86,7 @@ func (c Client) signAndBroadcastTxCommit(tx std.Tx, accountNumber, sequenceNumbe caller := c.Signer.Info().GetAddress() if sequenceNumber == 0 || accountNumber == 0 { - account, _, err := c.QueryAccount(caller.String()) + account, _, err := c.QueryAccount(caller) if err != nil { return nil, errors.Wrap(err, "query account") }