main.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  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. rs := tests.RuleSet{HomesteadBlock: params.MainNetHomesteadBlock}
  74. err = tests.RunStateTestWithReader(rs, r, skipTests)
  75. case "tx", "transactiontest", "transactiontests":
  76. err = tests.RunTransactionTestsWithReader(r, skipTests)
  77. case "vm", "vmtest", "vmtests":
  78. err = tests.RunVmTestWithReader(r, skipTests)
  79. case "rlp", "rlptest", "rlptests":
  80. err = tests.RunRLPTestWithReader(r, skipTests)
  81. default:
  82. err = fmt.Errorf("Invalid test type specified: %v", test)
  83. }
  84. if err != nil {
  85. return err
  86. }
  87. return nil
  88. }
  89. func getFiles(path string) ([]string, error) {
  90. glog.Infoln("getFiles", path)
  91. var files []string
  92. f, err := os.Open(path)
  93. if err != nil {
  94. return nil, err
  95. }
  96. defer f.Close()
  97. fi, err := f.Stat()
  98. if err != nil {
  99. return nil, err
  100. }
  101. switch mode := fi.Mode(); {
  102. case mode.IsDir():
  103. fi, _ := ioutil.ReadDir(path)
  104. files = make([]string, len(fi))
  105. for i, v := range fi {
  106. // only go 1 depth and leave directory entires blank
  107. if !v.IsDir() && v.Name()[len(v.Name())-len(testExtension):len(v.Name())] == testExtension {
  108. files[i] = filepath.Join(path, v.Name())
  109. glog.Infoln("Found file", files[i])
  110. }
  111. }
  112. case mode.IsRegular():
  113. files = make([]string, 1)
  114. files[0] = path
  115. }
  116. return files, nil
  117. }
  118. func runSuite(test, file string) {
  119. var tests []string
  120. if test == defaultTest {
  121. tests = allTests
  122. } else {
  123. tests = []string{test}
  124. }
  125. for _, curTest := range tests {
  126. glog.Infoln("runSuite", curTest, file)
  127. var err error
  128. var files []string
  129. if test == defaultTest {
  130. // check if we have an explicit directory mapping for the test
  131. if _, ok := testDirMapping[curTest]; ok {
  132. files, err = getFiles(filepath.Join(file, testDirMapping[curTest]))
  133. } else {
  134. // otherwise assume test name
  135. files, err = getFiles(filepath.Join(file, curTest))
  136. }
  137. } else {
  138. files, err = getFiles(file)
  139. }
  140. if err != nil {
  141. glog.Fatalln(err)
  142. }
  143. if len(files) == 0 {
  144. glog.Warningln("No files matched path")
  145. }
  146. for _, curFile := range files {
  147. // Skip blank entries
  148. if len(curFile) == 0 {
  149. continue
  150. }
  151. r, err := os.Open(curFile)
  152. if err != nil {
  153. glog.Fatalln(err)
  154. }
  155. defer r.Close()
  156. err = runTestWithReader(curTest, r)
  157. if err != nil {
  158. if continueOnError {
  159. glog.Errorln(err)
  160. } else {
  161. glog.Fatalln(err)
  162. }
  163. }
  164. }
  165. }
  166. }
  167. func setupApp(c *cli.Context) {
  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. if err := runTestWithReader(flagTest, os.Stdin); err != nil {
  177. glog.Fatalln(err)
  178. }
  179. }
  180. }
  181. func main() {
  182. glog.SetToStderr(true)
  183. app := cli.NewApp()
  184. app.Name = "ethtest"
  185. app.Usage = "go-ethereum test interface"
  186. app.Action = setupApp
  187. app.Version = "0.2.0"
  188. app.Author = "go-ethereum team"
  189. app.Flags = []cli.Flag{
  190. TestFlag,
  191. FileFlag,
  192. ContinueOnErrorFlag,
  193. ReadStdInFlag,
  194. SkipTestsFlag,
  195. TraceFlag,
  196. }
  197. if err := app.Run(os.Args); err != nil {
  198. glog.Fatalln(err)
  199. }
  200. }