diff options
author | zelig <viktor.tron@gmail.com> | 2015-03-31 07:49:38 +0800 |
---|---|---|
committer | zelig <viktor.tron@gmail.com> | 2015-04-20 03:57:48 +0800 |
commit | d0b3536593fafe19ad31225387d09a62cac6196c (patch) | |
tree | 53cc1d7ee502ed696be641d250dbcbefd7f3482f /common/natspec/natspec.go | |
parent | ea11dba00b0f387f4083eef5ab3eeee396953830 (diff) | |
download | go-tangerine-d0b3536593fafe19ad31225387d09a62cac6196c.tar.gz go-tangerine-d0b3536593fafe19ad31225387d09a62cac6196c.tar.zst go-tangerine-d0b3536593fafe19ad31225387d09a62cac6196c.zip |
new natspec
- constructor takes abidoc, userdoc
- json parsing of userdoc
- method found by abi data
- notice found from method
Diffstat (limited to 'common/natspec/natspec.go')
-rw-r--r-- | common/natspec/natspec.go | 93 |
1 files changed, 85 insertions, 8 deletions
diff --git a/common/natspec/natspec.go b/common/natspec/natspec.go index 8e6568cef..f6b91d03d 100644 --- a/common/natspec/natspec.go +++ b/common/natspec/natspec.go @@ -1,19 +1,40 @@ package natspec import ( + "encoding/json" "fmt" "github.com/robertkrimen/otto" + "strings" + + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/crypto" ) +type abi2method map[[8]byte]*method + type NatSpec struct { - jsvm *otto.Otto + jsvm *otto.Otto + methods abi2method + userDocJson, abiDocJson []byte + userDoc userDoc + // abiDoc abiDoc } // TODO: should initialise with abi and userdoc jsons func New() (self *NatSpec, err error) { + // fetch abi, userdoc + abi := []byte("") + userdoc := []byte("") + return NewWithDocs(abi, userdoc) +} - self = new(NatSpec) - self.jsvm = otto.New() +func NewWithDocs(abiDocJson, userDocJson []byte) (self *NatSpec, err error) { + + self = &NatSpec{ + jsvm: otto.New(), + abiDocJson: abiDocJson, + userDocJson: userDocJson, + } _, err = self.jsvm.Run(natspecJS) if err != nil { @@ -24,20 +45,76 @@ func New() (self *NatSpec, err error) { return } + err = json.Unmarshal(userDocJson, &self.userDoc) + // err = parseAbiJson(abiDocJson, &self.abiDoc) + + 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` +// } + +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) { + if self.methods != nil { + meth = self.methods[abiKey] + return + } + self.methods = make(abi2method) + 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]) + self.methods[key] = meth + meth.name = name + if key == abiKey { + meth = m + } + } + return +} + +func (self *NatSpec) Notice(tx string, abi string) (notice string, err error) { + var abiKey [8]byte + copy(abiKey[:], []byte(abi)[:8]) + meth := self.makeAbi2method(abiKey) + if meth == nil { + err = fmt.Errorf("abi key %x does not match any method %v") + return + } + notice, err = self.noticeForMethod(tx, meth.name, meth.Notice) return } -func (self *NatSpec) Notice(transaction, abi, method, expression string) (string, error) { - var err error - if _, err = self.jsvm.Run("var transaction = " + transaction + ";"); err != nil { +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 = " + abi + ";"); err != nil { + 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 = '" + method + "';"); err != nil { + if _, err = self.jsvm.Run("var method = '" + name + "';"); err != nil { return "", fmt.Errorf("natspec.js error setting method: %v", err) } |