/vendor/github.com/onsi/ginkgo/internal/suite/suite.go

https://github.com/backstage/backstage · Go · 183 lines · 150 code · 33 blank · 0 comment · 26 complexity · 50fe51b5510a9e735b5cdf41af4dcfbe MD5 · raw file

  1. package suite
  2. import (
  3. "math/rand"
  4. "net/http"
  5. "time"
  6. "github.com/onsi/ginkgo/internal/spec_iterator"
  7. "github.com/onsi/ginkgo/config"
  8. "github.com/onsi/ginkgo/internal/containernode"
  9. "github.com/onsi/ginkgo/internal/failer"
  10. "github.com/onsi/ginkgo/internal/leafnodes"
  11. "github.com/onsi/ginkgo/internal/spec"
  12. "github.com/onsi/ginkgo/internal/specrunner"
  13. "github.com/onsi/ginkgo/internal/writer"
  14. "github.com/onsi/ginkgo/reporters"
  15. "github.com/onsi/ginkgo/types"
  16. )
  17. type ginkgoTestingT interface {
  18. Fail()
  19. }
  20. type Suite struct {
  21. topLevelContainer *containernode.ContainerNode
  22. currentContainer *containernode.ContainerNode
  23. containerIndex int
  24. beforeSuiteNode leafnodes.SuiteNode
  25. afterSuiteNode leafnodes.SuiteNode
  26. runner *specrunner.SpecRunner
  27. failer *failer.Failer
  28. running bool
  29. }
  30. func New(failer *failer.Failer) *Suite {
  31. topLevelContainer := containernode.New("[Top Level]", types.FlagTypeNone, types.CodeLocation{})
  32. return &Suite{
  33. topLevelContainer: topLevelContainer,
  34. currentContainer: topLevelContainer,
  35. failer: failer,
  36. containerIndex: 1,
  37. }
  38. }
  39. func (suite *Suite) Run(t ginkgoTestingT, description string, reporters []reporters.Reporter, writer writer.WriterInterface, config config.GinkgoConfigType) (bool, bool) {
  40. if config.ParallelTotal < 1 {
  41. panic("ginkgo.parallel.total must be >= 1")
  42. }
  43. if config.ParallelNode > config.ParallelTotal || config.ParallelNode < 1 {
  44. panic("ginkgo.parallel.node is one-indexed and must be <= ginkgo.parallel.total")
  45. }
  46. r := rand.New(rand.NewSource(config.RandomSeed))
  47. suite.topLevelContainer.Shuffle(r)
  48. iterator, hasProgrammaticFocus := suite.generateSpecsIterator(description, config)
  49. suite.runner = specrunner.New(description, suite.beforeSuiteNode, iterator, suite.afterSuiteNode, reporters, writer, config)
  50. suite.running = true
  51. success := suite.runner.Run()
  52. if !success {
  53. t.Fail()
  54. }
  55. return success, hasProgrammaticFocus
  56. }
  57. func (suite *Suite) generateSpecsIterator(description string, config config.GinkgoConfigType) (spec_iterator.SpecIterator, bool) {
  58. specsSlice := []*spec.Spec{}
  59. suite.topLevelContainer.BackPropagateProgrammaticFocus()
  60. for _, collatedNodes := range suite.topLevelContainer.Collate() {
  61. specsSlice = append(specsSlice, spec.New(collatedNodes.Subject, collatedNodes.Containers, config.EmitSpecProgress))
  62. }
  63. specs := spec.NewSpecs(specsSlice)
  64. specs.RegexScansFilePath = config.RegexScansFilePath
  65. if config.RandomizeAllSpecs {
  66. specs.Shuffle(rand.New(rand.NewSource(config.RandomSeed)))
  67. }
  68. specs.ApplyFocus(description, config.FocusString, config.SkipString)
  69. if config.SkipMeasurements {
  70. specs.SkipMeasurements()
  71. }
  72. var iterator spec_iterator.SpecIterator
  73. if config.ParallelTotal > 1 {
  74. iterator = spec_iterator.NewParallelIterator(specs.Specs(), config.SyncHost)
  75. resp, err := http.Get(config.SyncHost + "/has-counter")
  76. if err != nil || resp.StatusCode != http.StatusOK {
  77. iterator = spec_iterator.NewShardedParallelIterator(specs.Specs(), config.ParallelTotal, config.ParallelNode)
  78. }
  79. } else {
  80. iterator = spec_iterator.NewSerialIterator(specs.Specs())
  81. }
  82. return iterator, specs.HasProgrammaticFocus()
  83. }
  84. func (suite *Suite) CurrentRunningSpecSummary() (*types.SpecSummary, bool) {
  85. return suite.runner.CurrentSpecSummary()
  86. }
  87. func (suite *Suite) SetBeforeSuiteNode(body interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  88. if suite.beforeSuiteNode != nil {
  89. panic("You may only call BeforeSuite once!")
  90. }
  91. suite.beforeSuiteNode = leafnodes.NewBeforeSuiteNode(body, codeLocation, timeout, suite.failer)
  92. }
  93. func (suite *Suite) SetAfterSuiteNode(body interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  94. if suite.afterSuiteNode != nil {
  95. panic("You may only call AfterSuite once!")
  96. }
  97. suite.afterSuiteNode = leafnodes.NewAfterSuiteNode(body, codeLocation, timeout, suite.failer)
  98. }
  99. func (suite *Suite) SetSynchronizedBeforeSuiteNode(bodyA interface{}, bodyB interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  100. if suite.beforeSuiteNode != nil {
  101. panic("You may only call BeforeSuite once!")
  102. }
  103. suite.beforeSuiteNode = leafnodes.NewSynchronizedBeforeSuiteNode(bodyA, bodyB, codeLocation, timeout, suite.failer)
  104. }
  105. func (suite *Suite) SetSynchronizedAfterSuiteNode(bodyA interface{}, bodyB interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  106. if suite.afterSuiteNode != nil {
  107. panic("You may only call AfterSuite once!")
  108. }
  109. suite.afterSuiteNode = leafnodes.NewSynchronizedAfterSuiteNode(bodyA, bodyB, codeLocation, timeout, suite.failer)
  110. }
  111. func (suite *Suite) PushContainerNode(text string, body func(), flag types.FlagType, codeLocation types.CodeLocation) {
  112. container := containernode.New(text, flag, codeLocation)
  113. suite.currentContainer.PushContainerNode(container)
  114. previousContainer := suite.currentContainer
  115. suite.currentContainer = container
  116. suite.containerIndex++
  117. body()
  118. suite.containerIndex--
  119. suite.currentContainer = previousContainer
  120. }
  121. func (suite *Suite) PushItNode(text string, body interface{}, flag types.FlagType, codeLocation types.CodeLocation, timeout time.Duration) {
  122. if suite.running {
  123. suite.failer.Fail("You may only call It from within a Describe or Context", codeLocation)
  124. }
  125. suite.currentContainer.PushSubjectNode(leafnodes.NewItNode(text, body, flag, codeLocation, timeout, suite.failer, suite.containerIndex))
  126. }
  127. func (suite *Suite) PushMeasureNode(text string, body interface{}, flag types.FlagType, codeLocation types.CodeLocation, samples int) {
  128. if suite.running {
  129. suite.failer.Fail("You may only call Measure from within a Describe or Context", codeLocation)
  130. }
  131. suite.currentContainer.PushSubjectNode(leafnodes.NewMeasureNode(text, body, flag, codeLocation, samples, suite.failer, suite.containerIndex))
  132. }
  133. func (suite *Suite) PushBeforeEachNode(body interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  134. if suite.running {
  135. suite.failer.Fail("You may only call BeforeEach from within a Describe or Context", codeLocation)
  136. }
  137. suite.currentContainer.PushSetupNode(leafnodes.NewBeforeEachNode(body, codeLocation, timeout, suite.failer, suite.containerIndex))
  138. }
  139. func (suite *Suite) PushJustBeforeEachNode(body interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  140. if suite.running {
  141. suite.failer.Fail("You may only call JustBeforeEach from within a Describe or Context", codeLocation)
  142. }
  143. suite.currentContainer.PushSetupNode(leafnodes.NewJustBeforeEachNode(body, codeLocation, timeout, suite.failer, suite.containerIndex))
  144. }
  145. func (suite *Suite) PushAfterEachNode(body interface{}, codeLocation types.CodeLocation, timeout time.Duration) {
  146. if suite.running {
  147. suite.failer.Fail("You may only call AfterEach from within a Describe or Context", codeLocation)
  148. }
  149. suite.currentContainer.PushSetupNode(leafnodes.NewAfterEachNode(body, codeLocation, timeout, suite.failer, suite.containerIndex))
  150. }