* common/compiler: solidity compiler + tests * rpc: eth_compilers, eth_compileSolidity + tests * fix natspec test using keystore API, notice exp dynamically changes addr, cleanup * resolver implements registrars and needs to create reg contract (temp) * xeth: solidity compiler. expose getter Solc() and paths setter SetSolc(solcPath) * ethereumApi: implement compiler related RPC calls using XEth - json struct tests * admin: make use of XEth.SetSolc to allow runtime setting of compiler paths * cli: command line flags solc to set custom solc bin path * js admin api with new features debug and contractInfo modules * wiki is the doc https://github.com/ethereum/go-ethereum/wiki/Contracts-and-Transactions
		
			
				
	
	
		
			233 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package resolver
 | |
| 
 | |
| import (
 | |
| 	"encoding/binary"
 | |
| 	"fmt"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/common"
 | |
| 	"github.com/ethereum/go-ethereum/crypto"
 | |
| 	"github.com/ethereum/go-ethereum/logger"
 | |
| 	"github.com/ethereum/go-ethereum/logger/glog"
 | |
| )
 | |
| 
 | |
| /*
 | |
| Resolver implements the Ethereum DNS mapping
 | |
| HashReg : Key Hash (hash of domain name or contract code) -> Content Hash
 | |
| UrlHint : Content Hash -> Url Hint
 | |
| 
 | |
| The resolver is meant to be called by the roundtripper transport implementation
 | |
| of a url scheme
 | |
| */
 | |
| 
 | |
| // // contract addresses will be hardcoded after they're created
 | |
| var UrlHintContractAddress, HashRegContractAddress string
 | |
| 
 | |
| const (
 | |
| 	txValue    = "0"
 | |
| 	txGas      = "100000"
 | |
| 	txGasPrice = "1000000000000"
 | |
| )
 | |
| 
 | |
| func abi(s string) string {
 | |
| 	return common.ToHex(crypto.Sha3([]byte(s))[:4])
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	registerContentHashAbi = abi("register(uint256,uint256)")
 | |
| 	registerUrlAbi         = abi("register(uint256,uint8,uint256)")
 | |
| 	setOwnerAbi            = abi("setowner()")
 | |
| )
 | |
| 
 | |
| type Backend interface {
 | |
| 	StorageAt(string, string) string
 | |
| 	Transact(fromStr, toStr, nonceStr, valueStr, gasStr, gasPriceStr, codeStr string) (string, error)
 | |
| }
 | |
| 
 | |
| type Resolver struct {
 | |
| 	backend Backend
 | |
| }
 | |
| 
 | |
| func New(eth Backend) *Resolver {
 | |
| 	return &Resolver{eth}
 | |
| }
 | |
| 
 | |
| // for testing and play temporarily
 | |
| // ideally the HashReg and UrlHint contracts should be in the genesis block
 | |
| // if we got build-in support for natspec/contract info
 | |
| // there should be only one of these officially endorsed
 | |
| // addresses as constants
 | |
| // TODO: could get around this with namereg, check
 | |
| func (self *Resolver) CreateContracts(addr common.Address) (err error) {
 | |
| 	HashRegContractAddress, err = self.backend.Transact(addr.Hex(), "", "", txValue, txGas, txGasPrice, ContractCodeHashReg)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	UrlHintContractAddress, err = self.backend.Transact(addr.Hex(), "", "", txValue, txGas, txGasPrice, ContractCodeURLhint)
 | |
| 	glog.V(logger.Detail).Infof("HashReg @ %v\nUrlHint @ %v\n", HashRegContractAddress, UrlHintContractAddress)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // called as first step in the registration process on HashReg
 | |
| func (self *Resolver) SetOwner(address common.Address) (txh string, err error) {
 | |
| 	return self.backend.Transact(
 | |
| 		address.Hex(),
 | |
| 		HashRegContractAddress,
 | |
| 		"", txValue, txGas, txGasPrice,
 | |
| 		setOwnerAbi,
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // registers some content hash to a key/code hash
 | |
| // e.g., the contract Info combined Json Doc's ContentHash
 | |
| // to CodeHash of a contract or hash of a domain
 | |
| // kept
 | |
| func (self *Resolver) RegisterContentHash(address common.Address, codehash, dochash common.Hash) (txh string, err error) {
 | |
| 	_, err = self.SetOwner(address)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	codehex := common.Bytes2Hex(codehash[:])
 | |
| 	dochex := common.Bytes2Hex(dochash[:])
 | |
| 
 | |
| 	data := registerContentHashAbi + codehex + dochex
 | |
| 	return self.backend.Transact(
 | |
| 		address.Hex(),
 | |
| 		HashRegContractAddress,
 | |
| 		"", txValue, txGas, txGasPrice,
 | |
| 		data,
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // registers a url to a content hash so that the content can be fetched
 | |
| // address is used as sender for the transaction and will be the owner of a new
 | |
| // registry entry on first time use
 | |
| // FIXME: silently doing nothing if sender is not the owner
 | |
| // note that with content addressed storage, this step is no longer necessary
 | |
| // it could be purely
 | |
| func (self *Resolver) RegisterUrl(address common.Address, hash common.Hash, url string) (txh string, err error) {
 | |
| 	hashHex := common.Bytes2Hex(hash[:])
 | |
| 	var urlHex string
 | |
| 	urlb := []byte(url)
 | |
| 	var cnt byte
 | |
| 	n := len(urlb)
 | |
| 
 | |
| 	for n > 0 {
 | |
| 		if n > 32 {
 | |
| 			n = 32
 | |
| 		}
 | |
| 		urlHex = common.Bytes2Hex(urlb[:n])
 | |
| 		urlb = urlb[n:]
 | |
| 		n = len(urlb)
 | |
| 		bcnt := make([]byte, 32)
 | |
| 		bcnt[31] = cnt
 | |
| 		data := registerUrlAbi +
 | |
| 			hashHex +
 | |
| 			common.Bytes2Hex(bcnt) +
 | |
| 			common.Bytes2Hex(common.Hex2BytesFixed(urlHex, 32))
 | |
| 		txh, err = self.backend.Transact(
 | |
| 			address.Hex(),
 | |
| 			UrlHintContractAddress,
 | |
| 			"", txValue, txGas, txGasPrice,
 | |
| 			data,
 | |
| 		)
 | |
| 		if err != nil {
 | |
| 			return
 | |
| 		}
 | |
| 		cnt++
 | |
| 	}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func (self *Resolver) Register(address common.Address, codehash, dochash common.Hash, url string) (txh string, err error) {
 | |
| 
 | |
| 	_, err = self.RegisterContentHash(address, codehash, dochash)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	return self.RegisterUrl(address, dochash, url)
 | |
| }
 | |
| 
 | |
| // resolution is costless non-transactional
 | |
| // implemented as direct retrieval from  db
 | |
| func (self *Resolver) KeyToContentHash(khash common.Hash) (chash common.Hash, err error) {
 | |
| 	// look up in hashReg
 | |
| 	at := common.Bytes2Hex(common.FromHex(HashRegContractAddress))
 | |
| 	key := storageAddress(storageMapping(storageIdx2Addr(1), khash[:]))
 | |
| 	hash := self.backend.StorageAt(at, key)
 | |
| 
 | |
| 	if hash == "0x0" || len(hash) < 3 {
 | |
| 		err = fmt.Errorf("content hash not found for '%v'", khash.Hex())
 | |
| 		return
 | |
| 	}
 | |
| 	copy(chash[:], common.Hex2BytesFixed(hash[2:], 32))
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // retrieves the url-hint for the content hash -
 | |
| // if we use content addressed storage, this step is no longer necessary
 | |
| func (self *Resolver) ContentHashToUrl(chash common.Hash) (uri string, err error) {
 | |
| 	// look up in URL reg
 | |
| 	var str string = " "
 | |
| 	var idx uint32
 | |
| 	for len(str) > 0 {
 | |
| 		mapaddr := storageMapping(storageIdx2Addr(1), chash[:])
 | |
| 		key := storageAddress(storageFixedArray(mapaddr, storageIdx2Addr(idx)))
 | |
| 		hex := self.backend.StorageAt(UrlHintContractAddress, key)
 | |
| 		str = string(common.Hex2Bytes(hex[2:]))
 | |
| 		l := len(str)
 | |
| 		for (l > 0) && (str[l-1] == 0) {
 | |
| 			l--
 | |
| 		}
 | |
| 		str = str[:l]
 | |
| 		uri = uri + str
 | |
| 		idx++
 | |
| 	}
 | |
| 
 | |
| 	if len(uri) == 0 {
 | |
| 		err = fmt.Errorf("GetURLhint: URL hint not found for '%v'", chash.Hex())
 | |
| 	}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func (self *Resolver) KeyToUrl(key common.Hash) (uri string, hash common.Hash, err error) {
 | |
| 	// look up in urlHint
 | |
| 	hash, err = self.KeyToContentHash(key)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 	uri, err = self.ContentHashToUrl(hash)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func storageIdx2Addr(varidx uint32) []byte {
 | |
| 	data := make([]byte, 32)
 | |
| 	binary.BigEndian.PutUint32(data[28:32], varidx)
 | |
| 	return data
 | |
| }
 | |
| 
 | |
| func storageMapping(addr, key []byte) []byte {
 | |
| 	data := make([]byte, 64)
 | |
| 	copy(data[0:32], key[0:32])
 | |
| 	copy(data[32:64], addr[0:32])
 | |
| 	sha := crypto.Sha3(data)
 | |
| 	return sha
 | |
| }
 | |
| 
 | |
| func storageFixedArray(addr, idx []byte) []byte {
 | |
| 	var carry byte
 | |
| 	for i := 31; i >= 0; i-- {
 | |
| 		var b byte = addr[i] + idx[i] + carry
 | |
| 		if b < addr[i] {
 | |
| 			carry = 1
 | |
| 		} else {
 | |
| 			carry = 0
 | |
| 		}
 | |
| 		addr[i] = b
 | |
| 	}
 | |
| 	return addr
 | |
| }
 | |
| 
 | |
| func storageAddress(addr []byte) string {
 | |
| 	return common.ToHex(addr)
 | |
| }
 |