Removed deprecated code
Этот коммит содержится в:
		
							родитель
							
								
									c7d739336b
								
							
						
					
					
						коммит
						dbd8cdf723
					
				
					 25 изменённых файлов: 309 добавлений и 1295 удалений
				
			
		
							
								
								
									
										11
									
								
								fmt.go
									
										
									
									
									
								
							
							
						
						
									
										11
									
								
								fmt.go
									
										
									
									
									
								
							|  | @ -75,17 +75,6 @@ type Formatter interface { | ||||||
| 	Summary() | 	Summary() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ConcurrentFormatter is an interface for a Concurrent |  | ||||||
| // version of the Formatter interface. |  | ||||||
| // |  | ||||||
| // Deprecated: The formatters need to handle concurrency |  | ||||||
| // instead of being copied and synchronized for each thread. |  | ||||||
| type ConcurrentFormatter interface { |  | ||||||
| 	Formatter |  | ||||||
| 	Copy(ConcurrentFormatter) |  | ||||||
| 	Sync(ConcurrentFormatter) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type storageFormatter interface { | type storageFormatter interface { | ||||||
| 	setStorage(*storage) | 	setStorage(*storage) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -60,13 +60,6 @@ func listFmtOutputTestsFeatureFiles() (featureFiles []string, err error) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func fmtOutputTest(fmtName, testName, featureFilePath string) func(*testing.T) { | func fmtOutputTest(fmtName, testName, featureFilePath string) func(*testing.T) { | ||||||
| 	fmtOutputSuiteInitializer := func(s *godog.Suite) { |  | ||||||
| 		s.Step(`^(?:a )?failing step`, failingStepDef) |  | ||||||
| 		s.Step(`^(?:a )?pending step$`, pendingStepDef) |  | ||||||
| 		s.Step(`^(?:a )?passing step$`, passingStepDef) |  | ||||||
| 		s.Step(`^odd (\d+) and even (\d+) number$`, oddEvenStepDef) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fmtOutputScenarioInitializer := func(ctx *godog.ScenarioContext) { | 	fmtOutputScenarioInitializer := func(ctx *godog.ScenarioContext) { | ||||||
| 		ctx.Step(`^(?:a )?failing step`, failingStepDef) | 		ctx.Step(`^(?:a )?failing step`, failingStepDef) | ||||||
| 		ctx.Step(`^(?:a )?pending step$`, pendingStepDef) | 		ctx.Step(`^(?:a )?pending step$`, pendingStepDef) | ||||||
|  | @ -93,22 +86,14 @@ func fmtOutputTest(fmtName, testName, featureFilePath string) func(*testing.T) { | ||||||
| 			Output: out, | 			Output: out, | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		godog.RunWithOptions(fmtName, fmtOutputSuiteInitializer, opts) |  | ||||||
| 
 |  | ||||||
| 		expected := string(expectedOutput) |  | ||||||
| 		actual := buf.String() |  | ||||||
| 		assert.Equalf(t, expected, actual, "path: %s", expectOutputPath) |  | ||||||
| 
 |  | ||||||
| 		buf.Reset() |  | ||||||
| 
 |  | ||||||
| 		godog.TestSuite{ | 		godog.TestSuite{ | ||||||
| 			Name:                fmtName, | 			Name:                fmtName, | ||||||
| 			ScenarioInitializer: fmtOutputScenarioInitializer, | 			ScenarioInitializer: fmtOutputScenarioInitializer, | ||||||
| 			Options:             &opts, | 			Options:             &opts, | ||||||
| 		}.Run() | 		}.Run() | ||||||
| 
 | 
 | ||||||
| 		expected = string(expectedOutput) | 		expected := string(expectedOutput) | ||||||
| 		actual = buf.String() | 		actual := buf.String() | ||||||
| 		assert.Equalf(t, expected, actual, "path: %s", expectOutputPath) | 		assert.Equalf(t, expected, actual, "path: %s", expectOutputPath) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -21,7 +21,7 @@ Feature: basic | ||||||
| 	Then two | 	Then two | ||||||
| ` | ` | ||||||
| 
 | 
 | ||||||
| func TestProgressFormatterWhenStepPanics(t *testing.T) { | func Test_ProgressFormatterWhenStepPanics(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -36,9 +36,9 @@ func TestProgressFormatterWhenStepPanics(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", w), | 		fmt:      progressFunc("progress", w), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two$`, func() error { panic("omg") }) | 			ctx.Step(`^two$`, func() error { panic("omg") }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -48,14 +48,14 @@ func TestProgressFormatterWhenStepPanics(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", w) }) | 	failed := r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| 
 | 
 | ||||||
| 	actual := buf.String() | 	actual := buf.String() | ||||||
| 	assert.Contains(t, actual, "godog/fmt_progress_test.go:41") | 	assert.Contains(t, actual, "godog/fmt_progress_test.go:41") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestProgressFormatterWithPanicInMultistep(t *testing.T) { | func Test_ProgressFormatterWithPanicInMultistep(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -70,12 +70,12 @@ func TestProgressFormatterWithPanicInMultistep(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", w), | 		fmt:      progressFunc("progress", w), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^sub1$`, func() error { return nil }) | 			ctx.Step(`^sub1$`, func() error { return nil }) | ||||||
| 			s.Step(`^sub-sub$`, func() error { return nil }) | 			ctx.Step(`^sub-sub$`, func() error { return nil }) | ||||||
| 			s.Step(`^sub2$`, func() []string { return []string{"sub-sub", "sub1", "one"} }) | 			ctx.Step(`^sub2$`, func() []string { return []string{"sub-sub", "sub1", "one"} }) | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two$`, func() []string { return []string{"sub1", "sub2"} }) | 			ctx.Step(`^two$`, func() []string { return []string{"sub1", "sub2"} }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -85,11 +85,11 @@ func TestProgressFormatterWithPanicInMultistep(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", w) }) | 	failed := r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestProgressFormatterMultistepTemplates(t *testing.T) { | func Test_ProgressFormatterMultistepTemplates(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -104,11 +104,11 @@ func TestProgressFormatterMultistepTemplates(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", w), | 		fmt:      progressFunc("progress", w), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^sub-sub$`, func() error { return nil }) | 			ctx.Step(`^sub-sub$`, func() error { return nil }) | ||||||
| 			s.Step(`^substep$`, func() Steps { return Steps{"sub-sub", `unavailable "John" cost 5`, "one", "three"} }) | 			ctx.Step(`^substep$`, func() Steps { return Steps{"sub-sub", `unavailable "John" cost 5`, "one", "three"} }) | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^(t)wo$`, func(s string) Steps { return Steps{"undef", "substep"} }) | 			ctx.Step(`^(t)wo$`, func(s string) Steps { return Steps{"undef", "substep"} }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -118,7 +118,7 @@ func TestProgressFormatterMultistepTemplates(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", w) }) | 	failed := r.concurrent(1) | ||||||
| 	require.False(t, failed) | 	require.False(t, failed) | ||||||
| 
 | 
 | ||||||
| 	expected := `.U 2 | 	expected := `.U 2 | ||||||
|  | @ -154,7 +154,7 @@ func FeatureContext(s *godog.Suite) { | ||||||
| 	assert.Equal(t, expected, actual) | 	assert.Equal(t, expected, actual) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestProgressFormatterWhenMultiStepHasArgument(t *testing.T) { | func Test_ProgressFormatterWhenMultiStepHasArgument(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	var featureSource = ` | 	var featureSource = ` | ||||||
|  | @ -180,9 +180,9 @@ Feature: basic | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", w), | 		fmt:      progressFunc("progress", w), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two:$`, func(doc *messages.PickleStepArgument_PickleDocString) Steps { return Steps{"one"} }) | 			ctx.Step(`^two:$`, func(doc *messages.PickleStepArgument_PickleDocString) Steps { return Steps{"one"} }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -192,11 +192,11 @@ Feature: basic | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", w) }) | 	failed := r.concurrent(1) | ||||||
| 	require.False(t, failed) | 	require.False(t, failed) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestProgressFormatterWhenMultiStepHasStepWithArgument(t *testing.T) { | func Test_ProgressFormatterWhenMultiStepHasStepWithArgument(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	var featureSource = ` | 	var featureSource = ` | ||||||
|  | @ -223,10 +223,10 @@ Feature: basic | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", w), | 		fmt:      progressFunc("progress", w), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two$`, func() Steps { return Steps{subStep} }) | 			ctx.Step(`^two$`, func() Steps { return Steps{subStep} }) | ||||||
| 			s.Step(`^three:$`, func(doc *messages.PickleStepArgument_PickleDocString) error { return nil }) | 			ctx.Step(`^three:$`, func(doc *messages.PickleStepArgument_PickleDocString) error { return nil }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -236,7 +236,7 @@ Feature: basic | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", w) }) | 	failed := r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| 
 | 
 | ||||||
| 	expected := `.F 2 | 	expected := `.F 2 | ||||||
|  |  | ||||||
|  | @ -48,7 +48,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 13, |                         "line": 13, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -60,7 +60,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 13, |                         "line": 13, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -72,7 +72,7 @@ | ||||||
|                         "name": "odd 1 and even 2 number", |                         "name": "odd 1 and even 2 number", | ||||||
|                         "line": 13, |                         "line": 13, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:118" |                             "location": "fmt_output_test.go:103" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -112,7 +112,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 14, |                         "line": 14, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -124,7 +124,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 14, |                         "line": 14, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -136,7 +136,7 @@ | ||||||
|                         "name": "odd 2 and even 0 number", |                         "name": "odd 2 and even 0 number", | ||||||
|                         "line": 14, |                         "line": 14, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:118" |                             "location": "fmt_output_test.go:103" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  | @ -177,7 +177,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 15, |                         "line": 15, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -189,7 +189,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 15, |                         "line": 15, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -201,7 +201,7 @@ | ||||||
|                         "name": "odd 3 and even 11 number", |                         "name": "odd 3 and even 11 number", | ||||||
|                         "line": 15, |                         "line": 15, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:118" |                             "location": "fmt_output_test.go:103" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  | @ -242,7 +242,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 20, |                         "line": 20, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -254,7 +254,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 20, |                         "line": 20, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -266,7 +266,7 @@ | ||||||
|                         "name": "odd 1 and even 14 number", |                         "name": "odd 1 and even 14 number", | ||||||
|                         "line": 20, |                         "line": 20, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:118" |                             "location": "fmt_output_test.go:103" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -306,7 +306,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 21, |                         "line": 21, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -318,7 +318,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 21, |                         "line": 21, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -330,7 +330,7 @@ | ||||||
|                         "name": "odd 3 and even 9 number", |                         "name": "odd 3 and even 9 number", | ||||||
|                         "line": 21, |                         "line": 21, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:118" |                             "location": "fmt_output_test.go:103" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 4, |                         "line": 4, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -32,7 +32,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 5, |                         "line": 5, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -44,7 +44,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 8, |                         "line": 8, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -56,7 +56,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 9, |                         "line": 9, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|                         "name": "a passing step", |                         "name": "a passing step", | ||||||
|                         "line": 7, |                         "line": 7, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 4, |                         "line": 4, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -32,7 +32,7 @@ | ||||||
|                         "name": "failing step", |                         "name": "failing step", | ||||||
|                         "line": 5, |                         "line": 5, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:132" |                             "location": "fmt_output_test.go:117" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  | @ -45,7 +45,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 6, |                         "line": 6, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  | @ -77,7 +77,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 10, |                         "line": 10, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  | @ -109,7 +109,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 14, |                         "line": 14, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 4, |                         "line": 4, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -32,7 +32,7 @@ | ||||||
|                         "name": "failing step", |                         "name": "failing step", | ||||||
|                         "line": 5, |                         "line": 5, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:132" |                             "location": "fmt_output_test.go:117" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  | @ -45,7 +45,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 8, |                         "line": 8, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  | @ -56,7 +56,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 9, |                         "line": 9, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  | @ -77,7 +77,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 4, |                         "line": 4, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "passed", |                             "status": "passed", | ||||||
|  | @ -89,7 +89,7 @@ | ||||||
|                         "name": "failing step", |                         "name": "failing step", | ||||||
|                         "line": 5, |                         "line": 5, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:132" |                             "location": "fmt_output_test.go:117" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "failed", |                             "status": "failed", | ||||||
|  | @ -102,7 +102,7 @@ | ||||||
|                         "name": "passing step", |                         "name": "passing step", | ||||||
|                         "line": 12, |                         "line": 12, | ||||||
|                         "match": { |                         "match": { | ||||||
|                             "location": "fmt_output_test.go:116" |                             "location": "fmt_output_test.go:101" | ||||||
|                         }, |                         }, | ||||||
|                         "result": { |                         "result": { | ||||||
|                             "status": "skipped" |                             "status": "skipped" | ||||||
|  |  | ||||||
|  | @ -1,57 +1,57 @@ | ||||||
| {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | ||||||
| {"event":"TestSource","location":"formatter-tests/features/scenario_outline.feature:2","source":"@outline @tag\nFeature: outline\n\n  @scenario\n  Scenario Outline: outline\n    Given passing step\n    When passing step\n    Then odd \u003codd\u003e and even \u003ceven\u003e number\n\n    @tagged\n    Examples: tagged\n      | odd | even |\n      | 1   | 2    |\n      | 2   | 0    |\n      | 3   | 11   |\n\n    @tag2\n    Examples:\n      | odd | even |\n      | 1   | 14   |\n      | 3   | 9    |\n"} | {"event":"TestSource","location":"formatter-tests/features/scenario_outline.feature:2","source":"@outline @tag\nFeature: outline\n\n  @scenario\n  Scenario Outline: outline\n    Given passing step\n    When passing step\n    Then odd \u003codd\u003e and even \u003ceven\u003e number\n\n    @tagged\n    Examples: tagged\n      | odd | even |\n      | 1   | 2    |\n      | 2   | 0    |\n      | 3   | 11   |\n\n    @tag2\n    Examples:\n      | odd | even |\n      | 1   | 14   |\n      | 3   | 9    |\n"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:13","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:13","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:118 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:103 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:13","timestamp":-6795364578871,"status":"passed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:13","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:14","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:14","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:118 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:103 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"2 is not odd"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"2 is not odd"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:14","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:14","timestamp":-6795364578871,"status":"failed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:15","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:15","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:118 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:103 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"11 is not even"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"11 is not even"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:15","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:15","timestamp":-6795364578871,"status":"failed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:20","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:20","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:118 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:103 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:20","timestamp":-6795364578871,"status":"passed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:20","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:21","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_outline.feature:21","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:118 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_outline.feature:8","definition_id":"fmt_output_test.go:103 -\u003e github.com/cucumber/godog_test.oddEvenStepDef","arguments":[[4,5],[5,15]]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"9 is not even"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_outline.feature:8","timestamp":-6795364578871,"status":"failed","summary":"9 is not even"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:21","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_outline.feature:21","timestamp":-6795364578871,"status":"failed"} | ||||||
|  |  | ||||||
|  | @ -1,16 +1,16 @@ | ||||||
| {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | ||||||
| {"event":"TestSource","location":"formatter-tests/features/scenario_with_background.feature:1","source":"Feature: single scenario with background\n\n  Background: named\n    Given passing step\n    And passing step\n\n  Scenario: scenario\n    When passing step\n    Then passing step\n"} | {"event":"TestSource","location":"formatter-tests/features/scenario_with_background.feature:1","source":"Feature: single scenario with background\n\n  Background: named\n    Given passing step\n    And passing step\n\n  Scenario: scenario\n    When passing step\n    Then passing step\n"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_with_background.feature:7","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/scenario_with_background.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:4","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:4","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:4","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:4","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:4","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:4","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:5","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:5","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:5","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:5","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:5","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:5","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:8","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:8","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:8","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:8","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:9","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/scenario_with_background.feature:9","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:9","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/scenario_with_background.feature:9","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:9","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/scenario_with_background.feature:9","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_with_background.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/scenario_with_background.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | ||||||
| {"event":"TestSource","location":"formatter-tests/features/single_scenario_with_passing_step.feature:1","source":"Feature: single passing scenario\n  describes\n  a single scenario\n  feature\n\n  Scenario: one step passing\n    Given a passing step\n"} | {"event":"TestSource","location":"formatter-tests/features/single_scenario_with_passing_step.feature:1","source":"Feature: single passing scenario\n  describes\n  a single scenario\n  feature\n\n  Scenario: one step passing\n    Given a passing step\n"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/single_scenario_with_passing_step.feature:6","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/single_scenario_with_passing_step.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/single_scenario_with_passing_step.feature:7","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/single_scenario_with_passing_step.feature:6","timestamp":-6795364578871,"status":"passed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/single_scenario_with_passing_step.feature:6","timestamp":-6795364578871,"status":"passed"} | ||||||
|  |  | ||||||
|  | @ -1,28 +1,28 @@ | ||||||
| {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | ||||||
| {"event":"TestSource","location":"formatter-tests/features/some_scenarions_including_failing.feature:1","source":"Feature: some scenarios\n\n  Scenario: failing\n    Given passing step\n    When failing step\n    Then passing step\n\n  Scenario: pending\n    When pending step\n    Then passing step\n\n  Scenario: undefined\n    When undefined\n    Then passing step\n"} | {"event":"TestSource","location":"formatter-tests/features/some_scenarions_including_failing.feature:1","source":"Feature: some scenarios\n\n  Scenario: failing\n    Given passing step\n    When failing step\n    Then passing step\n\n  Scenario: pending\n    When pending step\n    Then passing step\n\n  Scenario: undefined\n    When undefined\n    Then passing step\n"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:3","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:3","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:4","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","definition_id":"fmt_output_test.go:132 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","definition_id":"fmt_output_test.go:117 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:6","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:3","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:3","timestamp":-6795364578871,"status":"failed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:8","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","definition_id":"fmt_output_test.go:130 -\u003e github.com/cucumber/godog_test.pendingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","definition_id":"fmt_output_test.go:115 -\u003e github.com/cucumber/godog_test.pendingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","timestamp":-6795364578871,"status":"pending"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:9","timestamp":-6795364578871,"status":"pending"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:10","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:8","timestamp":-6795364578871,"status":"pending"} | {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:8","timestamp":-6795364578871,"status":"pending"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:12","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:12","timestamp":-6795364578871} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:13","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:13","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:13","timestamp":-6795364578871,"status":"undefined"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:13","timestamp":-6795364578871,"status":"undefined"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:14","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:12","timestamp":-6795364578871,"status":"undefined"} | {"event":"TestCaseFinished","location":"formatter-tests/features/some_scenarions_including_failing.feature:12","timestamp":-6795364578871,"status":"undefined"} | ||||||
|  |  | ||||||
|  | @ -1,27 +1,27 @@ | ||||||
| {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | {"event":"TestRunStarted","version":"0.1.0","timestamp":-6795364578871,"suite":"events"} | ||||||
| {"event":"TestSource","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:1","source":"Feature: two scenarios with background fail\n\n  Background:\n    Given passing step\n    And failing step\n\n  Scenario: one\n    When passing step\n    Then passing step\n\n  Scenario: two\n    Then passing step\n"} | {"event":"TestSource","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:1","source":"Feature: two scenarios with background fail\n\n  Background:\n    Given passing step\n    And failing step\n\n  Scenario: one\n    When passing step\n    Then passing step\n\n  Scenario: two\n    Then passing step\n"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:7","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:7","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","definition_id":"fmt_output_test.go:132 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","definition_id":"fmt_output_test.go:117 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:8","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:9","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:7","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:7","timestamp":-6795364578871,"status":"failed"} | ||||||
| {"event":"TestCaseStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:11","timestamp":-6795364578871} | {"event":"TestCaseStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:11","timestamp":-6795364578871} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871,"status":"passed"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:4","timestamp":-6795364578871,"status":"passed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","definition_id":"fmt_output_test.go:132 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","definition_id":"fmt_output_test.go:117 -\u003e github.com/cucumber/godog_test.failingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:5","timestamp":-6795364578871,"status":"failed","summary":"step failed"} | ||||||
| {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","definition_id":"fmt_output_test.go:116 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | {"event":"StepDefinitionFound","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","definition_id":"fmt_output_test.go:101 -\u003e github.com/cucumber/godog_test.passingStepDef","arguments":[]} | ||||||
| {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","timestamp":-6795364578871} | {"event":"TestStepStarted","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","timestamp":-6795364578871} | ||||||
| {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","timestamp":-6795364578871,"status":"skipped"} | {"event":"TestStepFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:12","timestamp":-6795364578871,"status":"skipped"} | ||||||
| {"event":"TestCaseFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:11","timestamp":-6795364578871,"status":"failed"} | {"event":"TestCaseFinished","location":"formatter-tests/features/two_scenarios_with_background_fail.feature:11","timestamp":-6795364578871,"status":"failed"} | ||||||
|  |  | ||||||
|  | @ -1,9 +1,9 @@ | ||||||
| <bold-white>Feature:</bold-white> outline | <bold-white>Feature:</bold-white> outline | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario Outline:</bold-white> outline               <bold-black># formatter-tests/features/scenario_outline.feature:5</bold-black> |   <bold-white>Scenario Outline:</bold-white> outline               <bold-black># formatter-tests/features/scenario_outline.feature:5</bold-black> | ||||||
|     <cyan>Given</cyan> <cyan>passing step</cyan>                    <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Given</cyan> <cyan>passing step</cyan>                    <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <cyan>When</cyan> <cyan>passing step</cyan>                     <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>When</cyan> <cyan>passing step</cyan>                     <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <cyan>Then</cyan> <cyan>odd </cyan><bold-cyan><odd></bold-cyan><cyan> and even </cyan><bold-cyan><even></bold-cyan><cyan> number</cyan> <bold-black># fmt_output_test.go:118 -> github.com/cucumber/godog_test.oddEvenStepDef</bold-black> |     <cyan>Then</cyan> <cyan>odd </cyan><bold-cyan><odd></bold-cyan><cyan> and even </cyan><bold-cyan><even></bold-cyan><cyan> number</cyan> <bold-black># fmt_output_test.go:103 -> github.com/cucumber/godog_test.oddEvenStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
|     <bold-white>Examples:</bold-white> tagged |     <bold-white>Examples:</bold-white> tagged | ||||||
|       | <cyan>odd</cyan> | <cyan>even</cyan> | |       | <cyan>odd</cyan> | <cyan>even</cyan> | | ||||||
|  |  | ||||||
|  | @ -1,12 +1,12 @@ | ||||||
| <bold-white>Feature:</bold-white> single scenario with background | <bold-white>Feature:</bold-white> single scenario with background | ||||||
| 
 | 
 | ||||||
|   <bold-white>Background:</bold-white> named |   <bold-white>Background:</bold-white> named | ||||||
|     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <green>And</green> <green>passing step</green>   <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>And</green> <green>passing step</green>   <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> scenario   <bold-black># formatter-tests/features/scenario_with_background.feature:7</bold-black> |   <bold-white>Scenario:</bold-white> scenario   <bold-black># formatter-tests/features/scenario_with_background.feature:7</bold-black> | ||||||
|     <green>When</green> <green>passing step</green>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>When</green> <green>passing step</green>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <green>Then</green> <green>passing step</green>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>Then</green> <green>passing step</green>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
| 1 scenarios (<green>1 passed</green>) | 1 scenarios (<green>1 passed</green>) | ||||||
| 4 steps (<green>4 passed</green>) | 4 steps (<green>4 passed</green>) | ||||||
|  |  | ||||||
|  | @ -4,7 +4,7 @@ | ||||||
|   feature |   feature | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> one step passing <bold-black># formatter-tests/features/single_scenario_with_passing_step.feature:6</bold-black> |   <bold-white>Scenario:</bold-white> one step passing <bold-black># formatter-tests/features/single_scenario_with_passing_step.feature:6</bold-black> | ||||||
|     <green>Given</green> <green>a passing step</green>     <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>Given</green> <green>a passing step</green>     <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
| 1 scenarios (<green>1 passed</green>) | 1 scenarios (<green>1 passed</green>) | ||||||
| 1 steps (<green>1 passed</green>) | 1 steps (<green>1 passed</green>) | ||||||
|  |  | ||||||
|  | @ -1,19 +1,19 @@ | ||||||
| <bold-white>Feature:</bold-white> some scenarios | <bold-white>Feature:</bold-white> some scenarios | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> failing    <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:3</bold-black> |   <bold-white>Scenario:</bold-white> failing    <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:3</bold-black> | ||||||
|     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <red>When</red> <red>failing step</red>  <bold-black># fmt_output_test.go:132 -> github.com/cucumber/godog_test.failingStepDef</bold-black> |     <red>When</red> <red>failing step</red>  <bold-black># fmt_output_test.go:117 -> github.com/cucumber/godog_test.failingStepDef</bold-black> | ||||||
|     <bold-red>step failed</bold-red> |     <bold-red>step failed</bold-red> | ||||||
|     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> pending   <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:8</bold-black> |   <bold-white>Scenario:</bold-white> pending   <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:8</bold-black> | ||||||
|     <yellow>When</yellow> <yellow>pending step</yellow> <bold-black># fmt_output_test.go:130 -> github.com/cucumber/godog_test.pendingStepDef</bold-black> |     <yellow>When</yellow> <yellow>pending step</yellow> <bold-black># fmt_output_test.go:115 -> github.com/cucumber/godog_test.pendingStepDef</bold-black> | ||||||
|       <yellow>TODO: write pending definition</yellow> |       <yellow>TODO: write pending definition</yellow> | ||||||
|     <cyan>Then</cyan> <cyan>passing step</cyan> <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Then</cyan> <cyan>passing step</cyan> <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> undefined <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:12</bold-black> |   <bold-white>Scenario:</bold-white> undefined <bold-black># formatter-tests/features/some_scenarions_including_failing.feature:12</bold-black> | ||||||
|     <yellow>When</yellow> <yellow>undefined</yellow> |     <yellow>When</yellow> <yellow>undefined</yellow> | ||||||
|     <cyan>Then</cyan> <cyan>passing step</cyan> <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Then</cyan> <cyan>passing step</cyan> <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
| --- <red>Failed steps:</red> | --- <red>Failed steps:</red> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,16 +1,16 @@ | ||||||
| <bold-white>Feature:</bold-white> two scenarios with background fail | <bold-white>Feature:</bold-white> two scenarios with background fail | ||||||
| 
 | 
 | ||||||
|   <bold-white>Background:</bold-white> |   <bold-white>Background:</bold-white> | ||||||
|     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <green>Given</green> <green>passing step</green> <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <red>And</red> <red>failing step</red>   <bold-black># fmt_output_test.go:132 -> github.com/cucumber/godog_test.failingStepDef</bold-black> |     <red>And</red> <red>failing step</red>   <bold-black># fmt_output_test.go:117 -> github.com/cucumber/godog_test.failingStepDef</bold-black> | ||||||
|     <bold-red>step failed</bold-red> |     <bold-red>step failed</bold-red> | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> one        <bold-black># formatter-tests/features/two_scenarios_with_background_fail.feature:7</bold-black> |   <bold-white>Scenario:</bold-white> one        <bold-black># formatter-tests/features/two_scenarios_with_background_fail.feature:7</bold-black> | ||||||
|     <cyan>When</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>When</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
|     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
|   <bold-white>Scenario:</bold-white> two        <bold-black># formatter-tests/features/two_scenarios_with_background_fail.feature:11</bold-black> |   <bold-white>Scenario:</bold-white> two        <bold-black># formatter-tests/features/two_scenarios_with_background_fail.feature:11</bold-black> | ||||||
|     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:116 -> github.com/cucumber/godog_test.passingStepDef</bold-black> |     <cyan>Then</cyan> <cyan>passing step</cyan>  <bold-black># fmt_output_test.go:101 -> github.com/cucumber/godog_test.passingStepDef</bold-black> | ||||||
| 
 | 
 | ||||||
| --- <red>Failed steps:</red> | --- <red>Failed steps:</red> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										196
									
								
								run.go
									
										
									
									
									
								
							
							
						
						
									
										196
									
								
								run.go
									
										
									
									
									
								
							|  | @ -21,7 +21,6 @@ const ( | ||||||
| 	exitOptionError | 	exitOptionError | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type initializer func(*Suite) |  | ||||||
| type testSuiteInitializer func(*TestSuiteContext) | type testSuiteInitializer func(*TestSuiteContext) | ||||||
| type scenarioInitializer func(*ScenarioContext) | type scenarioInitializer func(*ScenarioContext) | ||||||
| 
 | 
 | ||||||
|  | @ -31,7 +30,6 @@ type runner struct { | ||||||
| 
 | 
 | ||||||
| 	features []*feature | 	features []*feature | ||||||
| 
 | 
 | ||||||
| 	initializer          initializer |  | ||||||
| 	testSuiteInitializer testSuiteInitializer | 	testSuiteInitializer testSuiteInitializer | ||||||
| 	scenarioInitializer  scenarioInitializer | 	scenarioInitializer  scenarioInitializer | ||||||
| 
 | 
 | ||||||
|  | @ -39,104 +37,7 @@ type runner struct { | ||||||
| 	fmt     Formatter | 	fmt     Formatter | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (r *runner) concurrent(rate int, formatterFn func() Formatter) (failed bool) { | func (r *runner) concurrent(rate int) (failed bool) { | ||||||
| 	var useFmtCopy bool |  | ||||||
| 	var copyLock sync.Mutex |  | ||||||
| 
 |  | ||||||
| 	// special mode for concurrent-formatter |  | ||||||
| 	if _, ok := r.fmt.(ConcurrentFormatter); ok { |  | ||||||
| 		useFmtCopy = true |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if fmt, ok := r.fmt.(storageFormatter); ok { |  | ||||||
| 		fmt.setStorage(r.storage) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	testRunStarted := testRunStarted{StartedAt: timeNowFunc()} |  | ||||||
| 	r.storage.mustInsertTestRunStarted(testRunStarted) |  | ||||||
| 	r.fmt.TestRunStarted() |  | ||||||
| 
 |  | ||||||
| 	queue := make(chan int, rate) |  | ||||||
| 	for i, ft := range r.features { |  | ||||||
| 		queue <- i // reserve space in queue |  | ||||||
| 		ft := *ft |  | ||||||
| 
 |  | ||||||
| 		go func(fail *bool, feat *feature) { |  | ||||||
| 			var fmtCopy Formatter |  | ||||||
| 
 |  | ||||||
| 			defer func() { |  | ||||||
| 				<-queue // free a space in queue |  | ||||||
| 			}() |  | ||||||
| 
 |  | ||||||
| 			if r.stopOnFailure && *fail { |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			suite := &Suite{ |  | ||||||
| 				fmt:           r.fmt, |  | ||||||
| 				randomSeed:    r.randomSeed, |  | ||||||
| 				stopOnFailure: r.stopOnFailure, |  | ||||||
| 				strict:        r.strict, |  | ||||||
| 				features:      []*feature{feat}, |  | ||||||
| 				storage:       r.storage, |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			if useFmtCopy { |  | ||||||
| 				fmtCopy = formatterFn() |  | ||||||
| 				suite.fmt = fmtCopy |  | ||||||
| 
 |  | ||||||
| 				concurrentDestFmt, dOk := fmtCopy.(ConcurrentFormatter) |  | ||||||
| 				concurrentSourceFmt, sOk := r.fmt.(ConcurrentFormatter) |  | ||||||
| 
 |  | ||||||
| 				if dOk && sOk { |  | ||||||
| 					concurrentDestFmt.Sync(concurrentSourceFmt) |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			if fmt, ok := suite.fmt.(storageFormatter); ok { |  | ||||||
| 				fmt.setStorage(r.storage) |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			r.initializer(suite) |  | ||||||
| 
 |  | ||||||
| 			suite.run() |  | ||||||
| 
 |  | ||||||
| 			if suite.failed { |  | ||||||
| 				copyLock.Lock() |  | ||||||
| 				*fail = true |  | ||||||
| 				copyLock.Unlock() |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			if useFmtCopy { |  | ||||||
| 				copyLock.Lock() |  | ||||||
| 
 |  | ||||||
| 				concurrentDestFmt, dOk := r.fmt.(ConcurrentFormatter) |  | ||||||
| 				concurrentSourceFmt, sOk := fmtCopy.(ConcurrentFormatter) |  | ||||||
| 
 |  | ||||||
| 				if dOk && sOk { |  | ||||||
| 					concurrentDestFmt.Copy(concurrentSourceFmt) |  | ||||||
| 				} else if !dOk { |  | ||||||
| 					panic("cant cast dest formatter to progress-typed") |  | ||||||
| 				} else if !sOk { |  | ||||||
| 					panic("cant cast source formatter to progress-typed") |  | ||||||
| 				} |  | ||||||
| 
 |  | ||||||
| 				copyLock.Unlock() |  | ||||||
| 			} |  | ||||||
| 		}(&failed, &ft) |  | ||||||
| 	} |  | ||||||
| 	// wait until last are processed |  | ||||||
| 	for i := 0; i < rate; i++ { |  | ||||||
| 		queue <- i |  | ||||||
| 	} |  | ||||||
| 	close(queue) |  | ||||||
| 
 |  | ||||||
| 	// print summary |  | ||||||
| 	r.fmt.Summary() |  | ||||||
| 	return |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (r *runner) scenarioConcurrent(rate int) (failed bool) { |  | ||||||
| 	var copyLock sync.Mutex | 	var copyLock sync.Mutex | ||||||
| 
 | 
 | ||||||
| 	if fmt, ok := r.fmt.(storageFormatter); ok { | 	if fmt, ok := r.fmt.(storageFormatter); ok { | ||||||
|  | @ -188,7 +89,7 @@ func (r *runner) scenarioConcurrent(rate int) (failed bool) { | ||||||
| 					return | 					return | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				suite := &Suite{ | 				suite := &suite{ | ||||||
| 					fmt:        r.fmt, | 					fmt:        r.fmt, | ||||||
| 					randomSeed: r.randomSeed, | 					randomSeed: r.randomSeed, | ||||||
| 					strict:     r.strict, | 					strict:     r.strict, | ||||||
|  | @ -227,38 +128,7 @@ func (r *runner) scenarioConcurrent(rate int) (failed bool) { | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // RunWithOptions is same as Run function, except | func runWithOptions(suiteName string, runner runner, opt Options) int { | ||||||
| // it uses Options provided in order to run the |  | ||||||
| // test suite without parsing flags |  | ||||||
| // |  | ||||||
| // This method is useful in case if you run |  | ||||||
| // godog in for example TestMain function together |  | ||||||
| // with go tests |  | ||||||
| // |  | ||||||
| // The exit codes may vary from: |  | ||||||
| //  0 - success |  | ||||||
| //  1 - failed |  | ||||||
| //  2 - command line usage error |  | ||||||
| //  128 - or higher, os signal related error exit codes |  | ||||||
| // |  | ||||||
| // If there are flag related errors they will |  | ||||||
| // be directed to os.Stderr |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. |  | ||||||
| // Use: |  | ||||||
| //   godog.TestSuite{ |  | ||||||
| //     Name: name, |  | ||||||
| //     TestSuiteInitializer: testSuiteInitializer, |  | ||||||
| //     ScenarioInitializer: scenarioInitializer, |  | ||||||
| //     Options: &opts, |  | ||||||
| //   }.Run() |  | ||||||
| // instead. |  | ||||||
| func RunWithOptions(suite string, initializer func(*Suite), opt Options) int { |  | ||||||
| 	return runWithOptions(suite, runner{initializer: initializer}, opt) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func runWithOptions(suite string, runner runner, opt Options) int { |  | ||||||
| 	var output io.Writer = os.Stdout | 	var output io.Writer = os.Stdout | ||||||
| 	if nil != opt.Output { | 	if nil != opt.Output { | ||||||
| 		output = opt.Output | 		output = opt.Output | ||||||
|  | @ -271,8 +141,9 @@ func runWithOptions(suite string, runner runner, opt Options) int { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if opt.ShowStepDefinitions { | 	if opt.ShowStepDefinitions { | ||||||
| 		s := &Suite{} | 		s := suite{} | ||||||
| 		runner.initializer(s) | 		sc := ScenarioContext{suite: &s} | ||||||
|  | 		runner.scenarioInitializer(&sc) | ||||||
| 		printStepDefinitions(s.steps, output) | 		printStepDefinitions(s.steps, output) | ||||||
| 		return exitOptionError | 		return exitOptionError | ||||||
| 	} | 	} | ||||||
|  | @ -301,7 +172,7 @@ func runWithOptions(suite string, runner runner, opt Options) int { | ||||||
| 		)) | 		)) | ||||||
| 		return exitOptionError | 		return exitOptionError | ||||||
| 	} | 	} | ||||||
| 	runner.fmt = formatter(suite, output) | 	runner.fmt = formatter(suiteName, output) | ||||||
| 
 | 
 | ||||||
| 	var err error | 	var err error | ||||||
| 	if runner.features, err = parseFeatures(opt.Tags, opt.Paths); err != nil { | 	if runner.features, err = parseFeatures(opt.Tags, opt.Paths); err != nil { | ||||||
|  | @ -333,12 +204,7 @@ func runWithOptions(suite string, runner runner, opt Options) int { | ||||||
| 	_, filename, _, _ := runtime.Caller(1) | 	_, filename, _, _ := runtime.Caller(1) | ||||||
| 	os.Setenv("GODOG_TESTED_PACKAGE", runsFromPackage(filename)) | 	os.Setenv("GODOG_TESTED_PACKAGE", runsFromPackage(filename)) | ||||||
| 
 | 
 | ||||||
| 	var failed bool | 	failed := runner.concurrent(opt.Concurrency) | ||||||
| 	if runner.initializer != nil { |  | ||||||
| 		failed = runner.concurrent(opt.Concurrency, func() Formatter { return formatter(suite, output) }) |  | ||||||
| 	} else { |  | ||||||
| 		failed = runner.scenarioConcurrent(opt.Concurrency) |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	// @TODO: should prevent from having these | 	// @TODO: should prevent from having these | ||||||
| 	os.Setenv("GODOG_SEED", "") | 	os.Setenv("GODOG_SEED", "") | ||||||
|  | @ -360,52 +226,6 @@ func runsFromPackage(fp string) string { | ||||||
| 	return dir | 	return dir | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Run creates and runs the feature suite. |  | ||||||
| // Reads all configuration options from flags. |  | ||||||
| // uses contextInitializer to register contexts |  | ||||||
| // |  | ||||||
| // the concurrency option allows runner to |  | ||||||
| // initialize a number of suites to be run |  | ||||||
| // separately. Only progress formatter |  | ||||||
| // is supported when concurrency level is |  | ||||||
| // higher than 1 |  | ||||||
| // |  | ||||||
| // contextInitializer must be able to register |  | ||||||
| // the step definitions and event handlers. |  | ||||||
| // |  | ||||||
| // The exit codes may vary from: |  | ||||||
| //  0 - success |  | ||||||
| //  1 - failed |  | ||||||
| //  2 - command line usage error |  | ||||||
| //  128 - or higher, os signal related error exit codes |  | ||||||
| // |  | ||||||
| // If there are flag related errors they will |  | ||||||
| // be directed to os.Stderr |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. |  | ||||||
| // Use: |  | ||||||
| //   godog.TestSuite{ |  | ||||||
| //     Name: name, |  | ||||||
| //     TestSuiteInitializer: testSuiteInitializer, |  | ||||||
| //     ScenarioInitializer: scenarioInitializer, |  | ||||||
| //   }.Run() |  | ||||||
| // instead. |  | ||||||
| func Run(suite string, initializer func(*Suite)) int { |  | ||||||
| 	var opt Options |  | ||||||
| 	opt.Output = colors.Colored(os.Stdout) |  | ||||||
| 
 |  | ||||||
| 	flagSet := FlagSet(&opt) |  | ||||||
| 	if err := flagSet.Parse(os.Args[1:]); err != nil { |  | ||||||
| 		fmt.Fprintln(os.Stderr, err) |  | ||||||
| 		return exitOptionError |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	opt.Paths = flagSet.Args() |  | ||||||
| 
 |  | ||||||
| 	return RunWithOptions(suite, initializer, opt) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // TestSuite allows for configuration | // TestSuite allows for configuration | ||||||
| // of the Test Suite Execution | // of the Test Suite Execution | ||||||
| type TestSuite struct { | type TestSuite struct { | ||||||
|  |  | ||||||
							
								
								
									
										142
									
								
								run_test.go
									
										
									
									
									
								
							
							
						
						
									
										142
									
								
								run_test.go
									
										
									
									
									
								
							|  | @ -26,7 +26,8 @@ func okStep() error { | ||||||
| func TestPrintsStepDefinitions(t *testing.T) { | func TestPrintsStepDefinitions(t *testing.T) { | ||||||
| 	var buf bytes.Buffer | 	var buf bytes.Buffer | ||||||
| 	w := colors.Uncolored(&buf) | 	w := colors.Uncolored(&buf) | ||||||
| 	s := &Suite{} | 	s := suite{} | ||||||
|  | 	ctx := ScenarioContext{suite: &s} | ||||||
| 
 | 
 | ||||||
| 	steps := []string{ | 	steps := []string{ | ||||||
| 		"^passing step$", | 		"^passing step$", | ||||||
|  | @ -34,7 +35,7 @@ func TestPrintsStepDefinitions(t *testing.T) { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for _, step := range steps { | 	for _, step := range steps { | ||||||
| 		s.Step(step, okStep) | 		ctx.Step(step, okStep) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	printStepDefinitions(s.steps, w) | 	printStepDefinitions(s.steps, w) | ||||||
|  | @ -54,7 +55,7 @@ func TestPrintsStepDefinitions(t *testing.T) { | ||||||
| func TestPrintsNoStepDefinitionsIfNoneFound(t *testing.T) { | func TestPrintsNoStepDefinitionsIfNoneFound(t *testing.T) { | ||||||
| 	var buf bytes.Buffer | 	var buf bytes.Buffer | ||||||
| 	w := colors.Uncolored(&buf) | 	w := colors.Uncolored(&buf) | ||||||
| 	s := &Suite{} | 	s := &suite{} | ||||||
| 
 | 
 | ||||||
| 	printStepDefinitions(s.steps, w) | 	printStepDefinitions(s.steps, w) | ||||||
| 
 | 
 | ||||||
|  | @ -62,7 +63,7 @@ func TestPrintsNoStepDefinitionsIfNoneFound(t *testing.T) { | ||||||
| 	assert.Equal(t, "there were no contexts registered, could not find any step definition..", out) | 	assert.Equal(t, "there were no contexts registered, could not find any step definition..", out) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFailsOrPassesBasedOnStrictModeWhenHasPendingSteps(t *testing.T) { | func Test_FailsOrPassesBasedOnStrictModeWhenHasPendingSteps(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -75,9 +76,9 @@ func TestFailsOrPassesBasedOnStrictModeWhenHasPendingSteps(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", ioutil.Discard), | 		fmt:      progressFunc("progress", ioutil.Discard), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two$`, func() error { return ErrPending }) | 			ctx.Step(`^two$`, func() error { return ErrPending }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -87,15 +88,15 @@ func TestFailsOrPassesBasedOnStrictModeWhenHasPendingSteps(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", ioutil.Discard) }) | 	failed := r.concurrent(1) | ||||||
| 	require.False(t, failed) | 	require.False(t, failed) | ||||||
| 
 | 
 | ||||||
| 	r.strict = true | 	r.strict = true | ||||||
| 	failed = r.concurrent(1, func() Formatter { return progressFunc("progress", ioutil.Discard) }) | 	failed = r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFailsOrPassesBasedOnStrictModeWhenHasUndefinedSteps(t *testing.T) { | func Test_FailsOrPassesBasedOnStrictModeWhenHasUndefinedSteps(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -108,8 +109,8 @@ func TestFailsOrPassesBasedOnStrictModeWhenHasUndefinedSteps(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", ioutil.Discard), | 		fmt:      progressFunc("progress", ioutil.Discard), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			// two - is undefined | 			// two - is undefined | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  | @ -120,15 +121,15 @@ func TestFailsOrPassesBasedOnStrictModeWhenHasUndefinedSteps(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", ioutil.Discard) }) | 	failed := r.concurrent(1) | ||||||
| 	require.False(t, failed) | 	require.False(t, failed) | ||||||
| 
 | 
 | ||||||
| 	r.strict = true | 	r.strict = true | ||||||
| 	failed = r.concurrent(1, func() Formatter { return progressFunc("progress", ioutil.Discard) }) | 	failed = r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestShouldFailOnError(t *testing.T) { | func Test_ShouldFailOnError(t *testing.T) { | ||||||
| 	const path = "any.feature" | 	const path = "any.feature" | ||||||
| 
 | 
 | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(basicGherkinFeature), (&messages.Incrementing{}).NewId) | ||||||
|  | @ -141,9 +142,9 @@ func TestShouldFailOnError(t *testing.T) { | ||||||
| 	r := runner{ | 	r := runner{ | ||||||
| 		fmt:      progressFunc("progress", ioutil.Discard), | 		fmt:      progressFunc("progress", ioutil.Discard), | ||||||
| 		features: []*feature{&ft}, | 		features: []*feature{&ft}, | ||||||
| 		initializer: func(s *Suite) { | 		scenarioInitializer: func(ctx *ScenarioContext) { | ||||||
| 			s.Step(`^one$`, func() error { return nil }) | 			ctx.Step(`^one$`, func() error { return nil }) | ||||||
| 			s.Step(`^two$`, func() error { return fmt.Errorf("error") }) | 			ctx.Step(`^two$`, func() error { return fmt.Errorf("error") }) | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -153,22 +154,27 @@ func TestShouldFailOnError(t *testing.T) { | ||||||
| 		r.storage.mustInsertPickle(pickle) | 		r.storage.mustInsertPickle(pickle) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	failed := r.concurrent(1, func() Formatter { return progressFunc("progress", ioutil.Discard) }) | 	failed := r.concurrent(1) | ||||||
| 	require.True(t, failed) | 	require.True(t, failed) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFailsWithUnknownFormatterOptionError(t *testing.T) { | func Test_FailsWithUnknownFormatterOptionError(t *testing.T) { | ||||||
| 	stderr, closer := bufErrorPipe(t) | 	stderr, closer := bufErrorPipe(t) | ||||||
| 	defer closer() | 	defer closer() | ||||||
| 	defer stderr.Close() | 	defer stderr.Close() | ||||||
| 
 | 
 | ||||||
| 	opt := Options{ | 	opts := Options{ | ||||||
| 		Format: "unknown", | 		Format: "unknown", | ||||||
| 		Paths:  []string{"features/load:6"}, | 		Paths:  []string{"features/load:6"}, | ||||||
| 		Output: ioutil.Discard, | 		Output: ioutil.Discard, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	status := RunWithOptions("fails", func(_ *Suite) {}, opt) | 	status := TestSuite{ | ||||||
|  | 		Name:                "fails", | ||||||
|  | 		ScenarioInitializer: func(_ *ScenarioContext) {}, | ||||||
|  | 		Options:             &opts, | ||||||
|  | 	}.Run() | ||||||
|  | 
 | ||||||
| 	require.Equal(t, exitOptionError, status) | 	require.Equal(t, exitOptionError, status) | ||||||
| 
 | 
 | ||||||
| 	closer() | 	closer() | ||||||
|  | @ -180,18 +186,23 @@ func TestFailsWithUnknownFormatterOptionError(t *testing.T) { | ||||||
| 	assert.Contains(t, out, `unregistered formatter name: "unknown", use one of`) | 	assert.Contains(t, out, `unregistered formatter name: "unknown", use one of`) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFailsWithOptionErrorWhenLookingForFeaturesInUnavailablePath(t *testing.T) { | func Test_FailsWithOptionErrorWhenLookingForFeaturesInUnavailablePath(t *testing.T) { | ||||||
| 	stderr, closer := bufErrorPipe(t) | 	stderr, closer := bufErrorPipe(t) | ||||||
| 	defer closer() | 	defer closer() | ||||||
| 	defer stderr.Close() | 	defer stderr.Close() | ||||||
| 
 | 
 | ||||||
| 	opt := Options{ | 	opts := Options{ | ||||||
| 		Format: "progress", | 		Format: "progress", | ||||||
| 		Paths:  []string{"unavailable"}, | 		Paths:  []string{"unavailable"}, | ||||||
| 		Output: ioutil.Discard, | 		Output: ioutil.Discard, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	status := RunWithOptions("fails", func(_ *Suite) {}, opt) | 	status := TestSuite{ | ||||||
|  | 		Name:                "fails", | ||||||
|  | 		ScenarioInitializer: func(_ *ScenarioContext) {}, | ||||||
|  | 		Options:             &opts, | ||||||
|  | 	}.Run() | ||||||
|  | 
 | ||||||
| 	require.Equal(t, exitOptionError, status) | 	require.Equal(t, exitOptionError, status) | ||||||
| 
 | 
 | ||||||
| 	closer() | 	closer() | ||||||
|  | @ -203,19 +214,29 @@ func TestFailsWithOptionErrorWhenLookingForFeaturesInUnavailablePath(t *testing. | ||||||
| 	assert.Equal(t, `feature path "unavailable" is not available`, out) | 	assert.Equal(t, `feature path "unavailable" is not available`, out) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestByDefaultRunsFeaturesPath(t *testing.T) { | func Test_ByDefaultRunsFeaturesPath(t *testing.T) { | ||||||
| 	opt := Options{ | 	opts := Options{ | ||||||
| 		Format: "progress", | 		Format: "progress", | ||||||
| 		Output: ioutil.Discard, | 		Output: ioutil.Discard, | ||||||
| 		Strict: true, | 		Strict: true, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	status := RunWithOptions("fails", func(_ *Suite) {}, opt) | 	status := TestSuite{ | ||||||
|  | 		Name:                "fails", | ||||||
|  | 		ScenarioInitializer: func(_ *ScenarioContext) {}, | ||||||
|  | 		Options:             &opts, | ||||||
|  | 	}.Run() | ||||||
|  | 
 | ||||||
| 	// should fail in strict mode due to undefined steps | 	// should fail in strict mode due to undefined steps | ||||||
| 	assert.Equal(t, exitFailure, status) | 	assert.Equal(t, exitFailure, status) | ||||||
| 
 | 
 | ||||||
| 	opt.Strict = false | 	opts.Strict = false | ||||||
| 	status = RunWithOptions("succeeds", func(_ *Suite) {}, opt) | 	status = TestSuite{ | ||||||
|  | 		Name:                "succeeds", | ||||||
|  | 		ScenarioInitializer: func(_ *ScenarioContext) {}, | ||||||
|  | 		Options:             &opts, | ||||||
|  | 	}.Run() | ||||||
|  | 
 | ||||||
| 	// should succeed in non strict mode due to undefined steps | 	// should succeed in non strict mode due to undefined steps | ||||||
| 	assert.Equal(t, exitSuccess, status) | 	assert.Equal(t, exitSuccess, status) | ||||||
| } | } | ||||||
|  | @ -232,7 +253,7 @@ func bufErrorPipe(t *testing.T) (io.ReadCloser, func()) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFeatureFilePathParser(t *testing.T) { | func Test_FeatureFilePathParser(t *testing.T) { | ||||||
| 
 | 
 | ||||||
| 	type Case struct { | 	type Case struct { | ||||||
| 		input string | 		input string | ||||||
|  | @ -327,19 +348,8 @@ func Test_AllFeaturesRun(t *testing.T) { | ||||||
| 0s | 0s | ||||||
| ` | ` | ||||||
| 
 | 
 | ||||||
| 	fmtOutputSuiteInitializer := func(s *Suite) { SuiteContext(s) } | 	actualStatus, actualOutput := testRun(t, | ||||||
| 	fmtOutputScenarioInitializer := InitializeScenario | 		InitializeScenario, | ||||||
| 
 |  | ||||||
| 	actualStatus, actualOutput := testRunWithOptions(t, |  | ||||||
| 		fmtOutputSuiteInitializer, |  | ||||||
| 		format, concurrency, []string{"features"}, |  | ||||||
| 	) |  | ||||||
| 
 |  | ||||||
| 	assert.Equal(t, exitSuccess, actualStatus) |  | ||||||
| 	assert.Equal(t, expected, actualOutput) |  | ||||||
| 
 |  | ||||||
| 	actualStatus, actualOutput = testRun(t, |  | ||||||
| 		fmtOutputScenarioInitializer, |  | ||||||
| 		format, concurrency, | 		format, concurrency, | ||||||
| 		noRandomFlag, []string{"features"}, | 		noRandomFlag, []string{"features"}, | ||||||
| 	) | 	) | ||||||
|  | @ -348,7 +358,7 @@ func Test_AllFeaturesRun(t *testing.T) { | ||||||
| 	assert.Equal(t, expected, actualOutput) | 	assert.Equal(t, expected, actualOutput) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestFormatterConcurrencyRun(t *testing.T) { | func Test_FormatterConcurrencyRun(t *testing.T) { | ||||||
| 	formatters := []string{ | 	formatters := []string{ | ||||||
| 		"progress", | 		"progress", | ||||||
| 		"junit", | 		"junit", | ||||||
|  | @ -363,13 +373,6 @@ func TestFormatterConcurrencyRun(t *testing.T) { | ||||||
| 	const noRandomFlag = 0 | 	const noRandomFlag = 0 | ||||||
| 	const noConcurrency = 1 | 	const noConcurrency = 1 | ||||||
| 
 | 
 | ||||||
| 	fmtOutputSuiteInitializer := func(s *Suite) { |  | ||||||
| 		s.Step(`^(?:a )?failing step`, failingStepDef) |  | ||||||
| 		s.Step(`^(?:a )?pending step$`, pendingStepDef) |  | ||||||
| 		s.Step(`^(?:a )?passing step$`, passingStepDef) |  | ||||||
| 		s.Step(`^odd (\d+) and even (\d+) number$`, oddEvenStepDef) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fmtOutputScenarioInitializer := func(ctx *ScenarioContext) { | 	fmtOutputScenarioInitializer := func(ctx *ScenarioContext) { | ||||||
| 		ctx.Step(`^(?:a )?failing step`, failingStepDef) | 		ctx.Step(`^(?:a )?failing step`, failingStepDef) | ||||||
| 		ctx.Step(`^(?:a )?pending step$`, pendingStepDef) | 		ctx.Step(`^(?:a )?pending step$`, pendingStepDef) | ||||||
|  | @ -381,24 +384,12 @@ func TestFormatterConcurrencyRun(t *testing.T) { | ||||||
| 		t.Run( | 		t.Run( | ||||||
| 			fmt.Sprintf("%s/concurrency/%d", formatter, concurrency), | 			fmt.Sprintf("%s/concurrency/%d", formatter, concurrency), | ||||||
| 			func(t *testing.T) { | 			func(t *testing.T) { | ||||||
| 				expectedStatus, expectedOutput := testRunWithOptions(t, | 				expectedStatus, expectedOutput := testRun(t, | ||||||
| 					fmtOutputSuiteInitializer, |  | ||||||
| 					formatter, noConcurrency, featurePaths, |  | ||||||
| 				) |  | ||||||
| 				actualStatus, actualOutput := testRunWithOptions(t, |  | ||||||
| 					fmtOutputSuiteInitializer, |  | ||||||
| 					formatter, concurrency, featurePaths, |  | ||||||
| 				) |  | ||||||
| 
 |  | ||||||
| 				assert.Equal(t, expectedStatus, actualStatus) |  | ||||||
| 				assertOutput(t, formatter, expectedOutput, actualOutput) |  | ||||||
| 
 |  | ||||||
| 				expectedStatus, expectedOutput = testRun(t, |  | ||||||
| 					fmtOutputScenarioInitializer, | 					fmtOutputScenarioInitializer, | ||||||
| 					formatter, noConcurrency, | 					formatter, noConcurrency, | ||||||
| 					noRandomFlag, featurePaths, | 					noRandomFlag, featurePaths, | ||||||
| 				) | 				) | ||||||
| 				actualStatus, actualOutput = testRun(t, | 				actualStatus, actualOutput := testRun(t, | ||||||
| 					fmtOutputScenarioInitializer, | 					fmtOutputScenarioInitializer, | ||||||
| 					formatter, concurrency, | 					formatter, concurrency, | ||||||
| 					noRandomFlag, featurePaths, | 					noRandomFlag, featurePaths, | ||||||
|  | @ -411,25 +402,6 @@ func TestFormatterConcurrencyRun(t *testing.T) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testRunWithOptions(t *testing.T, initializer func(*Suite), format string, concurrency int, featurePaths []string) (int, string) { |  | ||||||
| 	output := new(bytes.Buffer) |  | ||||||
| 
 |  | ||||||
| 	opts := Options{ |  | ||||||
| 		Format:      format, |  | ||||||
| 		NoColors:    true, |  | ||||||
| 		Paths:       featurePaths, |  | ||||||
| 		Concurrency: concurrency, |  | ||||||
| 		Output:      output, |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	status := RunWithOptions("succeed", initializer, opts) |  | ||||||
| 
 |  | ||||||
| 	actual, err := ioutil.ReadAll(output) |  | ||||||
| 	require.NoError(t, err) |  | ||||||
| 
 |  | ||||||
| 	return status, string(actual) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func testRun( | func testRun( | ||||||
| 	t *testing.T, | 	t *testing.T, | ||||||
| 	scenarioInitializer func(*ScenarioContext), | 	scenarioInitializer func(*ScenarioContext), | ||||||
|  |  | ||||||
							
								
								
									
										233
									
								
								suite.go
									
										
									
									
									
								
							
							
						
						
									
										233
									
								
								suite.go
									
										
									
									
									
								
							|  | @ -2,9 +2,7 @@ package godog | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"math/rand" |  | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"regexp" |  | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
| 	"github.com/cucumber/messages-go/v10" | 	"github.com/cucumber/messages-go/v10" | ||||||
|  | @ -20,24 +18,8 @@ var ErrUndefined = fmt.Errorf("step is undefined") | ||||||
| // step implementation is pending | // step implementation is pending | ||||||
| var ErrPending = fmt.Errorf("step implementation is pending") | var ErrPending = fmt.Errorf("step implementation is pending") | ||||||
| 
 | 
 | ||||||
| // Suite allows various contexts | type suite struct { | ||||||
| // to register steps and event handlers. | 	steps []*StepDefinition | ||||||
| // |  | ||||||
| // When running a test suite, the instance of Suite |  | ||||||
| // is passed to all functions (contexts), which |  | ||||||
| // have it as a first and only argument. |  | ||||||
| // |  | ||||||
| // Note that all event hooks does not catch panic errors |  | ||||||
| // in order to have a trace information. Only step |  | ||||||
| // executions are catching panic error since it may |  | ||||||
| // be a context specific error. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. |  | ||||||
| // This struct will therefore not be exported in the future. |  | ||||||
| type Suite struct { |  | ||||||
| 	steps    []*StepDefinition |  | ||||||
| 	features []*feature |  | ||||||
| 
 | 
 | ||||||
| 	fmt     Formatter | 	fmt     Formatter | ||||||
| 	storage *storage | 	storage *storage | ||||||
|  | @ -48,175 +30,13 @@ type Suite struct { | ||||||
| 	strict        bool | 	strict        bool | ||||||
| 
 | 
 | ||||||
| 	// suite event handlers | 	// suite event handlers | ||||||
| 	beforeSuiteHandlers    []func() | 	beforeScenarioHandlers []func(*Scenario) | ||||||
| 	beforeScenarioHandlers []func(*messages.Pickle) | 	beforeStepHandlers     []func(*Step) | ||||||
| 	beforeStepHandlers     []func(*messages.Pickle_PickleStep) | 	afterStepHandlers      []func(*Step, error) | ||||||
| 	afterStepHandlers      []func(*messages.Pickle_PickleStep, error) | 	afterScenarioHandlers  []func(*Scenario, error) | ||||||
| 	afterScenarioHandlers  []func(*messages.Pickle, error) |  | ||||||
| 	afterSuiteHandlers     []func() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Step allows to register a *StepDefinition in Godog | func (s *suite) matchStep(step *messages.Pickle_PickleStep) *StepDefinition { | ||||||
| // feature suite, the definition will be applied |  | ||||||
| // to all steps matching the given Regexp expr. |  | ||||||
| // |  | ||||||
| // It will panic if expr is not a valid regular |  | ||||||
| // expression or stepFunc is not a valid step |  | ||||||
| // handler. |  | ||||||
| // |  | ||||||
| // Note that if there are two definitions which may match |  | ||||||
| // the same step, then only the first matched handler |  | ||||||
| // will be applied. |  | ||||||
| // |  | ||||||
| // If none of the *StepDefinition is matched, then |  | ||||||
| // ErrUndefined error will be returned when |  | ||||||
| // running steps. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *ScenarioContext) Step instead. |  | ||||||
| func (s *Suite) Step(expr interface{}, stepFunc interface{}) { |  | ||||||
| 	var regex *regexp.Regexp |  | ||||||
| 
 |  | ||||||
| 	switch t := expr.(type) { |  | ||||||
| 	case *regexp.Regexp: |  | ||||||
| 		regex = t |  | ||||||
| 	case string: |  | ||||||
| 		regex = regexp.MustCompile(t) |  | ||||||
| 	case []byte: |  | ||||||
| 		regex = regexp.MustCompile(string(t)) |  | ||||||
| 	default: |  | ||||||
| 		panic(fmt.Sprintf("expecting expr to be a *regexp.Regexp or a string, got type: %T", expr)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	v := reflect.ValueOf(stepFunc) |  | ||||||
| 	typ := v.Type() |  | ||||||
| 	if typ.Kind() != reflect.Func { |  | ||||||
| 		panic(fmt.Sprintf("expected handler to be func, but got: %T", stepFunc)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if typ.NumOut() != 1 { |  | ||||||
| 		panic(fmt.Sprintf("expected handler to return only one value, but it has: %d", typ.NumOut())) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	def := &StepDefinition{ |  | ||||||
| 		Handler: stepFunc, |  | ||||||
| 		Expr:    regex, |  | ||||||
| 		hv:      v, |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	typ = typ.Out(0) |  | ||||||
| 	switch typ.Kind() { |  | ||||||
| 	case reflect.Interface: |  | ||||||
| 		if !typ.Implements(errorInterface) { |  | ||||||
| 			panic(fmt.Sprintf("expected handler to return an error, but got: %s", typ.Kind())) |  | ||||||
| 		} |  | ||||||
| 	case reflect.Slice: |  | ||||||
| 		if typ.Elem().Kind() != reflect.String { |  | ||||||
| 			panic(fmt.Sprintf("expected handler to return []string for multistep, but got: []%s", typ.Kind())) |  | ||||||
| 		} |  | ||||||
| 		def.nested = true |  | ||||||
| 	default: |  | ||||||
| 		panic(fmt.Sprintf("expected handler to return an error or []string, but got: %s", typ.Kind())) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.steps = append(s.steps, def) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // BeforeSuite registers a function or method |  | ||||||
| // to be run once before suite runner. |  | ||||||
| // |  | ||||||
| // Use it to prepare the test suite for a spin. |  | ||||||
| // Connect and prepare database for instance... |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *TestSuiteContext) BeforeSuite instead. |  | ||||||
| func (s *Suite) BeforeSuite(fn func()) { |  | ||||||
| 	s.beforeSuiteHandlers = append(s.beforeSuiteHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // BeforeScenario registers a function or method |  | ||||||
| // to be run before every pickle. |  | ||||||
| // |  | ||||||
| // It is a good practice to restore the default state |  | ||||||
| // before every scenario so it would be isolated from |  | ||||||
| // any kind of state. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *ScenarioContext) BeforeScenario instead. |  | ||||||
| func (s *Suite) BeforeScenario(fn func(*messages.Pickle)) { |  | ||||||
| 	s.beforeScenarioHandlers = append(s.beforeScenarioHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // BeforeStep registers a function or method |  | ||||||
| // to be run before every step. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *ScenarioContext) BeforeStep instead. |  | ||||||
| func (s *Suite) BeforeStep(fn func(*messages.Pickle_PickleStep)) { |  | ||||||
| 	s.beforeStepHandlers = append(s.beforeStepHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // AfterStep registers an function or method |  | ||||||
| // to be run after every step. |  | ||||||
| // |  | ||||||
| // It may be convenient to return a different kind of error |  | ||||||
| // in order to print more state details which may help |  | ||||||
| // in case of step failure |  | ||||||
| // |  | ||||||
| // In some cases, for example when running a headless |  | ||||||
| // browser, to take a screenshot after failure. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *ScenarioContext) AfterStep instead. |  | ||||||
| func (s *Suite) AfterStep(fn func(*messages.Pickle_PickleStep, error)) { |  | ||||||
| 	s.afterStepHandlers = append(s.afterStepHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // AfterScenario registers an function or method |  | ||||||
| // to be run after every pickle. |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *ScenarioContext) AfterScenario instead. |  | ||||||
| func (s *Suite) AfterScenario(fn func(*messages.Pickle, error)) { |  | ||||||
| 	s.afterScenarioHandlers = append(s.afterScenarioHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // AfterSuite registers a function or method |  | ||||||
| // to be run once after suite runner |  | ||||||
| // |  | ||||||
| // Deprecated: The current Suite initializer will be removed and replaced by |  | ||||||
| // two initializers, one for the Test Suite and one for the Scenarios. Use |  | ||||||
| // func (ctx *TestSuiteContext) AfterSuite instead. |  | ||||||
| func (s *Suite) AfterSuite(fn func()) { |  | ||||||
| 	s.afterSuiteHandlers = append(s.afterSuiteHandlers, fn) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *Suite) run() { |  | ||||||
| 	// run before suite handlers |  | ||||||
| 	for _, f := range s.beforeSuiteHandlers { |  | ||||||
| 		f() |  | ||||||
| 	} |  | ||||||
| 	// run features |  | ||||||
| 	for _, f := range s.features { |  | ||||||
| 		s.runFeature(f) |  | ||||||
| 		if s.failed && s.stopOnFailure { |  | ||||||
| 			// stop on first failure |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	// run after suite handlers |  | ||||||
| 	for _, f := range s.afterSuiteHandlers { |  | ||||||
| 		f() |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *Suite) matchStep(step *messages.Pickle_PickleStep) *StepDefinition { |  | ||||||
| 	def := s.matchStepText(step.Text) | 	def := s.matchStepText(step.Text) | ||||||
| 	if def != nil && step.Argument != nil { | 	if def != nil && step.Argument != nil { | ||||||
| 		def.args = append(def.args, step.Argument) | 		def.args = append(def.args, step.Argument) | ||||||
|  | @ -224,7 +44,7 @@ func (s *Suite) matchStep(step *messages.Pickle_PickleStep) *StepDefinition { | ||||||
| 	return def | 	return def | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) runStep(pickle *messages.Pickle, step *messages.Pickle_PickleStep, prevStepErr error) (err error) { | func (s *suite) runStep(pickle *messages.Pickle, step *messages.Pickle_PickleStep, prevStepErr error) (err error) { | ||||||
| 	// run before step handlers | 	// run before step handlers | ||||||
| 	for _, f := range s.beforeStepHandlers { | 	for _, f := range s.beforeStepHandlers { | ||||||
| 		f(step) | 		f(step) | ||||||
|  | @ -312,7 +132,7 @@ func (s *Suite) runStep(pickle *messages.Pickle, step *messages.Pickle_PickleSte | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) maybeUndefined(text string, arg interface{}) ([]string, error) { | func (s *suite) maybeUndefined(text string, arg interface{}) ([]string, error) { | ||||||
| 	step := s.matchStepText(text) | 	step := s.matchStepText(text) | ||||||
| 	if nil == step { | 	if nil == step { | ||||||
| 		return []string{text}, nil | 		return []string{text}, nil | ||||||
|  | @ -345,7 +165,7 @@ func (s *Suite) maybeUndefined(text string, arg interface{}) ([]string, error) { | ||||||
| 	return undefined, nil | 	return undefined, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) maybeSubSteps(result interface{}) error { | func (s *suite) maybeSubSteps(result interface{}) error { | ||||||
| 	if nil == result { | 	if nil == result { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  | @ -369,7 +189,7 @@ func (s *Suite) maybeSubSteps(result interface{}) error { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) matchStepText(text string) *StepDefinition { | func (s *suite) matchStepText(text string) *StepDefinition { | ||||||
| 	for _, h := range s.steps { | 	for _, h := range s.steps { | ||||||
| 		if m := h.Expr.FindStringSubmatch(text); len(m) > 0 { | 		if m := h.Expr.FindStringSubmatch(text); len(m) > 0 { | ||||||
| 			var args []interface{} | 			var args []interface{} | ||||||
|  | @ -391,7 +211,7 @@ func (s *Suite) matchStepText(text string) *StepDefinition { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) runSteps(pickle *messages.Pickle, steps []*messages.Pickle_PickleStep) (err error) { | func (s *suite) runSteps(pickle *messages.Pickle, steps []*messages.Pickle_PickleStep) (err error) { | ||||||
| 	for _, step := range steps { | 	for _, step := range steps { | ||||||
| 		stepErr := s.runStep(pickle, step, err) | 		stepErr := s.runStep(pickle, step, err) | ||||||
| 		switch stepErr { | 		switch stepErr { | ||||||
|  | @ -410,7 +230,7 @@ func (s *Suite) runSteps(pickle *messages.Pickle, steps []*messages.Pickle_Pickl | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) shouldFail(err error) bool { | func (s *suite) shouldFail(err error) bool { | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
|  | @ -422,31 +242,6 @@ func (s *Suite) shouldFail(err error) bool { | ||||||
| 	return true | 	return true | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) runFeature(f *feature) { |  | ||||||
| 	s.fmt.Feature(f.GherkinDocument, f.Uri, f.content) |  | ||||||
| 
 |  | ||||||
| 	pickles := make([]*messages.Pickle, len(f.pickles)) |  | ||||||
| 	if s.randomSeed != 0 { |  | ||||||
| 		r := rand.New(rand.NewSource(s.randomSeed)) |  | ||||||
| 		perm := r.Perm(len(f.pickles)) |  | ||||||
| 		for i, v := range perm { |  | ||||||
| 			pickles[v] = f.pickles[i] |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		copy(pickles, f.pickles) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, pickle := range pickles { |  | ||||||
| 		err := s.runPickle(pickle) |  | ||||||
| 		if s.shouldFail(err) { |  | ||||||
| 			s.failed = true |  | ||||||
| 			if s.stopOnFailure { |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func isEmptyFeature(pickles []*messages.Pickle) bool { | func isEmptyFeature(pickles []*messages.Pickle) bool { | ||||||
| 	for _, pickle := range pickles { | 	for _, pickle := range pickles { | ||||||
| 		if len(pickle.Steps) > 0 { | 		if len(pickle.Steps) > 0 { | ||||||
|  | @ -457,7 +252,7 @@ func isEmptyFeature(pickles []*messages.Pickle) bool { | ||||||
| 	return true | 	return true | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Suite) runPickle(pickle *messages.Pickle) (err error) { | func (s *suite) runPickle(pickle *messages.Pickle) (err error) { | ||||||
| 	if len(pickle.Steps) == 0 { | 	if len(pickle.Steps) == 0 { | ||||||
| 		pr := pickleResult{PickleID: pickle.Id, StartedAt: timeNowFunc()} | 		pr := pickleResult{PickleID: pickle.Id, StartedAt: timeNowFunc()} | ||||||
| 		s.storage.mustInsertPickleResult(pr) | 		s.storage.mustInsertPickleResult(pr) | ||||||
|  |  | ||||||
							
								
								
									
										632
									
								
								suite_context.go
									
										
									
									
									
								
							
							
						
						
									
										632
									
								
								suite_context.go
									
										
									
									
									
								
							|  | @ -1,632 +0,0 @@ | ||||||
| package godog |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"bytes" |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"encoding/xml" |  | ||||||
| 	"fmt" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"regexp" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 
 |  | ||||||
| 	"github.com/cucumber/gherkin-go/v11" |  | ||||||
| 	"github.com/cucumber/messages-go/v10" |  | ||||||
| 	"github.com/stretchr/testify/assert" |  | ||||||
| 
 |  | ||||||
| 	"github.com/cucumber/godog/colors" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| // SuiteContext provides steps for godog suite execution and |  | ||||||
| // can be used for meta-testing of godog features/steps themselves. |  | ||||||
| // |  | ||||||
| // Beware, steps or their definitions might change without backward |  | ||||||
| // compatibility guarantees. A typical user of the godog library should never |  | ||||||
| // need this, rather it is provided for those developing add-on libraries for godog. |  | ||||||
| // |  | ||||||
| // For an example of how to use, see godog's own `features/` and `suite_test.go`. |  | ||||||
| // |  | ||||||
| // Deprecated: Use InitializeScenario instead. |  | ||||||
| func SuiteContext(s *Suite, additionalContextInitializers ...func(suite *Suite)) { |  | ||||||
| 	c := &suiteContext{ |  | ||||||
| 		extraCIs: additionalContextInitializers, |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// apply any additional context intializers to modify the context that the |  | ||||||
| 	// meta-tests will be run in |  | ||||||
| 	for _, ci := range additionalContextInitializers { |  | ||||||
| 		ci(s) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.BeforeScenario(c.ResetBeforeEachScenario) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?feature path "([^"]*)"$`, c.featurePath) |  | ||||||
| 	s.Step(`^I parse features$`, c.parseFeatures) |  | ||||||
| 	s.Step(`^I'm listening to suite events$`, c.iAmListeningToSuiteEvents) |  | ||||||
| 	s.Step(`^I run feature suite$`, c.iRunFeatureSuite) |  | ||||||
| 	s.Step(`^I run feature suite with tags "([^"]*)"$`, c.iRunFeatureSuiteWithTags) |  | ||||||
| 	s.Step(`^I run feature suite with formatter "([^"]*)"$`, c.iRunFeatureSuiteWithFormatter) |  | ||||||
| 	s.Step(`^(?:I )(allow|disable) variable injection`, c.iSetVariableInjectionTo) |  | ||||||
| 	s.Step(`^(?:a )?feature "([^"]*)"(?: file)?:$`, c.aFeatureFile) |  | ||||||
| 	s.Step(`^the suite should have (passed|failed)$`, c.theSuiteShouldHave) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^I should have ([\d]+) features? files?:$`, c.iShouldHaveNumFeatureFiles) |  | ||||||
| 	s.Step(`^I should have ([\d]+) scenarios? registered$`, c.numScenariosRegistered) |  | ||||||
| 	s.Step(`^there (was|were) ([\d]+) "([^"]*)" events? fired$`, c.thereWereNumEventsFired) |  | ||||||
| 	s.Step(`^there was event triggered before scenario "([^"]*)"$`, c.thereWasEventTriggeredBeforeScenario) |  | ||||||
| 	s.Step(`^these events had to be fired for a number of times:$`, c.theseEventsHadToBeFiredForNumberOfTimes) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?failing step`, c.aFailingStep) |  | ||||||
| 	s.Step(`^this step should fail`, c.aFailingStep) |  | ||||||
| 	s.Step(`^the following steps? should be (passed|failed|skipped|undefined|pending):`, c.followingStepsShouldHave) |  | ||||||
| 	s.Step(`^the undefined step snippets should be:$`, c.theUndefinedStepSnippetsShouldBe) |  | ||||||
| 
 |  | ||||||
| 	// event stream |  | ||||||
| 	s.Step(`^the following events should be fired:$`, c.thereShouldBeEventsFired) |  | ||||||
| 
 |  | ||||||
| 	// lt |  | ||||||
| 	s.Step(`^savybių aplankas "([^"]*)"$`, c.featurePath) |  | ||||||
| 	s.Step(`^aš išskaitau savybes$`, c.parseFeatures) |  | ||||||
| 	s.Step(`^aš turėčiau turėti ([\d]+) savybių failus:$`, c.iShouldHaveNumFeatureFiles) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?pending step$`, func() error { |  | ||||||
| 		return ErrPending |  | ||||||
| 	}) |  | ||||||
| 	s.Step(`^(?:a )?passing step$`, func() error { |  | ||||||
| 		return nil |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	// Introduced to test formatter/cucumber.feature |  | ||||||
| 	s.Step(`^the rendered json will be as follows:$`, c.theRenderJSONWillBe) |  | ||||||
| 
 |  | ||||||
| 	// Introduced to test formatter/pretty.feature |  | ||||||
| 	s.Step(`^the rendered output will be as follows:$`, c.theRenderOutputWillBe) |  | ||||||
| 
 |  | ||||||
| 	// Introduced to test formatter/junit.feature |  | ||||||
| 	s.Step(`^the rendered xml will be as follows:$`, c.theRenderXMLWillBe) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?failing multistep$`, func() Steps { |  | ||||||
| 		return Steps{"passing step", "failing step"} |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a |an )?undefined multistep$`, func() Steps { |  | ||||||
| 		return Steps{"passing step", "undefined step", "passing step"} |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?passing multistep$`, func() Steps { |  | ||||||
| 		return Steps{"passing step", "passing step", "passing step"} |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.Step(`^(?:a )?failing nested multistep$`, func() Steps { |  | ||||||
| 		return Steps{"passing step", "passing multistep", "failing multistep"} |  | ||||||
| 	}) |  | ||||||
| 	// Default recovery step |  | ||||||
| 	s.Step(`Ignore.*`, func() error { |  | ||||||
| 		return nil |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.BeforeStep(c.inject) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) inject(step *Step) { |  | ||||||
| 	if !s.allowInjection { |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	step.Text = injectAll(step.Text) |  | ||||||
| 
 |  | ||||||
| 	if table := step.Argument.GetDataTable(); table != nil { |  | ||||||
| 		for i := 0; i < len(table.Rows); i++ { |  | ||||||
| 			for n, cell := range table.Rows[i].Cells { |  | ||||||
| 				table.Rows[i].Cells[n].Value = injectAll(cell.Value) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if doc := step.Argument.GetDocString(); doc != nil { |  | ||||||
| 		doc.Content = injectAll(doc.Content) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func injectAll(src string) string { |  | ||||||
| 	re := regexp.MustCompile(`{{[^{}]+}}`) |  | ||||||
| 	return re.ReplaceAllStringFunc( |  | ||||||
| 		src, |  | ||||||
| 		func(key string) string { |  | ||||||
| 			injectRegex := regexp.MustCompile(`^{{.+}}$`) |  | ||||||
| 
 |  | ||||||
| 			if injectRegex.MatchString(key) { |  | ||||||
| 				return "someverylonginjectionsoweacanbesureitsurpasstheinitiallongeststeplenghtanditwillhelptestsmethodsafety" |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return key |  | ||||||
| 		}, |  | ||||||
| 	) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type firedEvent struct { |  | ||||||
| 	name string |  | ||||||
| 	args []interface{} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type suiteContext struct { |  | ||||||
| 	paths          []string |  | ||||||
| 	testedSuite    *Suite |  | ||||||
| 	extraCIs       []func(suite *Suite) |  | ||||||
| 	events         []*firedEvent |  | ||||||
| 	out            bytes.Buffer |  | ||||||
| 	allowInjection bool |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) ResetBeforeEachScenario(*Scenario) { |  | ||||||
| 	// reset whole suite with the state |  | ||||||
| 	s.out.Reset() |  | ||||||
| 	s.paths = []string{} |  | ||||||
| 	s.testedSuite = &Suite{} |  | ||||||
| 	// our tested suite will have the same context registered |  | ||||||
| 	SuiteContext(s.testedSuite, s.extraCIs...) |  | ||||||
| 	// reset all fired events |  | ||||||
| 	s.events = []*firedEvent{} |  | ||||||
| 	s.allowInjection = false |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iSetVariableInjectionTo(to string) error { |  | ||||||
| 	s.allowInjection = to == "allow" |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iRunFeatureSuiteWithTags(tags string) error { |  | ||||||
| 	if err := s.parseFeatures(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, feat := range s.testedSuite.features { |  | ||||||
| 		feat.pickles = applyTagFilter(tags, feat.pickles) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.storage = newStorage() |  | ||||||
| 	for _, feat := range s.testedSuite.features { |  | ||||||
| 		s.testedSuite.storage.mustInsertFeature(feat) |  | ||||||
| 
 |  | ||||||
| 		for _, pickle := range feat.pickles { |  | ||||||
| 			s.testedSuite.storage.mustInsertPickle(pickle) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fmt := newBaseFmt("godog", &s.out) |  | ||||||
| 	fmt.setStorage(s.testedSuite.storage) |  | ||||||
| 	s.testedSuite.fmt = fmt |  | ||||||
| 
 |  | ||||||
| 	testRunStarted := testRunStarted{StartedAt: timeNowFunc()} |  | ||||||
| 	s.testedSuite.storage.mustInsertTestRunStarted(testRunStarted) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.fmt.TestRunStarted() |  | ||||||
| 	s.testedSuite.run() |  | ||||||
| 	s.testedSuite.fmt.Summary() |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iRunFeatureSuiteWithFormatter(name string) error { |  | ||||||
| 	if err := s.parseFeatures(); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	f := FindFmt(name) |  | ||||||
| 	if f == nil { |  | ||||||
| 		return fmt.Errorf(`formatter "%s" is not available`, name) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.storage = newStorage() |  | ||||||
| 	for _, feat := range s.testedSuite.features { |  | ||||||
| 		s.testedSuite.storage.mustInsertFeature(feat) |  | ||||||
| 
 |  | ||||||
| 		for _, pickle := range feat.pickles { |  | ||||||
| 			s.testedSuite.storage.mustInsertPickle(pickle) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.fmt = f("godog", colors.Uncolored(&s.out)) |  | ||||||
| 	if fmt, ok := s.testedSuite.fmt.(storageFormatter); ok { |  | ||||||
| 		fmt.setStorage(s.testedSuite.storage) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	testRunStarted := testRunStarted{StartedAt: timeNowFunc()} |  | ||||||
| 	s.testedSuite.storage.mustInsertTestRunStarted(testRunStarted) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.fmt.TestRunStarted() |  | ||||||
| 	s.testedSuite.run() |  | ||||||
| 	s.testedSuite.fmt.Summary() |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) thereShouldBeEventsFired(doc *DocString) error { |  | ||||||
| 	actual := strings.Split(strings.TrimSpace(s.out.String()), "\n") |  | ||||||
| 	expect := strings.Split(strings.TrimSpace(doc.Content), "\n") |  | ||||||
| 
 |  | ||||||
| 	if len(expect) != len(actual) { |  | ||||||
| 		return fmt.Errorf("expected %d events, but got %d", len(expect), len(actual)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	type ev struct { |  | ||||||
| 		Event string |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for i, event := range actual { |  | ||||||
| 		exp := strings.TrimSpace(expect[i]) |  | ||||||
| 		var act ev |  | ||||||
| 
 |  | ||||||
| 		if err := json.Unmarshal([]byte(event), &act); err != nil { |  | ||||||
| 			return fmt.Errorf("failed to read event data: %v", err) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if act.Event != exp { |  | ||||||
| 			return fmt.Errorf(`expected event: "%s" at position: %d, but actual was "%s"`, exp, i, act.Event) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) cleanupSnippet(snip string) string { |  | ||||||
| 	lines := strings.Split(strings.TrimSpace(snip), "\n") |  | ||||||
| 	for i := 0; i < len(lines); i++ { |  | ||||||
| 		lines[i] = strings.TrimSpace(lines[i]) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return strings.Join(lines, "\n") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theUndefinedStepSnippetsShouldBe(body *DocString) error { |  | ||||||
| 	f, ok := s.testedSuite.fmt.(*basefmt) |  | ||||||
| 	if !ok { |  | ||||||
| 		return fmt.Errorf("this step requires *basefmt, but there is: %T", s.testedSuite.fmt) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	actual := s.cleanupSnippet(f.snippets()) |  | ||||||
| 	expected := s.cleanupSnippet(body.Content) |  | ||||||
| 
 |  | ||||||
| 	if actual != expected { |  | ||||||
| 		return fmt.Errorf("snippets do not match actual: %s", f.snippets()) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) followingStepsShouldHave(status string, steps *DocString) error { |  | ||||||
| 	var expected = strings.Split(steps.Content, "\n") |  | ||||||
| 	var actual, unmatched, matched []string |  | ||||||
| 
 |  | ||||||
| 	f, ok := s.testedSuite.fmt.(*basefmt) |  | ||||||
| 	if !ok { |  | ||||||
| 		return fmt.Errorf("this step requires *basefmt, but there is: %T", s.testedSuite.fmt) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	switch status { |  | ||||||
| 	case "passed": |  | ||||||
| 		for _, st := range f.storage.mustGetPickleStepResultsByStatus(passed) { |  | ||||||
| 			pickleStep := f.storage.mustGetPickleStep(st.PickleStepID) |  | ||||||
| 			actual = append(actual, pickleStep.Text) |  | ||||||
| 		} |  | ||||||
| 	case "failed": |  | ||||||
| 		for _, st := range f.storage.mustGetPickleStepResultsByStatus(failed) { |  | ||||||
| 			pickleStep := f.storage.mustGetPickleStep(st.PickleStepID) |  | ||||||
| 			actual = append(actual, pickleStep.Text) |  | ||||||
| 		} |  | ||||||
| 	case "skipped": |  | ||||||
| 		for _, st := range f.storage.mustGetPickleStepResultsByStatus(skipped) { |  | ||||||
| 			pickleStep := f.storage.mustGetPickleStep(st.PickleStepID) |  | ||||||
| 			actual = append(actual, pickleStep.Text) |  | ||||||
| 		} |  | ||||||
| 	case "undefined": |  | ||||||
| 		for _, st := range f.storage.mustGetPickleStepResultsByStatus(undefined) { |  | ||||||
| 			pickleStep := f.storage.mustGetPickleStep(st.PickleStepID) |  | ||||||
| 			actual = append(actual, pickleStep.Text) |  | ||||||
| 		} |  | ||||||
| 	case "pending": |  | ||||||
| 		for _, st := range f.storage.mustGetPickleStepResultsByStatus(pending) { |  | ||||||
| 			pickleStep := f.storage.mustGetPickleStep(st.PickleStepID) |  | ||||||
| 			actual = append(actual, pickleStep.Text) |  | ||||||
| 		} |  | ||||||
| 	default: |  | ||||||
| 		return fmt.Errorf("unexpected step status wanted: %s", status) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if len(expected) > len(actual) { |  | ||||||
| 		return fmt.Errorf("number of expected %s steps: %d is less than actual %s steps: %d", status, len(expected), status, len(actual)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, a := range actual { |  | ||||||
| 		for _, e := range expected { |  | ||||||
| 			if a == e { |  | ||||||
| 				matched = append(matched, e) |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if len(matched) >= len(expected) { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, s := range expected { |  | ||||||
| 		var found bool |  | ||||||
| 		for _, m := range matched { |  | ||||||
| 			if s == m { |  | ||||||
| 				found = true |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if !found { |  | ||||||
| 			unmatched = append(unmatched, s) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return fmt.Errorf("the steps: %s - are not %s", strings.Join(unmatched, ", "), status) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iAmListeningToSuiteEvents() error { |  | ||||||
| 	s.testedSuite.BeforeSuite(func() { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"BeforeSuite", []interface{}{}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.AfterSuite(func() { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"AfterSuite", []interface{}{}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.BeforeScenario(func(pickle *Scenario) { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"BeforeScenario", []interface{}{pickle}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.AfterScenario(func(pickle *Scenario, err error) { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"AfterScenario", []interface{}{pickle, err}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.BeforeStep(func(step *Step) { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"BeforeStep", []interface{}{step}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.AfterStep(func(step *Step, err error) { |  | ||||||
| 		s.events = append(s.events, &firedEvent{"AfterStep", []interface{}{step, err}}) |  | ||||||
| 	}) |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) aFailingStep() error { |  | ||||||
| 	return fmt.Errorf("intentional failure") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // parse a given feature file body as a feature |  | ||||||
| func (s *suiteContext) aFeatureFile(path string, body *DocString) error { |  | ||||||
| 	gd, err := gherkin.ParseGherkinDocument(strings.NewReader(body.Content), (&messages.Incrementing{}).NewId) |  | ||||||
| 	gd.Uri = path |  | ||||||
| 
 |  | ||||||
| 	pickles := gherkin.Pickles(*gd, path, (&messages.Incrementing{}).NewId) |  | ||||||
| 	s.testedSuite.features = append(s.testedSuite.features, &feature{GherkinDocument: gd, pickles: pickles}) |  | ||||||
| 
 |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) featurePath(path string) error { |  | ||||||
| 	s.paths = append(s.paths, path) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) parseFeatures() error { |  | ||||||
| 	fts, err := parseFeatures("", s.paths) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	s.testedSuite.features = append(s.testedSuite.features, fts...) |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theSuiteShouldHave(state string) error { |  | ||||||
| 	if s.testedSuite.failed && state == "passed" { |  | ||||||
| 		return fmt.Errorf("the feature suite has failed") |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if !s.testedSuite.failed && state == "failed" { |  | ||||||
| 		return fmt.Errorf("the feature suite has passed") |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iShouldHaveNumFeatureFiles(num int, files *DocString) error { |  | ||||||
| 	if len(s.testedSuite.features) != num { |  | ||||||
| 		return fmt.Errorf("expected %d features to be parsed, but have %d", num, len(s.testedSuite.features)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	expected := strings.Split(files.Content, "\n") |  | ||||||
| 
 |  | ||||||
| 	var actual []string |  | ||||||
| 
 |  | ||||||
| 	for _, ft := range s.testedSuite.features { |  | ||||||
| 		actual = append(actual, ft.Uri) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if len(expected) != len(actual) { |  | ||||||
| 		return fmt.Errorf("expected %d feature paths to be parsed, but have %d", len(expected), len(actual)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for i := 0; i < len(expected); i++ { |  | ||||||
| 		var matched bool |  | ||||||
| 		split := strings.Split(expected[i], "/") |  | ||||||
| 		exp := filepath.Join(split...) |  | ||||||
| 
 |  | ||||||
| 		for j := 0; j < len(actual); j++ { |  | ||||||
| 			split = strings.Split(actual[j], "/") |  | ||||||
| 			act := filepath.Join(split...) |  | ||||||
| 
 |  | ||||||
| 			if exp == act { |  | ||||||
| 				matched = true |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if !matched { |  | ||||||
| 			return fmt.Errorf(`expected feature path "%s" at position: %d, was not parsed, actual are %+v`, exp, i, actual) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) iRunFeatureSuite() error { |  | ||||||
| 	return s.iRunFeatureSuiteWithTags("") |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) numScenariosRegistered(expected int) (err error) { |  | ||||||
| 	var num int |  | ||||||
| 	for _, ft := range s.testedSuite.features { |  | ||||||
| 		num += len(ft.pickles) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if num != expected { |  | ||||||
| 		err = fmt.Errorf("expected %d scenarios to be registered, but got %d", expected, num) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) thereWereNumEventsFired(_ string, expected int, typ string) error { |  | ||||||
| 	var num int |  | ||||||
| 	for _, event := range s.events { |  | ||||||
| 		if event.name == typ { |  | ||||||
| 			num++ |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if num != expected { |  | ||||||
| 		return fmt.Errorf("expected %d %s events to be fired, but got %d", expected, typ, num) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) thereWasEventTriggeredBeforeScenario(expected string) error { |  | ||||||
| 	var found []string |  | ||||||
| 	for _, event := range s.events { |  | ||||||
| 		if event.name != "BeforeScenario" { |  | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		var name string |  | ||||||
| 		switch t := event.args[0].(type) { |  | ||||||
| 		case *Scenario: |  | ||||||
| 			name = t.Name |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if name == expected { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		found = append(found, name) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if len(found) == 0 { |  | ||||||
| 		return fmt.Errorf("before scenario event was never triggered or listened") |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return fmt.Errorf(`expected "%s" scenario, but got these fired %s`, expected, `"`+strings.Join(found, `", "`)+`"`) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theseEventsHadToBeFiredForNumberOfTimes(tbl *Table) error { |  | ||||||
| 	if len(tbl.Rows[0].Cells) != 2 { |  | ||||||
| 		return fmt.Errorf("expected two columns for event table row, got: %d", len(tbl.Rows[0].Cells)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, row := range tbl.Rows { |  | ||||||
| 		num, err := strconv.ParseInt(row.Cells[1].Value, 10, 0) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if err := s.thereWereNumEventsFired("", int(num), row.Cells[0].Value); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theRenderJSONWillBe(docstring *DocString) error { |  | ||||||
| 	suiteCtxReg := regexp.MustCompile(`suite_context.go:\d+`) |  | ||||||
| 
 |  | ||||||
| 	expectedString := docstring.Content |  | ||||||
| 	expectedString = suiteCtxReg.ReplaceAllString(expectedString, `suite_context.go:0`) |  | ||||||
| 
 |  | ||||||
| 	actualString := s.out.String() |  | ||||||
| 	actualString = suiteCtxReg.ReplaceAllString(actualString, `suite_context.go:0`) |  | ||||||
| 
 |  | ||||||
| 	var expected []cukeFeatureJSON |  | ||||||
| 	if err := json.Unmarshal([]byte(expectedString), &expected); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	var actual []cukeFeatureJSON |  | ||||||
| 	if err := json.Unmarshal([]byte(actualString), &actual); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return assertExpectedAndActual(assert.Equal, expected, actual) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theRenderOutputWillBe(docstring *DocString) error { |  | ||||||
| 	suiteCtxReg := regexp.MustCompile(`suite_context.go:\d+`) |  | ||||||
| 	suiteCtxFuncReg := regexp.MustCompile(`github.com/cucumber/godog.SuiteContext.func(\d+)`) |  | ||||||
| 
 |  | ||||||
| 	expected := docstring.Content |  | ||||||
| 	expected = trimAllLines(expected) |  | ||||||
| 	expected = suiteCtxReg.ReplaceAllString(expected, "suite_context.go:0") |  | ||||||
| 	expected = suiteCtxFuncReg.ReplaceAllString(expected, "SuiteContext.func$1") |  | ||||||
| 
 |  | ||||||
| 	actual := s.out.String() |  | ||||||
| 	actual = trimAllLines(actual) |  | ||||||
| 	actual = suiteCtxReg.ReplaceAllString(actual, "suite_context.go:0") |  | ||||||
| 	actual = suiteCtxFuncReg.ReplaceAllString(actual, "SuiteContext.func$1") |  | ||||||
| 
 |  | ||||||
| 	expectedRows := strings.Split(expected, "\n") |  | ||||||
| 	actualRows := strings.Split(actual, "\n") |  | ||||||
| 
 |  | ||||||
| 	return assertExpectedAndActual(assert.ElementsMatch, expectedRows, actualRows) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (s *suiteContext) theRenderXMLWillBe(docstring *DocString) error { |  | ||||||
| 	expectedString := docstring.Content |  | ||||||
| 	actualString := s.out.String() |  | ||||||
| 
 |  | ||||||
| 	var expected junitPackageSuite |  | ||||||
| 	if err := xml.Unmarshal([]byte(expectedString), &expected); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	var actual junitPackageSuite |  | ||||||
| 	if err := xml.Unmarshal([]byte(actualString), &actual); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return assertExpectedAndActual(assert.Equal, expected, actual) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func assertExpectedAndActual(a expectedAndActualAssertion, expected, actual interface{}, msgAndArgs ...interface{}) error { |  | ||||||
| 	var t asserter |  | ||||||
| 	a(&t, expected, actual, msgAndArgs...) |  | ||||||
| 	return t.err |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type expectedAndActualAssertion func(t assert.TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool |  | ||||||
| 
 |  | ||||||
| type asserter struct { |  | ||||||
| 	err error |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (a *asserter) Errorf(format string, args ...interface{}) { |  | ||||||
| 	a.err = fmt.Errorf(format, args...) |  | ||||||
| } |  | ||||||
|  | @ -117,9 +117,31 @@ func (tc *godogFeaturesScenario) inject(step *Step) { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func injectAll(src string) string { | ||||||
|  | 	re := regexp.MustCompile(`{{[^{}]+}}`) | ||||||
|  | 	return re.ReplaceAllStringFunc( | ||||||
|  | 		src, | ||||||
|  | 		func(key string) string { | ||||||
|  | 			injectRegex := regexp.MustCompile(`^{{.+}}$`) | ||||||
|  | 
 | ||||||
|  | 			if injectRegex.MatchString(key) { | ||||||
|  | 				return "someverylonginjectionsoweacanbesureitsurpasstheinitiallongeststeplenghtanditwillhelptestsmethodsafety" | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			return key | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type firedEvent struct { | ||||||
|  | 	name string | ||||||
|  | 	args []interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type godogFeaturesScenario struct { | type godogFeaturesScenario struct { | ||||||
| 	paths            []string | 	paths            []string | ||||||
| 	testedSuite      *Suite | 	features         []*feature | ||||||
|  | 	testedSuite      *suite | ||||||
| 	testSuiteContext TestSuiteContext | 	testSuiteContext TestSuiteContext | ||||||
| 	events           []*firedEvent | 	events           []*firedEvent | ||||||
| 	out              bytes.Buffer | 	out              bytes.Buffer | ||||||
|  | @ -131,7 +153,8 @@ func (tc *godogFeaturesScenario) ResetBeforeEachScenario(*Scenario) { | ||||||
| 	tc.out.Reset() | 	tc.out.Reset() | ||||||
| 	tc.paths = []string{} | 	tc.paths = []string{} | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite = &Suite{} | 	tc.features = []*feature{} | ||||||
|  | 	tc.testedSuite = &suite{} | ||||||
| 	tc.testSuiteContext = TestSuiteContext{} | 	tc.testSuiteContext = TestSuiteContext{} | ||||||
| 
 | 
 | ||||||
| 	// reset all fired events | 	// reset all fired events | ||||||
|  | @ -162,12 +185,12 @@ func (tc *godogFeaturesScenario) iRunFeatureSuiteWithTagsAndFormatter(tags strin | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for _, feat := range tc.testedSuite.features { | 	for _, feat := range tc.features { | ||||||
| 		feat.pickles = applyTagFilter(tags, feat.pickles) | 		feat.pickles = applyTagFilter(tags, feat.pickles) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.storage = newStorage() | 	tc.testedSuite.storage = newStorage() | ||||||
| 	for _, feat := range tc.testedSuite.features { | 	for _, feat := range tc.features { | ||||||
| 		tc.testedSuite.storage.mustInsertFeature(feat) | 		tc.testedSuite.storage.mustInsertFeature(feat) | ||||||
| 
 | 
 | ||||||
| 		for _, pickle := range feat.pickles { | 		for _, pickle := range feat.pickles { | ||||||
|  | @ -188,7 +211,7 @@ func (tc *godogFeaturesScenario) iRunFeatureSuiteWithTagsAndFormatter(tags strin | ||||||
| 		f() | 		f() | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for _, ft := range tc.testedSuite.features { | 	for _, ft := range tc.features { | ||||||
| 		tc.testedSuite.fmt.Feature(ft.GherkinDocument, ft.Uri, ft.content) | 		tc.testedSuite.fmt.Feature(ft.GherkinDocument, ft.Uri, ft.content) | ||||||
| 
 | 
 | ||||||
| 		for _, pickle := range ft.pickles { | 		for _, pickle := range ft.pickles { | ||||||
|  | @ -350,19 +373,21 @@ func (tc *godogFeaturesScenario) iAmListeningToSuiteEvents() error { | ||||||
| 		tc.events = append(tc.events, &firedEvent{"AfterSuite", []interface{}{}}) | 		tc.events = append(tc.events, &firedEvent{"AfterSuite", []interface{}{}}) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.BeforeScenario(func(pickle *Scenario) { | 	scenarioContext := ScenarioContext{suite: tc.testedSuite} | ||||||
|  | 
 | ||||||
|  | 	scenarioContext.BeforeScenario(func(pickle *Scenario) { | ||||||
| 		tc.events = append(tc.events, &firedEvent{"BeforeScenario", []interface{}{pickle}}) | 		tc.events = append(tc.events, &firedEvent{"BeforeScenario", []interface{}{pickle}}) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.AfterScenario(func(pickle *Scenario, err error) { | 	scenarioContext.AfterScenario(func(pickle *Scenario, err error) { | ||||||
| 		tc.events = append(tc.events, &firedEvent{"AfterScenario", []interface{}{pickle, err}}) | 		tc.events = append(tc.events, &firedEvent{"AfterScenario", []interface{}{pickle, err}}) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.BeforeStep(func(step *Step) { | 	scenarioContext.BeforeStep(func(step *Step) { | ||||||
| 		tc.events = append(tc.events, &firedEvent{"BeforeStep", []interface{}{step}}) | 		tc.events = append(tc.events, &firedEvent{"BeforeStep", []interface{}{step}}) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.AfterStep(func(step *Step, err error) { | 	scenarioContext.AfterStep(func(step *Step, err error) { | ||||||
| 		tc.events = append(tc.events, &firedEvent{"AfterStep", []interface{}{step, err}}) | 		tc.events = append(tc.events, &firedEvent{"AfterStep", []interface{}{step, err}}) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | @ -379,7 +404,7 @@ func (tc *godogFeaturesScenario) aFeatureFile(path string, body *DocString) erro | ||||||
| 	gd.Uri = path | 	gd.Uri = path | ||||||
| 
 | 
 | ||||||
| 	pickles := gherkin.Pickles(*gd, path, (&messages.Incrementing{}).NewId) | 	pickles := gherkin.Pickles(*gd, path, (&messages.Incrementing{}).NewId) | ||||||
| 	tc.testedSuite.features = append(tc.testedSuite.features, &feature{GherkinDocument: gd, pickles: pickles}) | 	tc.features = append(tc.features, &feature{GherkinDocument: gd, pickles: pickles}) | ||||||
| 
 | 
 | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  | @ -395,7 +420,7 @@ func (tc *godogFeaturesScenario) parseFeatures() error { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	tc.testedSuite.features = append(tc.testedSuite.features, fts...) | 	tc.features = append(tc.features, fts...) | ||||||
| 
 | 
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  | @ -413,15 +438,15 @@ func (tc *godogFeaturesScenario) theSuiteShouldHave(state string) error { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (tc *godogFeaturesScenario) iShouldHaveNumFeatureFiles(num int, files *DocString) error { | func (tc *godogFeaturesScenario) iShouldHaveNumFeatureFiles(num int, files *DocString) error { | ||||||
| 	if len(tc.testedSuite.features) != num { | 	if len(tc.features) != num { | ||||||
| 		return fmt.Errorf("expected %d features to be parsed, but have %d", num, len(tc.testedSuite.features)) | 		return fmt.Errorf("expected %d features to be parsed, but have %d", num, len(tc.features)) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	expected := strings.Split(files.Content, "\n") | 	expected := strings.Split(files.Content, "\n") | ||||||
| 
 | 
 | ||||||
| 	var actual []string | 	var actual []string | ||||||
| 
 | 
 | ||||||
| 	for _, ft := range tc.testedSuite.features { | 	for _, ft := range tc.features { | ||||||
| 		actual = append(actual, ft.Uri) | 		actual = append(actual, ft.Uri) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -458,7 +483,7 @@ func (tc *godogFeaturesScenario) iRunFeatureSuite() error { | ||||||
| 
 | 
 | ||||||
| func (tc *godogFeaturesScenario) numScenariosRegistered(expected int) (err error) { | func (tc *godogFeaturesScenario) numScenariosRegistered(expected int) (err error) { | ||||||
| 	var num int | 	var num int | ||||||
| 	for _, ft := range tc.testedSuite.features { | 	for _, ft := range tc.features { | ||||||
| 		num += len(ft.pickles) | 		num += len(ft.pickles) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -599,3 +624,19 @@ func (tc *godogFeaturesScenario) theRenderXMLWillBe(docstring *DocString) error | ||||||
| 
 | 
 | ||||||
| 	return assertExpectedAndActual(assert.Equal, expected, actual) | 	return assertExpectedAndActual(assert.Equal, expected, actual) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func assertExpectedAndActual(a expectedAndActualAssertion, expected, actual interface{}, msgAndArgs ...interface{}) error { | ||||||
|  | 	var t asserter | ||||||
|  | 	a(&t, expected, actual, msgAndArgs...) | ||||||
|  | 	return t.err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type expectedAndActualAssertion func(t assert.TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool | ||||||
|  | 
 | ||||||
|  | type asserter struct { | ||||||
|  | 	err error | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (a *asserter) Errorf(format string, args ...interface{}) { | ||||||
|  | 	a.err = fmt.Errorf(format, args...) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -1,6 +0,0 @@ | ||||||
| package godog |  | ||||||
| 
 |  | ||||||
| // needed in order to use godog cli |  | ||||||
| func GodogContext(s *Suite) { |  | ||||||
| 	SuiteContext(s) |  | ||||||
| } |  | ||||||
|  | @ -1,6 +1,12 @@ | ||||||
| package godog | package godog | ||||||
| 
 | 
 | ||||||
| import "github.com/cucumber/messages-go/v10" | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"reflect" | ||||||
|  | 	"regexp" | ||||||
|  | 
 | ||||||
|  | 	"github.com/cucumber/messages-go/v10" | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| // Scenario represents the executed scenario | // Scenario represents the executed scenario | ||||||
| type Scenario = messages.Pickle | type Scenario = messages.Pickle | ||||||
|  | @ -55,7 +61,7 @@ func (ctx *TestSuiteContext) AfterSuite(fn func()) { | ||||||
| // executions are catching panic error since it may | // executions are catching panic error since it may | ||||||
| // be a context specific error. | // be a context specific error. | ||||||
| type ScenarioContext struct { | type ScenarioContext struct { | ||||||
| 	suite *Suite | 	suite *suite | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // BeforeScenario registers a function or method | // BeforeScenario registers a function or method | ||||||
|  | @ -65,19 +71,19 @@ type ScenarioContext struct { | ||||||
| // before every scenario so it would be isolated from | // before every scenario so it would be isolated from | ||||||
| // any kind of state. | // any kind of state. | ||||||
| func (ctx *ScenarioContext) BeforeScenario(fn func(sc *Scenario)) { | func (ctx *ScenarioContext) BeforeScenario(fn func(sc *Scenario)) { | ||||||
| 	ctx.suite.BeforeScenario(fn) | 	ctx.suite.beforeScenarioHandlers = append(ctx.suite.beforeScenarioHandlers, fn) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AfterScenario registers an function or method | // AfterScenario registers an function or method | ||||||
| // to be run after every scenario. | // to be run after every scenario. | ||||||
| func (ctx *ScenarioContext) AfterScenario(fn func(sc *Scenario, err error)) { | func (ctx *ScenarioContext) AfterScenario(fn func(sc *Scenario, err error)) { | ||||||
| 	ctx.suite.AfterScenario(fn) | 	ctx.suite.afterScenarioHandlers = append(ctx.suite.afterScenarioHandlers, fn) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // BeforeStep registers a function or method | // BeforeStep registers a function or method | ||||||
| // to be run before every step. | // to be run before every step. | ||||||
| func (ctx *ScenarioContext) BeforeStep(fn func(st *Step)) { | func (ctx *ScenarioContext) BeforeStep(fn func(st *Step)) { | ||||||
| 	ctx.suite.BeforeStep(fn) | 	ctx.suite.beforeStepHandlers = append(ctx.suite.beforeStepHandlers, fn) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AfterStep registers an function or method | // AfterStep registers an function or method | ||||||
|  | @ -90,7 +96,7 @@ func (ctx *ScenarioContext) BeforeStep(fn func(st *Step)) { | ||||||
| // In some cases, for example when running a headless | // In some cases, for example when running a headless | ||||||
| // browser, to take a screenshot after failure. | // browser, to take a screenshot after failure. | ||||||
| func (ctx *ScenarioContext) AfterStep(fn func(st *Step, err error)) { | func (ctx *ScenarioContext) AfterStep(fn func(st *Step, err error)) { | ||||||
| 	ctx.suite.AfterStep(fn) | 	ctx.suite.afterStepHandlers = append(ctx.suite.afterStepHandlers, fn) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Step allows to register a *StepDefinition in the | // Step allows to register a *StepDefinition in the | ||||||
|  | @ -121,5 +127,49 @@ func (ctx *ScenarioContext) AfterStep(fn func(st *Step, err error)) { | ||||||
| // ErrUndefined error will be returned when | // ErrUndefined error will be returned when | ||||||
| // running steps. | // running steps. | ||||||
| func (ctx *ScenarioContext) Step(expr, stepFunc interface{}) { | func (ctx *ScenarioContext) Step(expr, stepFunc interface{}) { | ||||||
| 	ctx.suite.Step(expr, stepFunc) | 	var regex *regexp.Regexp | ||||||
|  | 
 | ||||||
|  | 	switch t := expr.(type) { | ||||||
|  | 	case *regexp.Regexp: | ||||||
|  | 		regex = t | ||||||
|  | 	case string: | ||||||
|  | 		regex = regexp.MustCompile(t) | ||||||
|  | 	case []byte: | ||||||
|  | 		regex = regexp.MustCompile(string(t)) | ||||||
|  | 	default: | ||||||
|  | 		panic(fmt.Sprintf("expecting expr to be a *regexp.Regexp or a string, got type: %T", expr)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	v := reflect.ValueOf(stepFunc) | ||||||
|  | 	typ := v.Type() | ||||||
|  | 	if typ.Kind() != reflect.Func { | ||||||
|  | 		panic(fmt.Sprintf("expected handler to be func, but got: %T", stepFunc)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if typ.NumOut() != 1 { | ||||||
|  | 		panic(fmt.Sprintf("expected handler to return only one value, but it has: %d", typ.NumOut())) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	def := &StepDefinition{ | ||||||
|  | 		Handler: stepFunc, | ||||||
|  | 		Expr:    regex, | ||||||
|  | 		hv:      v, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	typ = typ.Out(0) | ||||||
|  | 	switch typ.Kind() { | ||||||
|  | 	case reflect.Interface: | ||||||
|  | 		if !typ.Implements(errorInterface) { | ||||||
|  | 			panic(fmt.Sprintf("expected handler to return an error, but got: %s", typ.Kind())) | ||||||
|  | 		} | ||||||
|  | 	case reflect.Slice: | ||||||
|  | 		if typ.Elem().Kind() != reflect.String { | ||||||
|  | 			panic(fmt.Sprintf("expected handler to return []string for multistep, but got: []%s", typ.Kind())) | ||||||
|  | 		} | ||||||
|  | 		def.nested = true | ||||||
|  | 	default: | ||||||
|  | 		panic(fmt.Sprintf("expected handler to return an error or []string, but got: %s", typ.Kind())) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	ctx.suite.steps = append(ctx.suite.steps, def) | ||||||
| } | } | ||||||
|  |  | ||||||
		Загрузка…
	
	Создание таблицы
		
		Сослаться в новой задаче
	
	 Fredrik Lönnblad
						Fredrik Lönnblad