310 lines
8.6 KiB
Go
Raw Normal View History

// Copyright 2017-2019 Weald Technology Trading
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package ens
import (
"bytes"
"compress/zlib"
"errors"
"io"
"io/ioutil"
"math/big"
"strings"
"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core/types"
"github.com/wealdtech/go-ens/v3/contracts/resolver"
)
var zeroHash = make([]byte, 32)
// UnknownAddress is the address to which unknown entries resolve
var UnknownAddress = common.HexToAddress("00")
// Resolver is the structure for the resolver contract
type Resolver struct {
Contract *resolver.Contract
ContractAddr common.Address
domain string
}
// NewResolver obtains an ENS resolver for a given domain
func NewResolver(backend bind.ContractBackend, domain string) (*Resolver, error) {
registry, err := NewRegistry(backend)
if err != nil {
return nil, err
}
// Ensure the name is registered
ownerAddress, err := registry.Owner(domain)
if err != nil {
return nil, err
}
if bytes.Equal(ownerAddress.Bytes(), UnknownAddress.Bytes()) {
return nil, errors.New("unregistered name")
}
// Obtain the resolver address for this domain
resolver, err := registry.ResolverAddress(domain)
if err != nil {
return nil, err
}
return NewResolverAt(backend, domain, resolver)
}
// NewResolverAt obtains an ENS resolver at a given address
func NewResolverAt(backend bind.ContractBackend, domain string, address common.Address) (*Resolver, error) {
contract, err := resolver.NewContract(address, backend)
if err != nil {
return nil, err
}
// Ensure this really is a resolver contract
nameHash, err := NameHash("test.eth")
if err != nil {
return nil, err
}
_, err = contract.Addr(nil, nameHash)
if err != nil {
if err.Error() == "no contract code at given address" {
return nil, errors.New("no resolver")
}
return nil, err
}
return &Resolver{
Contract: contract,
ContractAddr: address,
domain: domain,
}, nil
}
// PublicResolverAddress obtains the address of the public resolver for a chain
func PublicResolverAddress(backend bind.ContractBackend) (common.Address, error) {
return Resolve(backend, "resolver.eth")
}
// Address returns the Ethereum address of the domain
func (r *Resolver) Address() (common.Address, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return UnknownAddress, err
}
return r.Contract.Addr(nil, nameHash)
}
// SetAddress sets the Ethereum address of the domain
func (r *Resolver) SetAddress(opts *bind.TransactOpts, address common.Address) (*types.Transaction, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetAddr(opts, nameHash, address)
}
// MultiAddress returns the address of the domain for a given coin type.
// The coin type is as per https://github.com/satoshilabs/slips/blob/master/slip-0044.md
func (r *Resolver) MultiAddress(coinType uint64) ([]byte, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.Addr0(nil, nameHash, big.NewInt(int64(coinType)))
}
// SetMultiAddress sets the iaddress of the domain for a given coin type.
// The coin type is as per https://github.com/satoshilabs/slips/blob/master/slip-0044.md
func (r *Resolver) SetMultiAddress(opts *bind.TransactOpts, coinType uint64, address []byte) (*types.Transaction, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetAddr0(opts, nameHash, big.NewInt(int64(coinType)), address)
}
// PubKey returns the public key of the domain
func (r *Resolver) PubKey() ([32]byte, [32]byte, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return [32]byte{}, [32]byte{}, err
}
res, err := r.Contract.Pubkey(nil, nameHash)
return res.X, res.Y, err
}
// SetPubKey sets the public key of the domain
func (r *Resolver) SetPubKey(opts *bind.TransactOpts, x [32]byte, y [32]byte) (*types.Transaction, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetPubkey(opts, nameHash, x, y)
}
// Contenthash returns the content hash of the domain
func (r *Resolver) Contenthash() ([]byte, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.Contenthash(nil, nameHash)
}
// SetContenthash sets the content hash of the domain
func (r *Resolver) SetContenthash(opts *bind.TransactOpts, contenthash []byte) (*types.Transaction, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetContenthash(opts, nameHash, contenthash)
}
// InterfaceImplementer returns the address of the contract that implements the given interface for the given domain
func (r *Resolver) InterfaceImplementer(interfaceID [4]byte) (common.Address, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return UnknownAddress, err
}
return r.Contract.InterfaceImplementer(nil, nameHash, interfaceID)
}
// Resolve resolves an ENS name in to an Etheruem address
// This will return an error if the name is not found or otherwise 0
func Resolve(backend bind.ContractBackend, input string) (address common.Address, err error) {
if strings.Contains(input, ".") {
return resolveName(backend, input)
}
if (strings.HasPrefix(input, "0x") && len(input) > 42) || (!strings.HasPrefix(input, "0x") && len(input) > 40) {
err = errors.New("address too long")
} else {
address = common.HexToAddress(input)
if address == UnknownAddress {
err = errors.New("could not parse address")
}
}
return
}
func resolveName(backend bind.ContractBackend, input string) (address common.Address, err error) {
nameHash, err := NameHash(input)
if err != nil {
return UnknownAddress, err
}
if bytes.Equal(nameHash[:], zeroHash) {
err = errors.New("bad name")
} else {
address, err = resolveHash(backend, input)
}
return
}
func resolveHash(backend bind.ContractBackend, domain string) (address common.Address, err error) {
resolver, err := NewResolver(backend, domain)
if err != nil {
return UnknownAddress, err
}
// Resolve the domain
address, err = resolver.Address()
if err != nil {
return UnknownAddress, err
}
if bytes.Equal(address.Bytes(), UnknownAddress.Bytes()) {
return UnknownAddress, errors.New("no address")
}
return
}
// SetText sets the text associated with a name
func (r *Resolver) SetText(opts *bind.TransactOpts, name string, value string) (*types.Transaction, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetText(opts, nameHash, name, value)
}
// Text obtains the text associated with a name
func (r *Resolver) Text(name string) (string, error) {
nameHash, err := NameHash(r.domain)
if err != nil {
return "", err
}
return r.Contract.Text(nil, nameHash, name)
}
// SetABI sets the ABI associated with a name
func (r *Resolver) SetABI(opts *bind.TransactOpts, name string, abi string, contentType *big.Int) (*types.Transaction, error) {
var data []byte
switch contentType.Uint64() {
case 1:
// Uncompressed JSON
data = []byte(abi)
case 2:
// Zlib-compressed JSON
var b bytes.Buffer
w := zlib.NewWriter(&b)
if _, err := w.Write([]byte(abi)); err != nil {
return nil, err
}
w.Close()
data = b.Bytes()
default:
return nil, errors.New("unsupported content type")
}
nameHash, err := NameHash(r.domain)
if err != nil {
return nil, err
}
return r.Contract.SetABI(opts, nameHash, contentType, data)
}
// ABI returns the ABI associated with a name
func (r *Resolver) ABI(name string) (string, error) {
contentTypes := big.NewInt(3)
nameHash, err := NameHash(name)
if err != nil {
return "", err
}
contentType, data, err := r.Contract.ABI(nil, nameHash, contentTypes)
var abi string
if err == nil {
if contentType.Cmp(big.NewInt(1)) == 0 {
// Uncompressed JSON
abi = string(data)
} else if contentType.Cmp(big.NewInt(2)) == 0 {
// Zlib-compressed JSON
b := bytes.NewReader(data)
var z io.ReadCloser
z, err = zlib.NewReader(b)
if err != nil {
return "", err
}
defer z.Close()
var uncompressed []byte
uncompressed, err = ioutil.ReadAll(z)
if err != nil {
return "", err
}
abi = string(uncompressed)
}
}
return abi, nil
}