Solidity Compiler - solc new API
* adapt to new compiler versioning * use compiler version as language version * implement new solc API for versions >= 0.1.[2-9][0-9]* fixes #1770 * add optimize=1 to options * backward compatibility (for now) for <= 0.1.1, and old versions (0.[2-9][0-9]*.[0-9]+) * introduce compilerOptions to ContractInfo * clean up flair, include full version string to version line and ContractInfo
This commit is contained in:
		| @@ -34,15 +34,10 @@ import ( | |||||||
| 	"github.com/ethereum/go-ethereum/logger/glog" | 	"github.com/ethereum/go-ethereum/logger/glog" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	// flair           = "Christian <c@ethdev.com> and Lefteris <lefteris@ethdev.com> (c) 2014-2015" |  | ||||||
| 	flair           = "" |  | ||||||
| 	languageVersion = "0" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	versionRegExp = regexp.MustCompile("[0-9]+.[0-9]+.[0-9]+") | 	versionRegExp = regexp.MustCompile("[0-9]+\\.[0-9]+\\.[0-9]+") | ||||||
| 	params        = []string{ | 	newAPIRegexp  = regexp.MustCompile("0\\.1\\.[2-9][0-9]*") | ||||||
|  | 	paramsLegacy  = []string{ | ||||||
| 		"--binary",       // Request to output the contract in binary (hexadecimal). | 		"--binary",       // Request to output the contract in binary (hexadecimal). | ||||||
| 		"file",           // | 		"file",           // | ||||||
| 		"--json-abi",     // Request to output the contract's JSON ABI interface. | 		"--json-abi",     // Request to output the contract's JSON ABI interface. | ||||||
| @@ -54,6 +49,15 @@ var ( | |||||||
| 		"--add-std", | 		"--add-std", | ||||||
| 		"1", | 		"1", | ||||||
| 	} | 	} | ||||||
|  | 	paramsNew = []string{ | ||||||
|  | 		"--bin",        // Request to output the contract in binary (hexadecimal). | ||||||
|  | 		"--abi",        // Request to output the contract's JSON ABI interface. | ||||||
|  | 		"--userdoc",    // Request to output the contract's Natspec user documentation. | ||||||
|  | 		"--devdoc",     // Request to output the contract's Natspec developer documentation. | ||||||
|  | 		"--add-std",    // include standard lib contracts | ||||||
|  | 		"--optimize=1", // code optimizer switched on | ||||||
|  | 		"-o",           // output directory | ||||||
|  | 	} | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Contract struct { | type Contract struct { | ||||||
| @@ -66,14 +70,17 @@ type ContractInfo struct { | |||||||
| 	Language        string      `json:"language"` | 	Language        string      `json:"language"` | ||||||
| 	LanguageVersion string      `json:"languageVersion"` | 	LanguageVersion string      `json:"languageVersion"` | ||||||
| 	CompilerVersion string      `json:"compilerVersion"` | 	CompilerVersion string      `json:"compilerVersion"` | ||||||
|  | 	CompilerOptions string      `json:"compilerOptions"` | ||||||
| 	AbiDefinition   interface{} `json:"abiDefinition"` | 	AbiDefinition   interface{} `json:"abiDefinition"` | ||||||
| 	UserDoc         interface{} `json:"userDoc"` | 	UserDoc         interface{} `json:"userDoc"` | ||||||
| 	DeveloperDoc    interface{} `json:"developerDoc"` | 	DeveloperDoc    interface{} `json:"developerDoc"` | ||||||
| } | } | ||||||
|  |  | ||||||
| type Solidity struct { | type Solidity struct { | ||||||
| 	solcPath string | 	solcPath    string | ||||||
| 	version  string | 	version     string | ||||||
|  | 	fullVersion string | ||||||
|  | 	legacy      bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func New(solcPath string) (sol *Solidity, err error) { | func New(solcPath string) (sol *Solidity, err error) { | ||||||
| @@ -94,17 +101,22 @@ func New(solcPath string) (sol *Solidity, err error) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	version := versionRegExp.FindString(out.String()) | 	fullVersion := out.String() | ||||||
|  | 	version := versionRegExp.FindString(fullVersion) | ||||||
|  | 	legacy := !newAPIRegexp.MatchString(version) | ||||||
|  |  | ||||||
| 	sol = &Solidity{ | 	sol = &Solidity{ | ||||||
| 		solcPath: solcPath, | 		solcPath:    solcPath, | ||||||
| 		version:  version, | 		version:     version, | ||||||
|  | 		fullVersion: fullVersion, | ||||||
|  | 		legacy:      legacy, | ||||||
| 	} | 	} | ||||||
| 	glog.V(logger.Info).Infoln(sol.Info()) | 	glog.V(logger.Info).Infoln(sol.Info()) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  |  | ||||||
| func (sol *Solidity) Info() string { | func (sol *Solidity) Info() string { | ||||||
| 	return fmt.Sprintf("solc v%s\nSolidity Compiler: %s\n%s", sol.version, sol.solcPath, flair) | 	return fmt.Sprintf("%s\npath: %s", sol.fullVersion, sol.solcPath) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (sol *Solidity) Version() string { | func (sol *Solidity) Version() string { | ||||||
| @@ -127,6 +139,15 @@ func (sol *Solidity) Compile(source string) (map[string]*Contract, error) { | |||||||
| 	// Assemble the compiler command, change to the temp folder and capture any errors | 	// Assemble the compiler command, change to the temp folder and capture any errors | ||||||
| 	stderr := new(bytes.Buffer) | 	stderr := new(bytes.Buffer) | ||||||
|  |  | ||||||
|  | 	var params []string | ||||||
|  | 	if sol.legacy { | ||||||
|  | 		params = paramsLegacy | ||||||
|  | 	} else { | ||||||
|  | 		params = paramsNew | ||||||
|  | 		params = append(params, wd) | ||||||
|  | 	} | ||||||
|  | 	compilerOptions := strings.Join(params, " ") | ||||||
|  |  | ||||||
| 	cmd := exec.Command(sol.solcPath, params...) | 	cmd := exec.Command(sol.solcPath, params...) | ||||||
| 	cmd.Dir = wd | 	cmd.Dir = wd | ||||||
| 	cmd.Stdin = strings.NewReader(source) | 	cmd.Stdin = strings.NewReader(source) | ||||||
| @@ -136,7 +157,7 @@ func (sol *Solidity) Compile(source string) (map[string]*Contract, error) { | |||||||
| 		return nil, fmt.Errorf("solc: %v\n%s", err, string(stderr.Bytes())) | 		return nil, fmt.Errorf("solc: %v\n%s", err, string(stderr.Bytes())) | ||||||
| 	} | 	} | ||||||
| 	// Sanity check that something was actually built | 	// Sanity check that something was actually built | ||||||
| 	matches, _ := filepath.Glob(wd + "/*.binary") | 	matches, _ := filepath.Glob(wd + "/*\\.bin*") | ||||||
| 	if len(matches) < 1 { | 	if len(matches) < 1 { | ||||||
| 		return nil, fmt.Errorf("solc: no build results found") | 		return nil, fmt.Errorf("solc: no build results found") | ||||||
| 	} | 	} | ||||||
| @@ -148,7 +169,11 @@ func (sol *Solidity) Compile(source string) (map[string]*Contract, error) { | |||||||
|  |  | ||||||
| 		// Parse the individual compilation results (code binary, ABI definitions, user and dev docs) | 		// Parse the individual compilation results (code binary, ABI definitions, user and dev docs) | ||||||
| 		var binary []byte | 		var binary []byte | ||||||
| 		if binary, err = ioutil.ReadFile(filepath.Join(wd, base+".binary")); err != nil { | 		binext := ".bin" | ||||||
|  | 		if sol.legacy { | ||||||
|  | 			binext = ".binary" | ||||||
|  | 		} | ||||||
|  | 		if binary, err = ioutil.ReadFile(filepath.Join(wd, base+binext)); err != nil { | ||||||
| 			return nil, fmt.Errorf("solc: error reading compiler output for code: %v", err) | 			return nil, fmt.Errorf("solc: error reading compiler output for code: %v", err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| @@ -178,8 +203,9 @@ func (sol *Solidity) Compile(source string) (map[string]*Contract, error) { | |||||||
| 			Info: ContractInfo{ | 			Info: ContractInfo{ | ||||||
| 				Source:          source, | 				Source:          source, | ||||||
| 				Language:        "Solidity", | 				Language:        "Solidity", | ||||||
| 				LanguageVersion: languageVersion, | 				LanguageVersion: sol.version, | ||||||
| 				CompilerVersion: sol.version, | 				CompilerVersion: sol.version, | ||||||
|  | 				CompilerOptions: compilerOptions, | ||||||
| 				AbiDefinition:   abi, | 				AbiDefinition:   abi, | ||||||
| 				UserDoc:         userdoc, | 				UserDoc:         userdoc, | ||||||
| 				DeveloperDoc:    devdoc, | 				DeveloperDoc:    devdoc, | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ import ( | |||||||
| 	"github.com/ethereum/go-ethereum/common" | 	"github.com/ethereum/go-ethereum/common" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const solcVersion = "0.9.23" | const solcVersion = "0.1.1" | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	source = ` | 	source = ` | ||||||
| @@ -37,16 +37,16 @@ contract test { | |||||||
|    } |    } | ||||||
| } | } | ||||||
| ` | ` | ||||||
| 	code = "0x605880600c6000396000f3006000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa114602e57005b603d6004803590602001506047565b8060005260206000f35b60006007820290506053565b91905056" | 	code = "0x6060604052606d8060116000396000f30060606040526000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa1146037576035565b005b6046600480359060200150605c565b6040518082815260200191505060405180910390f35b60006007820290506068565b91905056" | ||||||
| 	info = `{"source":"\ncontract test {\n   /// @notice Will multiply ` + "`a`" + ` by 7.\n   function multiply(uint a) returns(uint d) {\n       return a * 7;\n   }\n}\n","language":"Solidity","languageVersion":"0","compilerVersion":"0.9.23","abiDefinition":[{"constant":false,"inputs":[{"name":"a","type":"uint256"}],"name":"multiply","outputs":[{"name":"d","type":"uint256"}],"type":"function"}],"userDoc":{"methods":{"multiply(uint256)":{"notice":"Will multiply ` + "`a`" + ` by 7."}}},"developerDoc":{"methods":{}}}` | 	info = `{"source":"\ncontract test {\n   /// @notice Will multiply ` + "`a`" + ` by 7.\n   function multiply(uint a) returns(uint d) {\n       return a * 7;\n   }\n}\n","language":"Solidity","languageVersion":"0.1.1","compilerVersion":"0.1.1","compilerOptions":"--binary file --json-abi file --natspec-user file --natspec-dev file --add-std 1","abiDefinition":[{"constant":false,"inputs":[{"name":"a","type":"uint256"}],"name":"multiply","outputs":[{"name":"d","type":"uint256"}],"type":"function"}],"userDoc":{"methods":{"multiply(uint256)":{"notice":"Will multiply ` + "`a`" + ` by 7."}}},"developerDoc":{"methods":{}}}` | ||||||
|  |  | ||||||
| 	infohash = common.HexToHash("0xea782f674eb898e477c20e8a7cf11c2c28b09fa68b5278732104f7a101aed255") | 	infohash = common.HexToHash("0x9f3803735e7f16120c5a140ab3f02121fd3533a9655c69b33a10e78752cc49b0") | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestCompiler(t *testing.T) { | func TestCompiler(t *testing.T) { | ||||||
| 	sol, err := New("") | 	sol, err := New("") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Skip("solc not found: skip") | 		t.Skip("solc not found: skip: %v", err) | ||||||
| 	} else if sol.Version() != solcVersion { | 	} else if sol.Version() != solcVersion { | ||||||
| 		t.Skip("WARNING: skipping due to a newer version of solc found (%v, expect %v)", sol.Version(), solcVersion) | 		t.Skip("WARNING: skipping due to a newer version of solc found (%v, expect %v)", sol.Version(), solcVersion) | ||||||
| 	} | 	} | ||||||
| @@ -111,4 +111,4 @@ func TestSaveInfo(t *testing.T) { | |||||||
| 	if cinfohash != infohash { | 	if cinfohash != infohash { | ||||||
| 		t.Errorf("content hash for info is incorrect. expected %v, got %v", infohash.Hex(), cinfohash.Hex()) | 		t.Errorf("content hash for info is incorrect. expected %v, got %v", infohash.Hex(), cinfohash.Hex()) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user