func (self *Expanse) Solc() (*compiler.Solidity, error) { var err error if self.solc == nil { self.solc, err = compiler.New(self.SolcPath) } return self.solc, err }
func TestCompileSolidity(t *testing.T) { solc, err := compiler.New("") if solc == nil { t.Skip("no solc found: skip") } else if solc.Version() != solcVersion { t.Skip("WARNING: skipping test because of solc different version (%v, test written for %v, may need to update)", solc.Version(), solcVersion) } source := `contract test {\n` + " /// @notice Will multiply `a` by 7." + `\n` + ` function multiply(uint a) returns(uint d) {\n` + ` return a * 7;\n` + ` }\n` + `}\n` jsonstr := `{"jsonrpc":"2.0","method":"eth_compileSolidity","params":["` + source + `"],"id":64}` expCode := "0x605880600c6000396000f3006000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa114602e57005b603d6004803590602001506047565b8060005260206000f35b60006007820290506053565b91905056" expAbiDefinition := `[{"constant":false,"inputs":[{"name":"a","type":"uint256"}],"name":"multiply","outputs":[{"name":"d","type":"uint256"}],"type":"function"}]` expUserDoc := `{"methods":{"multiply(uint256)":{"notice":"Will multiply ` + "`a`" + ` by 7."}}}` expDeveloperDoc := `{"methods":{}}` expCompilerVersion := solc.Version() expLanguage := "Solidity" expLanguageVersion := "0" expSource := source exp := &exp.Expanse{} xeth := xeth.NewTest(exp, nil) api := NewEthApi(xeth, exp, codec.JSON) var rpcRequest shared.Request json.Unmarshal([]byte(jsonstr), &rpcRequest) response, err := api.CompileSolidity(&rpcRequest) if err != nil { t.Errorf("Execution failed, %v", err) } respjson, err := json.Marshal(response) if err != nil { t.Errorf("expected no error, got %v", err) } var contracts = make(map[string]*compiler.Contract) err = json.Unmarshal(respjson, &contracts) if err != nil { t.Errorf("expected no error, got %v", err) } if len(contracts) != 1 { t.Errorf("expected one contract, got %v", len(contracts)) } contract := contracts["test"] if contract.Code != expCode { t.Errorf("Expected \n%s got \n%s", expCode, contract.Code) } if strconv.Quote(contract.Info.Source) != `"`+expSource+`"` { t.Errorf("Expected \n'%s' got \n'%s'", expSource, strconv.Quote(contract.Info.Source)) } if contract.Info.Language != expLanguage { t.Errorf("Expected %s got %s", expLanguage, contract.Info.Language) } if contract.Info.LanguageVersion != expLanguageVersion { t.Errorf("Expected %s got %s", expLanguageVersion, contract.Info.LanguageVersion) } if contract.Info.CompilerVersion != expCompilerVersion { t.Errorf("Expected %s got %s", expCompilerVersion, contract.Info.CompilerVersion) } userdoc, err := json.Marshal(contract.Info.UserDoc) if err != nil { t.Errorf("expected no error, got %v", err) } devdoc, err := json.Marshal(contract.Info.DeveloperDoc) if err != nil { t.Errorf("expected no error, got %v", err) } abidef, err := json.Marshal(contract.Info.AbiDefinition) if err != nil { t.Errorf("expected no error, got %v", err) } if string(abidef) != expAbiDefinition { t.Errorf("Expected \n'%s' got \n'%s'", expAbiDefinition, string(abidef)) } if string(userdoc) != expUserDoc { t.Errorf("Expected \n'%s' got \n'%s'", expUserDoc, string(userdoc)) } if string(devdoc) != expDeveloperDoc { t.Errorf("Expected %s got %s", expDeveloperDoc, string(devdoc)) } }
func TestContract(t *testing.T) { t.Skip("contract testing is implemented with mining in ethash test mode. This takes about 7seconds to run. Unskip and run on demand") coinbase := common.HexToAddress(testAddress) tmp, repl, expanse := testREPL(t, func(conf *exp.Config) { conf.Etherbase = coinbase conf.PowTest = true }) if err := expanse.Start(); err != nil { t.Errorf("error starting expanse: %v", err) return } defer expanse.Stop() defer os.RemoveAll(tmp) reg := registrar.New(repl.xeth) _, err := reg.SetGlobalRegistrar("", coinbase) if err != nil { t.Errorf("error setting HashReg: %v", err) } _, err = reg.SetHashReg("", coinbase) if err != nil { t.Errorf("error setting HashReg: %v", err) } _, err = reg.SetUrlHint("", coinbase) if err != nil { t.Errorf("error setting HashReg: %v", err) } /* TODO: * lookup receipt and contract addresses by tx hash * name registration for HashReg and UrlHint addresses * mine those transactions * then set once more SetHashReg SetUrlHint */ source := `contract test {\n` + " /// @notice Will multiply `a` by 7." + `\n` + ` function multiply(uint a) returns(uint d) {\n` + ` return a * 7;\n` + ` }\n` + `}\n` if checkEvalJSON(t, repl, `admin.stopNatSpec()`, `true`) != nil { return } contractInfo, err := ioutil.ReadFile("info_test.json") if err != nil { t.Fatalf("%v", err) } if checkEvalJSON(t, repl, `primary = exp.accounts[0]`, `"`+testAddress+`"`) != nil { return } if checkEvalJSON(t, repl, `source = "`+source+`"`, `"`+source+`"`) != nil { return } // if solc is found with right version, test it, otherwise read from file sol, err := compiler.New("") if err != nil { t.Logf("solc not found: mocking contract compilation step") } else if sol.Version() != solcVersion { t.Logf("WARNING: solc different version found (%v, test written for %v, may need to update)", sol.Version(), solcVersion) } if err != nil { info, err := ioutil.ReadFile("info_test.json") if err != nil { t.Fatalf("%v", err) } _, err = repl.re.Run(`contract = JSON.parse(` + strconv.Quote(string(info)) + `)`) if err != nil { t.Errorf("%v", err) } } else { if checkEvalJSON(t, repl, `contract = exp.compile.solidity(source).test`, string(contractInfo)) != nil { return } } if checkEvalJSON(t, repl, `contract.code`, `"0x605880600c6000396000f3006000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa114602e57005b603d6004803590602001506047565b8060005260206000f35b60006007820290506053565b91905056"`) != nil { return } if checkEvalJSON( t, repl, `contractaddress = exp.sendTransaction({from: primary, data: contract.code})`, `"0x46d69d55c3c4b86a924a92c9fc4720bb7bce1d74"`, ) != nil { return } if !processTxs(repl, t, 8) { return } callSetup := `abiDef = JSON.parse('[{"constant":false,"inputs":[{"name":"a","type":"uint256"}],"name":"multiply","outputs":[{"name":"d","type":"uint256"}],"type":"function"}]'); Multiply7 = exp.contract(abiDef); multiply7 = Multiply7.at(contractaddress); ` _, err = repl.re.Run(callSetup) if err != nil { t.Errorf("unexpected error setting up contract, got %v", err) return } expNotice := "" if repl.lastConfirm != expNotice { t.Errorf("incorrect confirmation message: expected %v, got %v", expNotice, repl.lastConfirm) return } if checkEvalJSON(t, repl, `admin.startNatSpec()`, `true`) != nil { return } if checkEvalJSON(t, repl, `multiply7.multiply.sendTransaction(6, { from: primary })`, `"0x4ef9088431a8033e4580d00e4eb2487275e031ff4163c7529df0ef45af17857b"`) != nil { return } if !processTxs(repl, t, 1) { return } expNotice = `About to submit transaction (no NatSpec info found for contract: content hash not found for '0x87e2802265838c7f14bb69eecd2112911af6767907a702eeaa445239fb20711b'): {"params":[{"to":"0x46d69d55c3c4b86a924a92c9fc4720bb7bce1d74","data": "0xc6888fa10000000000000000000000000000000000000000000000000000000000000006"}]}` if repl.lastConfirm != expNotice { t.Errorf("incorrect confirmation message: expected\n%v, got\n%v", expNotice, repl.lastConfirm) return } var contentHash = `"0x86d2b7cf1e72e9a7a3f8d96601f0151742a2f780f1526414304fbe413dc7f9bd"` if sol != nil && solcVersion != sol.Version() { modContractInfo := versionRE.ReplaceAll(contractInfo, []byte(`"compilerVersion":"`+sol.Version()+`"`)) fmt.Printf("modified contractinfo:\n%s\n", modContractInfo) contentHash = `"` + common.ToHex(crypto.Sha3([]byte(modContractInfo))) + `"` } if checkEvalJSON(t, repl, `filename = "/tmp/info.json"`, `"/tmp/info.json"`) != nil { return } if checkEvalJSON(t, repl, `contentHash = admin.saveInfo(contract.info, filename)`, contentHash) != nil { return } if checkEvalJSON(t, repl, `admin.register(primary, contractaddress, contentHash)`, `true`) != nil { return } if checkEvalJSON(t, repl, `admin.registerUrl(primary, contentHash, "file://"+filename)`, `true`) != nil { return } if checkEvalJSON(t, repl, `admin.startNatSpec()`, `true`) != nil { return } if !processTxs(repl, t, 3) { return } if checkEvalJSON(t, repl, `multiply7.multiply.sendTransaction(6, { from: primary })`, `"0x66d7635c12ad0b231e66da2f987ca3dfdca58ffe49c6442aa55960858103fd0c"`) != nil { return } if !processTxs(repl, t, 1) { return } expNotice = "Will multiply 6 by 7." if repl.lastConfirm != expNotice { t.Errorf("incorrect confirmation message: expected\n%v, got\n%v", expNotice, repl.lastConfirm) return } }
func main() { // Parse and ensure all needed inputs are specified flag.Parse() if *abiFlag == "" && *solFlag == "" { fmt.Printf("No contract ABI (--abi) or Solidity source (--sol) specified\n") os.Exit(-1) } else if (*abiFlag != "" || *binFlag != "" || *typFlag != "") && *solFlag != "" { fmt.Printf("Contract ABI (--abi), bytecode (--bin) and type (--type) flags are mutually exclusive with the Solidity source (--sol) flag\n") os.Exit(-1) } if *pkgFlag == "" { fmt.Printf("No destination Go package specified (--pkg)\n") os.Exit(-1) } // If the entire solidity code was specified, build and bind based on that var ( abis []string bins []string types []string ) if *solFlag != "" { // Generate the list of types to exclude from binding exclude := make(map[string]bool) for _, kind := range strings.Split(*excFlag, ",") { exclude[strings.ToLower(kind)] = true } // Build the Solidity source into bindable components solc, err := compiler.New(*solcFlag) if err != nil { fmt.Printf("Failed to locate Solidity compiler: %v\n", err) os.Exit(-1) } source, err := ioutil.ReadFile(*solFlag) if err != nil { fmt.Printf("Failed to read Soldity source code: %v\n", err) os.Exit(-1) } contracts, err := solc.Compile(string(source)) if err != nil { fmt.Printf("Failed to build Solidity contract: %v\n", err) os.Exit(-1) } // Gather all non-excluded contract for binding for name, contract := range contracts { if exclude[strings.ToLower(name)] { continue } abi, _ := json.Marshal(contract.Info.AbiDefinition) // Flatten the compiler parse abis = append(abis, string(abi)) bins = append(bins, contract.Code) types = append(types, name) } } else { // Otherwise load up the ABI, optional bytecode and type name from the parameters abi, err := ioutil.ReadFile(*abiFlag) if err != nil { fmt.Printf("Failed to read input ABI: %v\n", err) os.Exit(-1) } abis = append(abis, string(abi)) bin := []byte{} if *binFlag != "" { if bin, err = ioutil.ReadFile(*binFlag); err != nil { fmt.Printf("Failed to read input bytecode: %v\n", err) os.Exit(-1) } } bins = append(bins, string(bin)) kind := *typFlag if kind == "" { kind = *pkgFlag } types = append(types, kind) } // Generate the contract binding code, err := bind.Bind(types, abis, bins, *pkgFlag) if err != nil { fmt.Printf("Failed to generate ABI binding: %v\n", err) os.Exit(-1) } // Either flush it out to a file or display on the standard output if *outFlag == "" { fmt.Printf("%s\n", code) return } if err := ioutil.WriteFile(*outFlag, []byte(code), 0600); err != nil { fmt.Printf("Failed to write ABI binding: %v\n", err) os.Exit(-1) } }