main.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. // Copyright 2014 The go-ethereum Authors
  2. // This file is part of go-ethereum.
  3. //
  4. // go-ethereum is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // go-ethereum is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU General Public License
  15. // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
  16. // ethtest executes Ethereum JSON tests.
  17. package main
  18. import (
  19. "fmt"
  20. "io"
  21. "io/ioutil"
  22. "os"
  23. "path/filepath"
  24. "strings"
  25. "github.com/ethereum/go-ethereum/log"
  26. "github.com/ethereum/go-ethereum/params"
  27. "github.com/ethereum/go-ethereum/tests"
  28. "gopkg.in/urfave/cli.v1"
  29. )
  30. var (
  31. continueOnError = false
  32. testExtension = ".json"
  33. defaultTest = "all"
  34. defaultDir = "."
  35. allTests = []string{"BlockTests", "StateTests", "TransactionTests", "VMTests", "RLPTests"}
  36. testDirMapping = map[string]string{"BlockTests": "BlockchainTests"}
  37. skipTests = []string{}
  38. TestFlag = cli.StringFlag{
  39. Name: "test",
  40. Usage: "Test type (string): VMTests, TransactionTests, StateTests, BlockTests",
  41. Value: defaultTest,
  42. }
  43. FileFlag = cli.StringFlag{
  44. Name: "file",
  45. Usage: "Test file or directory. Directories are searched for .json files 1 level deep",
  46. Value: defaultDir,
  47. EnvVar: "ETHEREUM_TEST_PATH",
  48. }
  49. ContinueOnErrorFlag = cli.BoolFlag{
  50. Name: "continue",
  51. Usage: "Continue running tests on error (true) or [default] exit immediately (false)",
  52. }
  53. ReadStdInFlag = cli.BoolFlag{
  54. Name: "stdin",
  55. Usage: "Accept input from stdin instead of reading from file",
  56. }
  57. SkipTestsFlag = cli.StringFlag{
  58. Name: "skip",
  59. Usage: "Tests names to skip",
  60. }
  61. TraceFlag = cli.BoolFlag{
  62. Name: "trace",
  63. Usage: "Enable VM tracing",
  64. }
  65. )
  66. func runTestWithReader(test string, r io.Reader) error {
  67. log.Info("Running test", "test", test)
  68. var err error
  69. switch strings.ToLower(test) {
  70. case "bk", "block", "blocktest", "blockchaintest", "blocktests", "blockchaintests":
  71. err = tests.RunBlockTestWithReader(params.MainNetHomesteadBlock, params.MainNetDAOForkBlock, params.MainNetHomesteadGasRepriceBlock, r, skipTests)
  72. case "st", "state", "statetest", "statetests":
  73. rs := &params.ChainConfig{HomesteadBlock: params.MainNetHomesteadBlock, DAOForkBlock: params.MainNetDAOForkBlock, DAOForkSupport: true, EIP150Block: params.MainNetHomesteadGasRepriceBlock}
  74. err = tests.RunStateTestWithReader(rs, r, skipTests)
  75. case "tx", "transactiontest", "transactiontests":
  76. rs := &params.ChainConfig{HomesteadBlock: params.MainNetHomesteadBlock, DAOForkBlock: params.MainNetDAOForkBlock, DAOForkSupport: true, EIP150Block: params.MainNetHomesteadGasRepriceBlock}
  77. err = tests.RunTransactionTestsWithReader(rs, r, skipTests)
  78. case "vm", "vmtest", "vmtests":
  79. err = tests.RunVmTestWithReader(r, skipTests)
  80. case "rlp", "rlptest", "rlptests":
  81. err = tests.RunRLPTestWithReader(r, skipTests)
  82. default:
  83. err = fmt.Errorf("Invalid test type specified: %v", test)
  84. }
  85. return err
  86. }
  87. func getFiles(path string) ([]string, error) {
  88. log.Info("Listing files", "path", path)
  89. var files []string
  90. f, err := os.Open(path)
  91. if err != nil {
  92. return nil, err
  93. }
  94. defer f.Close()
  95. fi, err := f.Stat()
  96. if err != nil {
  97. return nil, err
  98. }
  99. switch mode := fi.Mode(); {
  100. case mode.IsDir():
  101. fi, _ := ioutil.ReadDir(path)
  102. files = make([]string, len(fi))
  103. for i, v := range fi {
  104. // only go 1 depth and leave directory entires blank
  105. if !v.IsDir() && v.Name()[len(v.Name())-len(testExtension):len(v.Name())] == testExtension {
  106. files[i] = filepath.Join(path, v.Name())
  107. log.Info("Found test file", "file", files[i])
  108. }
  109. }
  110. case mode.IsRegular():
  111. files = make([]string, 1)
  112. files[0] = path
  113. }
  114. return files, nil
  115. }
  116. func runSuite(test, file string) {
  117. var tests []string
  118. if test == defaultTest {
  119. tests = allTests
  120. } else {
  121. tests = []string{test}
  122. }
  123. for _, curTest := range tests {
  124. suiteLogger := log.New("suite", file, "test", curTest)
  125. suiteLogger.Info("Running test suite")
  126. var err error
  127. var files []string
  128. if test == defaultTest {
  129. // check if we have an explicit directory mapping for the test
  130. if _, ok := testDirMapping[curTest]; ok {
  131. files, err = getFiles(filepath.Join(file, testDirMapping[curTest]))
  132. } else {
  133. // otherwise assume test name
  134. files, err = getFiles(filepath.Join(file, curTest))
  135. }
  136. } else {
  137. files, err = getFiles(file)
  138. }
  139. if err != nil {
  140. suiteLogger.Crit("Failed to gather files", "error", err)
  141. }
  142. if len(files) == 0 {
  143. suiteLogger.Warn("No files matched path")
  144. }
  145. for _, curFile := range files {
  146. // Skip blank entries
  147. if len(curFile) == 0 {
  148. continue
  149. }
  150. testLogger := suiteLogger.New("file", curFile)
  151. r, err := os.Open(curFile)
  152. if err != nil {
  153. testLogger.Crit("Failed to open file")
  154. }
  155. defer r.Close()
  156. err = runTestWithReader(curTest, r)
  157. if err != nil {
  158. if continueOnError {
  159. testLogger.Error("Test failed, continuing", "error", err)
  160. } else {
  161. testLogger.Crit("Test failed, aborting", "error", err)
  162. }
  163. }
  164. }
  165. }
  166. }
  167. func setupApp(c *cli.Context) error {
  168. flagTest := c.GlobalString(TestFlag.Name)
  169. flagFile := c.GlobalString(FileFlag.Name)
  170. continueOnError = c.GlobalBool(ContinueOnErrorFlag.Name)
  171. useStdIn := c.GlobalBool(ReadStdInFlag.Name)
  172. skipTests = strings.Split(c.GlobalString(SkipTestsFlag.Name), " ")
  173. if !useStdIn {
  174. runSuite(flagTest, flagFile)
  175. } else {
  176. return runTestWithReader(flagTest, os.Stdin)
  177. }
  178. return nil
  179. }
  180. func main() {
  181. log.Root().SetHandler(log.StreamHandler(os.Stderr, log.TerminalFormat()))
  182. app := cli.NewApp()
  183. app.Name = "ethtest"
  184. app.Usage = "go-ethereum test interface"
  185. app.Action = setupApp
  186. app.Version = "0.2.0"
  187. app.Author = "go-ethereum team"
  188. app.Flags = []cli.Flag{
  189. TestFlag,
  190. FileFlag,
  191. ContinueOnErrorFlag,
  192. ReadStdInFlag,
  193. SkipTestsFlag,
  194. TraceFlag,
  195. }
  196. if err := app.Run(os.Args); err != nil {
  197. log.Crit("Failed to run the tester", "error", err)
  198. }
  199. }