main.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  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/codegangsta/cli"
  26. "github.com/ethereum/go-ethereum/logger/glog"
  27. "github.com/ethereum/go-ethereum/params"
  28. "github.com/ethereum/go-ethereum/tests"
  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. glog.Infoln("runTest", test)
  68. var err error
  69. switch strings.ToLower(test) {
  70. case "bk", "block", "blocktest", "blockchaintest", "blocktests", "blockchaintests":
  71. err = tests.RunBlockTestWithReader(params.MainNetHomesteadBlock, r, skipTests)
  72. case "st", "state", "statetest", "statetests":
  73. err = tests.RunStateTestWithReader(tests.RuleSet{params.MainNetHomesteadBlock}, r, skipTests)
  74. case "tx", "transactiontest", "transactiontests":
  75. err = tests.RunTransactionTestsWithReader(r, skipTests)
  76. case "vm", "vmtest", "vmtests":
  77. err = tests.RunVmTestWithReader(r, skipTests)
  78. case "rlp", "rlptest", "rlptests":
  79. err = tests.RunRLPTestWithReader(r, skipTests)
  80. default:
  81. err = fmt.Errorf("Invalid test type specified: %v", test)
  82. }
  83. if err != nil {
  84. return err
  85. }
  86. return nil
  87. }
  88. func getFiles(path string) ([]string, error) {
  89. glog.Infoln("getFiles", path)
  90. var files []string
  91. f, err := os.Open(path)
  92. if err != nil {
  93. return nil, err
  94. }
  95. defer f.Close()
  96. fi, err := f.Stat()
  97. if err != nil {
  98. return nil, err
  99. }
  100. switch mode := fi.Mode(); {
  101. case mode.IsDir():
  102. fi, _ := ioutil.ReadDir(path)
  103. files = make([]string, len(fi))
  104. for i, v := range fi {
  105. // only go 1 depth and leave directory entires blank
  106. if !v.IsDir() && v.Name()[len(v.Name())-len(testExtension):len(v.Name())] == testExtension {
  107. files[i] = filepath.Join(path, v.Name())
  108. glog.Infoln("Found file", files[i])
  109. }
  110. }
  111. case mode.IsRegular():
  112. files = make([]string, 1)
  113. files[0] = path
  114. }
  115. return files, nil
  116. }
  117. func runSuite(test, file string) {
  118. var tests []string
  119. if test == defaultTest {
  120. tests = allTests
  121. } else {
  122. tests = []string{test}
  123. }
  124. for _, curTest := range tests {
  125. glog.Infoln("runSuite", curTest, file)
  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. glog.Fatalln(err)
  141. }
  142. if len(files) == 0 {
  143. glog.Warningln("No files matched path")
  144. }
  145. for _, curFile := range files {
  146. // Skip blank entries
  147. if len(curFile) == 0 {
  148. continue
  149. }
  150. r, err := os.Open(curFile)
  151. if err != nil {
  152. glog.Fatalln(err)
  153. }
  154. defer r.Close()
  155. err = runTestWithReader(curTest, r)
  156. if err != nil {
  157. if continueOnError {
  158. glog.Errorln(err)
  159. } else {
  160. glog.Fatalln(err)
  161. }
  162. }
  163. }
  164. }
  165. }
  166. func setupApp(c *cli.Context) {
  167. flagTest := c.GlobalString(TestFlag.Name)
  168. flagFile := c.GlobalString(FileFlag.Name)
  169. continueOnError = c.GlobalBool(ContinueOnErrorFlag.Name)
  170. useStdIn := c.GlobalBool(ReadStdInFlag.Name)
  171. skipTests = strings.Split(c.GlobalString(SkipTestsFlag.Name), " ")
  172. if !useStdIn {
  173. runSuite(flagTest, flagFile)
  174. } else {
  175. if err := runTestWithReader(flagTest, os.Stdin); err != nil {
  176. glog.Fatalln(err)
  177. }
  178. }
  179. }
  180. func main() {
  181. glog.SetToStderr(true)
  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. glog.Fatalln(err)
  198. }
  199. }