feat: ens register api (#2473)

This commit is contained in:
Anthony Laibe 2022-01-14 12:17:31 +01:00 committed by GitHub
parent f5b8774f51
commit 22bb09a94f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 12130 additions and 73 deletions

View File

@ -357,7 +357,7 @@ func (b *StatusNode) browsersService() *browsers.Service {
func (b *StatusNode) ensService() *ens.Service { func (b *StatusNode) ensService() *ens.Service {
if b.ensSrvc == nil { if b.ensSrvc == nil {
b.ensSrvc = ens.NewService(b.rpcClient) b.ensSrvc = ens.NewService(b.rpcClient, b.gethAccountManager, b.rpcFiltersSrvc, b.config)
} }
return b.ensSrvc return b.ensSrvc
} }

View File

@ -5,6 +5,8 @@ import (
"encoding/binary" "encoding/binary"
"encoding/hex" "encoding/hex"
"fmt" "fmt"
"math/big"
"strings"
"github.com/ipfs/go-cid" "github.com/ipfs/go-cid"
"github.com/multiformats/go-multibase" "github.com/multiformats/go-multibase"
@ -12,17 +14,30 @@ import (
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/wealdtech/go-multicodec" "github.com/wealdtech/go-multicodec"
"github.com/ethereum/go-ethereum"
"github.com/ethereum/go-ethereum/accounts/abi"
"github.com/ethereum/go-ethereum/accounts/abi/bind" "github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common"
"github.com/status-im/status-go/eth-node/crypto" ethTypes "github.com/ethereum/go-ethereum/core/types"
"github.com/status-im/status-go/account"
"github.com/status-im/status-go/eth-node/types"
"github.com/status-im/status-go/params"
"github.com/status-im/status-go/rpc" "github.com/status-im/status-go/rpc"
"github.com/status-im/status-go/services/ens/erc20"
"github.com/status-im/status-go/services/ens/registrar"
"github.com/status-im/status-go/services/ens/resolver"
"github.com/status-im/status-go/services/rpcfilters"
"github.com/status-im/status-go/transactions"
) )
func NewAPI(rpcClient *rpc.Client) *API { func NewAPI(rpcClient *rpc.Client, accountsManager *account.GethManager, rpcFiltersSrvc *rpcfilters.Service, config *params.NodeConfig) *API {
return &API{ return &API{
contractMaker: &contractMaker{ contractMaker: &contractMaker{
rpcClient: rpcClient, RPCClient: rpcClient,
}, },
accountsManager: accountsManager,
rpcFiltersSrvc: rpcFiltersSrvc,
config: config,
} }
} }
@ -33,7 +48,10 @@ type uri struct {
} }
type API struct { type API struct {
contractMaker *contractMaker contractMaker *contractMaker
accountsManager *account.GethManager
rpcFiltersSrvc *rpcfilters.Service
config *params.NodeConfig
} }
func (api *API) Resolver(ctx context.Context, chainID uint64, username string) (*common.Address, error) { func (api *API) Resolver(ctx context.Context, chainID uint64, username string) (*common.Address, error) {
@ -151,16 +169,13 @@ func (api *API) AddressOf(ctx context.Context, chainID uint64, username string)
} }
func (api *API) ExpireAt(ctx context.Context, chainID uint64, username string) (string, error) { func (api *API) ExpireAt(ctx context.Context, chainID uint64, username string) (string, error) {
usernameHashed := crypto.Keccak256([]byte(username))
var label [32]byte
copy(label[:], usernameHashed)
registrar, err := api.contractMaker.newUsernameRegistrar(chainID) registrar, err := api.contractMaker.newUsernameRegistrar(chainID)
if err != nil { if err != nil {
return "", err return "", err
} }
callOpts := &bind.CallOpts{Context: ctx, Pending: false} callOpts := &bind.CallOpts{Context: ctx, Pending: false}
expTime, err := registrar.GetExpirationTime(callOpts, label) expTime, err := registrar.GetExpirationTime(callOpts, usernameToLabel(username))
if err != nil { if err != nil {
return "", err return "", err
} }
@ -183,66 +198,202 @@ func (api *API) Price(ctx context.Context, chainID uint64) (string, error) {
return fmt.Sprintf("%x", price), nil return fmt.Sprintf("%x", price), nil
} }
// TODO: implement once the send tx as been refactored func (api *API) getSigner(chainID uint64, from types.Address, password string) bind.SignerFn {
// func (api *API) Release(ctx context.Context, chainID uint64, from string, gasPrice *big.Int, gasLimit uint64, password string, username string) (string, error) { return func(addr common.Address, tx *ethTypes.Transaction) (*ethTypes.Transaction, error) {
// err := validateENSUsername(username) selectedAccount, err := api.accountsManager.VerifyAccountPassword(api.config.KeyStoreDir, from.Hex(), password)
// if err != nil { if err != nil {
// return "", err return nil, err
// } }
s := ethTypes.NewLondonSigner(new(big.Int).SetUint64(chainID))
return ethTypes.SignTx(tx, s, selectedAccount.PrivateKey)
}
}
// registrar, err := api.contractMaker.newUsernameRegistrar(chainID) func (api *API) Release(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, password string, username string) (string, error) {
// if err != nil { registrar, err := api.contractMaker.newUsernameRegistrar(chainID)
// return "", err if err != nil {
// } return "", err
}
// txOpts := &bind.TransactOpts{ txOpts := txArgs.ToTransactOpts(api.getSigner(chainID, txArgs.From, password))
// From: common.HexToAddress(from), tx, err := registrar.Release(txOpts, usernameToLabel(username))
// Signer: func(addr common.Address, tx *types.Transaction) (*types.Transaction, error) { if err != nil {
// // return types.SignTx(tx, types.NewLondonSigner(chainID), selectedAccount.AccountKey.PrivateKey) return "", err
// return nil, nil }
// },
// GasPrice: gasPrice,
// GasLimit: gasLimit,
// }
// tx, err := registrar.Release(txOpts, nameHash(username))
// if err != nil {
// return "", err
// }
// return tx.Hash().String(), nil
// }
// func (api *API) Register(ctx context.Context, chainID uint64, from string, gasPrice *big.Int, gasLimit uint64, password string, username string, x [32]byte, y [32]byte) (string, error) { go api.rpcFiltersSrvc.TriggerTransactionSentToUpstreamEvent(types.Hash(tx.Hash()))
// err := validateENSUsername(username) return tx.Hash().String(), nil
// if err != nil { }
// return "", err
// }
// registrar, err := api.contractMaker.newUsernameRegistrar(chainID) func (api *API) ReleaseEstimate(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, username string) (uint64, error) {
// if err != nil { registrarABI, err := abi.JSON(strings.NewReader(registrar.UsernameRegistrarABI))
// return "", err if err != nil {
// } return 0, err
}
// txOpts := &bind.TransactOpts{ data, err := registrarABI.Pack("release", nameHash(username))
// From: common.HexToAddress(from), if err != nil {
// Signer: func(addr common.Address, tx *types.Transaction) (*types.Transaction, error) { return 0, err
// // return types.SignTx(tx, types.NewLondonSigner(chainID), selectedAccount.AccountKey.PrivateKey) }
// return nil, nil
// }, ethClient, err := api.contractMaker.RPCClient.EthClient(chainID)
// GasPrice: gasPrice, if err != nil {
// GasLimit: gasLimit, return 0, err
// } }
// tx, err := registrar.Register(
// txOpts, registrarAddress := usernameRegistrarsByChainID[chainID]
// nameHash(username), return ethClient.EstimateGas(ctx, ethereum.CallMsg{
// common.HexToAddress(from), From: common.Address(txArgs.From),
// x, To: &registrarAddress,
// y, Value: big.NewInt(0),
// ) Data: data,
// if err != nil { })
// return "", err }
// }
// return tx.Hash().String(), nil func (api *API) Register(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, password string, username string, pubkey string) (string, error) {
// } snt, err := api.contractMaker.newSNT(chainID)
if err != nil {
return "", err
}
priceHex, err := api.Price(ctx, chainID)
if err != nil {
return "", err
}
price := new(big.Int)
price.SetString(priceHex, 16)
registrarABI, err := abi.JSON(strings.NewReader(registrar.UsernameRegistrarABI))
if err != nil {
return "", err
}
x, y := extractCoordinates(pubkey)
extraData, err := registrarABI.Pack("register", usernameToLabel(username), common.Address(txArgs.From), x, y)
if err != nil {
return "", err
}
txOpts := txArgs.ToTransactOpts(api.getSigner(chainID, txArgs.From, password))
tx, err := snt.ApproveAndCall(
txOpts,
usernameRegistrarsByChainID[chainID],
price,
extraData,
)
if err != nil {
return "", err
}
go api.rpcFiltersSrvc.TriggerTransactionSentToUpstreamEvent(types.Hash(tx.Hash()))
return tx.Hash().String(), nil
}
func (api *API) RegisterEstimate(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, username string, pubkey string) (uint64, error) {
priceHex, err := api.Price(ctx, chainID)
if err != nil {
return 0, err
}
price := new(big.Int)
price.SetString(priceHex, 16)
registrarABI, err := abi.JSON(strings.NewReader(registrar.UsernameRegistrarABI))
if err != nil {
return 0, err
}
x, y := extractCoordinates(pubkey)
extraData, err := registrarABI.Pack("register", usernameToLabel(username), common.Address(txArgs.From), x, y)
if err != nil {
return 0, err
}
sntABI, err := abi.JSON(strings.NewReader(erc20.SNTABI))
if err != nil {
return 0, err
}
data, err := sntABI.Pack("approveAndCall", usernameRegistrarsByChainID[chainID], price, extraData)
if err != nil {
return 0, err
}
ethClient, err := api.contractMaker.RPCClient.EthClient(chainID)
if err != nil {
return 0, err
}
contractAddress := sntByChainID[chainID]
return ethClient.EstimateGas(ctx, ethereum.CallMsg{
From: common.Address(txArgs.From),
To: &contractAddress,
Value: big.NewInt(0),
Data: data,
})
}
func (api *API) SetPubKey(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, password string, username string, pubkey string) (string, error) {
err := validateENSUsername(username)
if err != nil {
return "", err
}
resolverAddress, err := api.Resolver(ctx, chainID, username)
if err != nil {
return "", err
}
resolver, err := api.contractMaker.newPublicResolver(chainID, resolverAddress)
if err != nil {
return "", err
}
x, y := extractCoordinates(pubkey)
txOpts := txArgs.ToTransactOpts(api.getSigner(chainID, txArgs.From, password))
tx, err := resolver.SetPubkey(txOpts, nameHash(username), x, y)
if err != nil {
return "", err
}
go api.rpcFiltersSrvc.TriggerTransactionSentToUpstreamEvent(types.Hash(tx.Hash()))
return tx.Hash().String(), nil
}
func (api *API) SetPubKeyEstimate(ctx context.Context, chainID uint64, txArgs transactions.SendTxArgs, username string, pubkey string) (uint64, error) {
err := validateENSUsername(username)
if err != nil {
return 0, err
}
x, y := extractCoordinates(pubkey)
resolverABI, err := abi.JSON(strings.NewReader(resolver.PublicResolverABI))
if err != nil {
return 0, err
}
data, err := resolverABI.Pack("setPubkey", nameHash(username), x, y)
if err != nil {
return 0, err
}
ethClient, err := api.contractMaker.RPCClient.EthClient(chainID)
if err != nil {
return 0, err
}
resolverAddress, err := api.Resolver(ctx, chainID, username)
if err != nil {
return 0, err
}
return ethClient.EstimateGas(ctx, ethereum.CallMsg{
From: common.Address(txArgs.From),
To: resolverAddress,
Value: big.NewInt(0),
Data: data,
})
}
func (api *API) ResourceURL(ctx context.Context, chainID uint64, username string) (*uri, error) { func (api *API) ResourceURL(ctx context.Context, chainID uint64, username string) (*uri, error) {
scheme := "https" scheme := "https"

View File

@ -54,7 +54,7 @@ func setupTestAPI(t *testing.T) (*API, func()) {
utils.Init() utils.Init()
require.NoError(t, utils.ImportTestAccount(keyStoreDir, utils.GetAccount1PKFile())) require.NoError(t, utils.ImportTestAccount(keyStoreDir, utils.GetAccount1PKFile()))
return NewAPI(rpcClient), cancel return NewAPI(rpcClient, nil, nil, nil), cancel
} }
func TestResolver(t *testing.T) { func TestResolver(t *testing.T) {

View File

@ -5,6 +5,7 @@ import (
"github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common"
"github.com/status-im/status-go/rpc" "github.com/status-im/status-go/rpc"
"github.com/status-im/status-go/services/ens/erc20"
"github.com/status-im/status-go/services/ens/registrar" "github.com/status-im/status-go/services/ens/registrar"
"github.com/status-im/status-go/services/ens/resolver" "github.com/status-im/status-go/services/ens/resolver"
) )
@ -21,8 +22,13 @@ var usernameRegistrarsByChainID = map[uint64]common.Address{
3: common.HexToAddress("0xdaae165beb8c06e0b7613168138ebba774aff071"), // ropsten 3: common.HexToAddress("0xdaae165beb8c06e0b7613168138ebba774aff071"), // ropsten
} }
var sntByChainID = map[uint64]common.Address{
1: common.HexToAddress("0x744d70fdbe2ba4cf95131626614a1763df805b9e"), // mainnet
3: common.HexToAddress("0xc55cf4b03948d7ebc8b9e8bad92643703811d162"), // ropsten
}
type contractMaker struct { type contractMaker struct {
rpcClient *rpc.Client RPCClient *rpc.Client
} }
func (c *contractMaker) newRegistry(chainID uint64) (*resolver.ENSRegistryWithFallback, error) { func (c *contractMaker) newRegistry(chainID uint64) (*resolver.ENSRegistryWithFallback, error) {
@ -30,7 +36,7 @@ func (c *contractMaker) newRegistry(chainID uint64) (*resolver.ENSRegistryWithFa
return nil, errorNotAvailableOnChainID return nil, errorNotAvailableOnChainID
} }
backend, err := c.rpcClient.EthClient(chainID) backend, err := c.RPCClient.EthClient(chainID)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -42,7 +48,7 @@ func (c *contractMaker) newRegistry(chainID uint64) (*resolver.ENSRegistryWithFa
} }
func (c *contractMaker) newPublicResolver(chainID uint64, resolverAddress *common.Address) (*resolver.PublicResolver, error) { func (c *contractMaker) newPublicResolver(chainID uint64, resolverAddress *common.Address) (*resolver.PublicResolver, error) {
backend, err := c.rpcClient.EthClient(chainID) backend, err := c.RPCClient.EthClient(chainID)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -55,7 +61,7 @@ func (c *contractMaker) newUsernameRegistrar(chainID uint64) (*registrar.Usernam
return nil, errorNotAvailableOnChainID return nil, errorNotAvailableOnChainID
} }
backend, err := c.rpcClient.EthClient(chainID) backend, err := c.RPCClient.EthClient(chainID)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -65,3 +71,16 @@ func (c *contractMaker) newUsernameRegistrar(chainID uint64) (*registrar.Usernam
backend, backend,
) )
} }
func (c *contractMaker) newSNT(chainID uint64) (*erc20.SNT, error) {
if _, ok := sntByChainID[chainID]; !ok {
return nil, errorNotAvailableOnChainID
}
backend, err := c.RPCClient.EthClient(chainID)
if err != nil {
return nil, err
}
return erc20.NewSNT(sntByChainID[chainID], backend)
}

View File

@ -0,0 +1,3 @@
package erc20
//go:generate abigen -sol erc20.sol -pkg erc20 -out erc20.go

9975
services/ens/erc20/erc20.go Normal file

File diff suppressed because one or more lines are too long

1843
services/ens/erc20/erc20.sol Normal file

File diff suppressed because it is too large Load Diff

View File

@ -3,17 +3,23 @@ package ens
import ( import (
"github.com/ethereum/go-ethereum/p2p" "github.com/ethereum/go-ethereum/p2p"
ethRpc "github.com/ethereum/go-ethereum/rpc" ethRpc "github.com/ethereum/go-ethereum/rpc"
"github.com/status-im/status-go/account"
"github.com/status-im/status-go/params"
"github.com/status-im/status-go/rpc" "github.com/status-im/status-go/rpc"
"github.com/status-im/status-go/services/rpcfilters"
) )
// NewService initializes service instance. // NewService initializes service instance.
func NewService(rpcClient *rpc.Client) *Service { func NewService(rpcClient *rpc.Client, accountsManager *account.GethManager, rpcFiltersSrvc *rpcfilters.Service, config *params.NodeConfig) *Service {
return &Service{rpcClient} return &Service{rpcClient, accountsManager, rpcFiltersSrvc, config}
} }
// Service is a browsers service. // Service is a browsers service.
type Service struct { type Service struct {
rpcClient *rpc.Client rpcClient *rpc.Client
accountsManager *account.GethManager
rpcFiltersSrvc *rpcfilters.Service
config *params.NodeConfig
} }
// Start a service. // Start a service.
@ -32,7 +38,7 @@ func (s *Service) APIs() []ethRpc.API {
{ {
Namespace: "ens", Namespace: "ens",
Version: "0.1.0", Version: "0.1.0",
Service: NewAPI(s.rpcClient), Service: NewAPI(s.rpcClient, s.accountsManager, s.rpcFiltersSrvc, s.config),
}, },
} }
} }

View File

@ -30,3 +30,24 @@ func validateENSUsername(username string) error {
return nil return nil
} }
func usernameToLabel(username string) [32]byte {
usernameHashed := crypto.Keccak256([]byte(username))
var label [32]byte
copy(label[:], usernameHashed)
return label
}
func extractCoordinates(pubkey string) ([32]byte, [32]byte) {
x := []byte("0x" + pubkey[4:68])
y := []byte("0x" + pubkey[68:132])
var xByte [32]byte
copy(xByte[:], x)
var yByte [32]byte
copy(yByte[:], y)
return xByte, yByte
}

View File

@ -4,8 +4,10 @@ import (
"bytes" "bytes"
"context" "context"
"errors" "errors"
"math/big"
ethereum "github.com/ethereum/go-ethereum" ethereum "github.com/ethereum/go-ethereum"
"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/hexutil" "github.com/ethereum/go-ethereum/common/hexutil"
@ -78,6 +80,43 @@ func (args SendTxArgs) GetInput() types.HexBytes {
return args.Data return args.Data
} }
func (args SendTxArgs) ToTransactOpts(signerFn bind.SignerFn) *bind.TransactOpts {
var gasFeeCap *big.Int
if args.MaxFeePerGas != nil {
gasFeeCap = (*big.Int)(args.MaxFeePerGas)
}
var gasTipCap *big.Int
if args.MaxPriorityFeePerGas != nil {
gasTipCap = (*big.Int)(args.MaxPriorityFeePerGas)
}
var nonce *big.Int
if args.Nonce != nil {
nonce = new(big.Int).SetUint64((uint64)(*args.Nonce))
}
var gasPrice *big.Int
if args.GasPrice != nil {
gasPrice = (*big.Int)(args.GasPrice)
}
var gasLimit uint64
if args.Gas != nil {
gasLimit = uint64(*args.Gas)
}
return &bind.TransactOpts{
From: common.Address(args.From),
Signer: signerFn,
GasPrice: gasPrice,
GasLimit: gasLimit,
GasFeeCap: gasFeeCap,
GasTipCap: gasTipCap,
Nonce: nonce,
}
}
func isNilOrEmpty(bytes types.HexBytes) bool { func isNilOrEmpty(bytes types.HexBytes) bool {
return bytes == nil || len(bytes) == 0 return bytes == nil || len(bytes) == 0
} }