* 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.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package natspec
 | 
						|
 | 
						|
import (
 | 
						|
	"bytes"
 | 
						|
	"encoding/json"
 | 
						|
	"fmt"
 | 
						|
	"github.com/robertkrimen/otto"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/ethereum/go-ethereum/common"
 | 
						|
	"github.com/ethereum/go-ethereum/common/docserver"
 | 
						|
	"github.com/ethereum/go-ethereum/common/resolver"
 | 
						|
	"github.com/ethereum/go-ethereum/crypto"
 | 
						|
	"github.com/ethereum/go-ethereum/xeth"
 | 
						|
)
 | 
						|
 | 
						|
type abi2method map[[8]byte]*method
 | 
						|
 | 
						|
type NatSpec struct {
 | 
						|
	jsvm       *otto.Otto
 | 
						|
	abiDocJson []byte
 | 
						|
	userDoc    userDoc
 | 
						|
	tx, data   string
 | 
						|
}
 | 
						|
 | 
						|
// main entry point for to get natspec notice for a transaction
 | 
						|
// the implementation is frontend friendly in that it always gives back
 | 
						|
// a notice that is safe to display
 | 
						|
// :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) {
 | 
						|
	ns, err := New(xeth, tx, http)
 | 
						|
	if err != nil {
 | 
						|
		if ns == nil {
 | 
						|
			return getFallbackNotice(fmt.Sprintf("no NatSpec info found for contract: %v", err), tx)
 | 
						|
		} else {
 | 
						|
			return getFallbackNotice(fmt.Sprintf("invalid NatSpec info: %v", err), tx)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	notice, err = ns.Notice()
 | 
						|
	if err != nil {
 | 
						|
		return getFallbackNotice(fmt.Sprintf("NatSpec notice error: %v", err), tx)
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func getFallbackNotice(comment, tx string) string {
 | 
						|
	return fmt.Sprintf("About to submit transaction (%s): %s", comment, tx)
 | 
						|
}
 | 
						|
 | 
						|
type transaction struct {
 | 
						|
	To   string `json:"to"`
 | 
						|
	Data string `json:"data"`
 | 
						|
}
 | 
						|
 | 
						|
type jsonTx struct {
 | 
						|
	Params []transaction `json:"params"`
 | 
						|
}
 | 
						|
 | 
						|
type contractInfo struct {
 | 
						|
	Source        string          `json:"source"`
 | 
						|
	Language      string          `json:"language"`
 | 
						|
	Version       string          `json:"compilerVersion"`
 | 
						|
	AbiDefinition json.RawMessage `json:"abiDefinition"`
 | 
						|
	UserDoc       userDoc         `json:"userDoc"`
 | 
						|
	DeveloperDoc  json.RawMessage `json:"developerDoc"`
 | 
						|
}
 | 
						|
 | 
						|
func New(xeth *xeth.XEth, jsontx string, http *docserver.DocServer) (self *NatSpec, err error) {
 | 
						|
 | 
						|
	// extract contract address from tx
 | 
						|
	var tx jsonTx
 | 
						|
	err = json.Unmarshal([]byte(jsontx), &tx)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	t := tx.Params[0]
 | 
						|
	contractAddress := t.To
 | 
						|
 | 
						|
	content, err := FetchDocsForContract(contractAddress, xeth, http)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	self, err = NewWithDocs(content, jsontx, t.Data)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// also called by admin.contractInfo.get
 | 
						|
func FetchDocsForContract(contractAddress string, xeth *xeth.XEth, http *docserver.DocServer) (content []byte, err error) {
 | 
						|
	// retrieve contract hash from state
 | 
						|
	codehex := xeth.CodeAt(contractAddress)
 | 
						|
	codeb := xeth.CodeAtBytes(contractAddress)
 | 
						|
 | 
						|
	if codehex == "0x" {
 | 
						|
		err = fmt.Errorf("contract (%v) not found", contractAddress)
 | 
						|
		return
 | 
						|
	}
 | 
						|
	codehash := common.BytesToHash(crypto.Sha3(codeb))
 | 
						|
	// set up nameresolver with natspecreg + urlhint contract addresses
 | 
						|
	res := resolver.New(xeth)
 | 
						|
 | 
						|
	// resolve host via HashReg/UrlHint Resolver
 | 
						|
	uri, hash, err := res.KeyToUrl(codehash)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// get content via http client and authenticate content using hash
 | 
						|
	content, err = http.GetAuthContent(uri, hash)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func NewWithDocs(infoDoc []byte, tx string, data string) (self *NatSpec, err error) {
 | 
						|
 | 
						|
	var contract contractInfo
 | 
						|
	err = json.Unmarshal(infoDoc, &contract)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	self = &NatSpec{
 | 
						|
		jsvm:       otto.New(),
 | 
						|
		abiDocJson: []byte(contract.AbiDefinition),
 | 
						|
		userDoc:    contract.UserDoc,
 | 
						|
		tx:         tx,
 | 
						|
		data:       data,
 | 
						|
	}
 | 
						|
 | 
						|
	// load and require natspec js (but it is meant to be protected environment)
 | 
						|
	_, err = self.jsvm.Run(natspecJS)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	_, err = self.jsvm.Run("var natspec = require('natspec');")
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// type abiDoc []method
 | 
						|
 | 
						|
// type method struct {
 | 
						|
// 	Name   string  `json:name`
 | 
						|
// 	Inputs []input `json:inputs`
 | 
						|
// 	abiKey [8]byte
 | 
						|
// }
 | 
						|
 | 
						|
// type input struct {
 | 
						|
// 	Name string `json:name`
 | 
						|
// 	Type string `json:type`
 | 
						|
// }
 | 
						|
 | 
						|
// json skeleton for abi doc (contract method definitions)
 | 
						|
type method struct {
 | 
						|
	Notice string `json:notice`
 | 
						|
	name   string
 | 
						|
}
 | 
						|
 | 
						|
type userDoc struct {
 | 
						|
	Methods map[string]*method `json:methods`
 | 
						|
}
 | 
						|
 | 
						|
func (self *NatSpec) makeAbi2method(abiKey [8]byte) (meth *method) {
 | 
						|
	for signature, m := range self.userDoc.Methods {
 | 
						|
		name := strings.Split(signature, "(")[0]
 | 
						|
		hash := []byte(common.Bytes2Hex(crypto.Sha3([]byte(signature))))
 | 
						|
		var key [8]byte
 | 
						|
		copy(key[:], hash[:8])
 | 
						|
		if bytes.Equal(key[:], abiKey[:]) {
 | 
						|
			meth = m
 | 
						|
			meth.name = name
 | 
						|
			return
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (self *NatSpec) Notice() (notice string, err error) {
 | 
						|
	var abiKey [8]byte
 | 
						|
	if len(self.data) < 10 {
 | 
						|
		err = fmt.Errorf("Invalid transaction data")
 | 
						|
		return
 | 
						|
	}
 | 
						|
	copy(abiKey[:], self.data[2:10])
 | 
						|
	meth := self.makeAbi2method(abiKey)
 | 
						|
 | 
						|
	if meth == nil {
 | 
						|
		err = fmt.Errorf("abi key does not match any method")
 | 
						|
		return
 | 
						|
	}
 | 
						|
	notice, err = self.noticeForMethod(self.tx, meth.name, meth.Notice)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (self *NatSpec) noticeForMethod(tx string, name, expression string) (notice string, err error) {
 | 
						|
 | 
						|
	if _, err = self.jsvm.Run("var transaction = " + tx + ";"); err != nil {
 | 
						|
		return "", fmt.Errorf("natspec.js error setting transaction: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	if _, err = self.jsvm.Run("var abi = " + string(self.abiDocJson) + ";"); err != nil {
 | 
						|
		return "", fmt.Errorf("natspec.js error setting abi: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	if _, err = self.jsvm.Run("var method = '" + name + "';"); err != nil {
 | 
						|
		return "", fmt.Errorf("natspec.js error setting method: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	if _, err = self.jsvm.Run("var expression = \"" + expression + "\";"); err != nil {
 | 
						|
		return "", fmt.Errorf("natspec.js error setting expression: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	self.jsvm.Run("var call = {method: method,abi: abi,transaction: transaction};")
 | 
						|
	value, err := self.jsvm.Run("natspec.evaluateExpression(expression, call);")
 | 
						|
	if err != nil {
 | 
						|
		return "", fmt.Errorf("natspec.js error evaluating expression: %v", err)
 | 
						|
	}
 | 
						|
	evalError := "Natspec evaluation failed, wrong input params"
 | 
						|
	if value.String() == evalError {
 | 
						|
		return "", fmt.Errorf("natspec.js error evaluating expression: wrong input params in expression '%s'", expression)
 | 
						|
	}
 | 
						|
	if len(value.String()) == 0 {
 | 
						|
		return "", fmt.Errorf("natspec.js error evaluating expression")
 | 
						|
	}
 | 
						|
 | 
						|
	return value.String(), nil
 | 
						|
 | 
						|
}
 |