cucumber 3.1.2 → 7.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/CHANGELOG.md +1780 -1146
- data/CONTRIBUTING.md +224 -61
- data/README.md +144 -22
- data/bin/cucumber +1 -1
- data/lib/autotest/cucumber_mixin.rb +46 -53
- data/lib/cucumber/cli/configuration.rb +28 -6
- data/lib/cucumber/cli/main.rb +12 -12
- data/lib/cucumber/cli/options.rb +103 -77
- data/lib/cucumber/cli/profile_loader.rb +49 -26
- data/lib/cucumber/configuration.rb +44 -29
- data/lib/cucumber/constantize.rb +2 -5
- data/lib/cucumber/deprecate.rb +31 -7
- data/lib/cucumber/errors.rb +5 -7
- data/lib/cucumber/events/envelope.rb +9 -0
- data/lib/cucumber/events/gherkin_source_parsed.rb +11 -0
- data/lib/cucumber/events/hook_test_step_created.rb +13 -0
- data/lib/cucumber/events/step_activated.rb +2 -1
- data/lib/cucumber/events/test_case_created.rb +13 -0
- data/lib/cucumber/events/test_case_ready.rb +12 -0
- data/lib/cucumber/events/test_step_created.rb +13 -0
- data/lib/cucumber/events/undefined_parameter_type.rb +10 -0
- data/lib/cucumber/events.rb +14 -7
- data/lib/cucumber/file_specs.rb +6 -6
- data/lib/cucumber/filters/activate_steps.rb +5 -3
- data/lib/cucumber/filters/broadcast_test_case_ready_event.rb +12 -0
- data/lib/cucumber/filters/prepare_world.rb +5 -9
- data/lib/cucumber/filters/quit.rb +1 -3
- data/lib/cucumber/filters/tag_limits/verifier.rb +2 -4
- data/lib/cucumber/filters.rb +1 -0
- data/lib/cucumber/formatter/ansicolor.rb +40 -52
- data/lib/cucumber/formatter/ast_lookup.rb +163 -0
- data/lib/cucumber/formatter/backtrace_filter.rb +10 -8
- data/lib/cucumber/formatter/console.rb +69 -69
- data/lib/cucumber/formatter/console_counts.rb +4 -9
- data/lib/cucumber/formatter/console_issues.rb +6 -3
- data/lib/cucumber/formatter/duration.rb +1 -1
- data/lib/cucumber/formatter/duration_extractor.rb +3 -1
- data/lib/cucumber/formatter/errors.rb +6 -0
- data/lib/cucumber/formatter/fanout.rb +2 -0
- data/lib/cucumber/formatter/html.rb +11 -598
- data/lib/cucumber/formatter/http_io.rb +147 -0
- data/lib/cucumber/formatter/ignore_missing_messages.rb +1 -1
- data/lib/cucumber/formatter/interceptor.rb +11 -30
- data/lib/cucumber/formatter/io.rb +55 -13
- data/lib/cucumber/formatter/json.rb +115 -122
- data/lib/cucumber/formatter/junit.rb +72 -55
- data/lib/cucumber/formatter/message.rb +23 -0
- data/lib/cucumber/formatter/message_builder.rb +255 -0
- data/lib/cucumber/formatter/pretty.rb +360 -153
- data/lib/cucumber/formatter/progress.rb +30 -32
- data/lib/cucumber/formatter/publish_banner_printer.rb +77 -0
- data/lib/cucumber/formatter/query/hook_by_test_step.rb +31 -0
- data/lib/cucumber/formatter/query/pickle_by_test.rb +26 -0
- data/lib/cucumber/formatter/query/pickle_step_by_test_step.rb +26 -0
- data/lib/cucumber/formatter/query/step_definitions_by_test_step.rb +40 -0
- data/lib/cucumber/formatter/query/test_case_started_by_test_case.rb +40 -0
- data/lib/cucumber/formatter/rerun.rb +22 -4
- data/lib/cucumber/formatter/stepdefs.rb +1 -2
- data/lib/cucumber/formatter/steps.rb +8 -6
- data/lib/cucumber/formatter/summary.rb +16 -8
- data/lib/cucumber/formatter/unicode.rb +15 -17
- data/lib/cucumber/formatter/url_reporter.rb +17 -0
- data/lib/cucumber/formatter/usage.rb +17 -14
- data/lib/cucumber/gherkin/data_table_parser.rb +17 -6
- data/lib/cucumber/gherkin/formatter/ansi_escapes.rb +13 -17
- data/lib/cucumber/gherkin/formatter/escaping.rb +2 -2
- data/lib/cucumber/gherkin/steps_parser.rb +17 -8
- data/lib/cucumber/glue/dsl.rb +19 -0
- data/lib/cucumber/glue/hook.rb +34 -11
- data/lib/cucumber/glue/invoke_in_world.rb +13 -18
- data/lib/cucumber/glue/proto_world.rb +37 -44
- data/lib/cucumber/glue/registry_and_more.rb +71 -12
- data/lib/cucumber/glue/registry_wrapper.rb +31 -0
- data/lib/cucumber/glue/snippet.rb +23 -22
- data/lib/cucumber/glue/step_definition.rb +42 -20
- data/lib/cucumber/glue/world_factory.rb +1 -1
- data/lib/cucumber/hooks.rb +11 -11
- data/lib/cucumber/multiline_argument/data_table/diff_matrices.rb +2 -2
- data/lib/cucumber/multiline_argument/data_table.rb +97 -64
- data/lib/cucumber/multiline_argument/doc_string.rb +1 -1
- data/lib/cucumber/multiline_argument.rb +4 -6
- data/lib/cucumber/platform.rb +3 -3
- data/lib/cucumber/rake/task.rb +16 -18
- data/lib/cucumber/rspec/disable_option_parser.rb +9 -8
- data/lib/cucumber/rspec/doubles.rb +3 -5
- data/lib/cucumber/running_test_case.rb +2 -53
- data/lib/cucumber/runtime/after_hooks.rb +8 -4
- data/lib/cucumber/runtime/before_hooks.rb +8 -4
- data/lib/cucumber/runtime/for_programming_languages.rb +4 -2
- data/lib/cucumber/runtime/step_hooks.rb +6 -2
- data/lib/cucumber/runtime/support_code.rb +13 -15
- data/lib/cucumber/runtime/user_interface.rb +6 -16
- data/lib/cucumber/runtime.rb +77 -59
- data/lib/cucumber/step_definition_light.rb +4 -3
- data/lib/cucumber/step_definitions.rb +2 -2
- data/lib/cucumber/step_match.rb +12 -17
- data/lib/cucumber/step_match_search.rb +2 -1
- data/lib/cucumber/term/ansicolor.rb +9 -9
- data/lib/cucumber/term/banner.rb +56 -0
- data/lib/cucumber/version +1 -1
- data/lib/cucumber.rb +1 -1
- metadata +272 -81
- data/lib/cucumber/core_ext/string.rb +0 -11
- data/lib/cucumber/events/gherkin_source_parsed.rb~ +0 -14
- data/lib/cucumber/formatter/ast_lookup.rb~ +0 -9
- data/lib/cucumber/formatter/cucumber.css +0 -286
- data/lib/cucumber/formatter/cucumber.sass +0 -247
- data/lib/cucumber/formatter/hook_query_visitor.rb +0 -42
- data/lib/cucumber/formatter/html_builder.rb +0 -121
- data/lib/cucumber/formatter/inline-js.js +0 -30
- data/lib/cucumber/formatter/jquery-min.js +0 -154
- data/lib/cucumber/formatter/json_pretty.rb +0 -11
- data/lib/cucumber/formatter/legacy_api/adapter.rb +0 -1028
- data/lib/cucumber/formatter/legacy_api/ast.rb +0 -394
- data/lib/cucumber/formatter/legacy_api/results.rb +0 -50
- data/lib/cucumber/formatter/legacy_api/runtime_facade.rb +0 -32
- data/lib/cucumber/step_argument.rb +0 -25
@@ -0,0 +1,163 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Cucumber
|
4
|
+
module Formatter
|
5
|
+
class AstLookup
|
6
|
+
def initialize(config)
|
7
|
+
@gherkin_documents = {}
|
8
|
+
@test_case_lookups = {}
|
9
|
+
@test_step_lookups = {}
|
10
|
+
@step_keyword_lookups = {}
|
11
|
+
config.on_event :gherkin_source_parsed, &method(:on_gherkin_source_parsed)
|
12
|
+
end
|
13
|
+
|
14
|
+
def on_gherkin_source_parsed(event)
|
15
|
+
@gherkin_documents[event.gherkin_document.uri] = event.gherkin_document
|
16
|
+
end
|
17
|
+
|
18
|
+
def gherkin_document(uri)
|
19
|
+
@gherkin_documents[uri]
|
20
|
+
end
|
21
|
+
|
22
|
+
def scenario_source(test_case)
|
23
|
+
uri = test_case.location.file
|
24
|
+
@test_case_lookups[uri] ||= TestCaseLookupBuilder.new(gherkin_document(uri)).lookup_hash
|
25
|
+
@test_case_lookups[uri][test_case.location.lines.max]
|
26
|
+
end
|
27
|
+
|
28
|
+
def step_source(test_step)
|
29
|
+
uri = test_step.location.file
|
30
|
+
@test_step_lookups[uri] ||= TestStepLookupBuilder.new(gherkin_document(uri)).lookup_hash
|
31
|
+
@test_step_lookups[uri][test_step.location.lines.min]
|
32
|
+
end
|
33
|
+
|
34
|
+
def snippet_step_keyword(test_step)
|
35
|
+
uri = test_step.location.file
|
36
|
+
document = gherkin_document(uri)
|
37
|
+
dialect = ::Gherkin::Dialect.for(document.feature.language)
|
38
|
+
given_when_then_keywords = [dialect.given_keywords, dialect.when_keywords, dialect.then_keywords].flatten.uniq.reject { |kw| kw == '* ' }
|
39
|
+
keyword_lookup = step_keyword_lookup(uri)
|
40
|
+
keyword = nil
|
41
|
+
node = keyword_lookup[test_step.location.lines.min]
|
42
|
+
while keyword.nil?
|
43
|
+
if given_when_then_keywords.include?(node.keyword)
|
44
|
+
keyword = node.keyword
|
45
|
+
break
|
46
|
+
end
|
47
|
+
break if node.previous_node.nil?
|
48
|
+
node = node.previous_node
|
49
|
+
end
|
50
|
+
keyword = dialect.given_keywords.reject { |kw| kw == '* ' }[0] if keyword.nil?
|
51
|
+
keyword = Cucumber::Gherkin::I18n.code_keyword_for(keyword)
|
52
|
+
keyword
|
53
|
+
end
|
54
|
+
|
55
|
+
ScenarioSource = Struct.new(:type, :scenario)
|
56
|
+
|
57
|
+
ScenarioOutlineSource = Struct.new(:type, :scenario_outline, :examples, :row)
|
58
|
+
|
59
|
+
StepSource = Struct.new(:type, :step)
|
60
|
+
|
61
|
+
private
|
62
|
+
|
63
|
+
def step_keyword_lookup(uri)
|
64
|
+
@step_keyword_lookups[uri] ||= KeywordLookupBuilder.new(gherkin_document(uri)).lookup_hash
|
65
|
+
end
|
66
|
+
|
67
|
+
class TestCaseLookupBuilder
|
68
|
+
attr_reader :lookup_hash
|
69
|
+
|
70
|
+
def initialize(gherkin_document)
|
71
|
+
@lookup_hash = {}
|
72
|
+
process_scenario_container(gherkin_document.feature)
|
73
|
+
end
|
74
|
+
|
75
|
+
private
|
76
|
+
|
77
|
+
def process_scenario_container(container)
|
78
|
+
container.children.each do |child|
|
79
|
+
if child.respond_to?(:rule) && child.rule
|
80
|
+
process_scenario_container(child.rule)
|
81
|
+
elsif child.respond_to?(:scenario) && child.scenario
|
82
|
+
process_scenario(child)
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|
86
|
+
|
87
|
+
def process_scenario(child)
|
88
|
+
if child.scenario.examples.empty?
|
89
|
+
@lookup_hash[child.scenario.location.line] = ScenarioSource.new(:Scenario, child.scenario)
|
90
|
+
else
|
91
|
+
child.scenario.examples.each do |examples|
|
92
|
+
examples.table_body.each do |row|
|
93
|
+
@lookup_hash[row.location.line] = ScenarioOutlineSource.new(:ScenarioOutline, child.scenario, examples, row)
|
94
|
+
end
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
100
|
+
class TestStepLookupBuilder
|
101
|
+
attr_reader :lookup_hash
|
102
|
+
|
103
|
+
def initialize(gherkin_document)
|
104
|
+
@lookup_hash = {}
|
105
|
+
process_scenario_container(gherkin_document.feature)
|
106
|
+
end
|
107
|
+
|
108
|
+
private
|
109
|
+
|
110
|
+
def process_scenario_container(container)
|
111
|
+
container.children.each do |child|
|
112
|
+
if child.respond_to?(:rule) && child.rule
|
113
|
+
process_scenario_container(child.rule)
|
114
|
+
elsif child.respond_to?(:scenario) && child.scenario
|
115
|
+
child.scenario.steps.each do |step|
|
116
|
+
@lookup_hash[step.location.line] = StepSource.new(:Step, step)
|
117
|
+
end
|
118
|
+
elsif !child.background.nil?
|
119
|
+
child.background.steps.each do |step|
|
120
|
+
@lookup_hash[step.location.line] = StepSource.new(:Step, step)
|
121
|
+
end
|
122
|
+
end
|
123
|
+
end
|
124
|
+
end
|
125
|
+
end
|
126
|
+
|
127
|
+
KeywordSearchNode = Struct.new(:keyword, :previous_node)
|
128
|
+
|
129
|
+
class KeywordLookupBuilder
|
130
|
+
attr_reader :lookup_hash
|
131
|
+
|
132
|
+
def initialize(gherkin_document)
|
133
|
+
@lookup_hash = {}
|
134
|
+
process_scenario_container(gherkin_document.feature, nil)
|
135
|
+
end
|
136
|
+
|
137
|
+
private
|
138
|
+
|
139
|
+
def process_scenario_container(container, original_previous_node)
|
140
|
+
container.children.each do |child|
|
141
|
+
previous_node = original_previous_node
|
142
|
+
if child.respond_to?(:rule) && child.rule
|
143
|
+
process_scenario_container(child.rule, original_previous_node)
|
144
|
+
elsif child.respond_to?(:scenario) && child.scenario
|
145
|
+
child.scenario.steps.each do |step|
|
146
|
+
node = KeywordSearchNode.new(step.keyword, previous_node)
|
147
|
+
@lookup_hash[step.location.line] = node
|
148
|
+
previous_node = node
|
149
|
+
end
|
150
|
+
elsif child.respond_to?(:background) && child.background
|
151
|
+
child.background.steps.each do |step|
|
152
|
+
node = KeywordSearchNode.new(step.keyword, previous_node)
|
153
|
+
@lookup_hash[step.location.line] = node
|
154
|
+
previous_node = node
|
155
|
+
original_previous_node = previous_node
|
156
|
+
end
|
157
|
+
end
|
158
|
+
end
|
159
|
+
end
|
160
|
+
end
|
161
|
+
end
|
162
|
+
end
|
163
|
+
end
|
@@ -4,22 +4,24 @@ require 'cucumber/platform'
|
|
4
4
|
|
5
5
|
module Cucumber
|
6
6
|
module Formatter
|
7
|
-
@backtrace_filters = %w
|
7
|
+
@backtrace_filters = %w[
|
8
8
|
/vendor/rails
|
9
9
|
lib/cucumber
|
10
10
|
bin/cucumber:
|
11
11
|
lib/rspec
|
12
12
|
gems/
|
13
|
+
site_ruby/
|
13
14
|
minitest
|
14
15
|
test/unit
|
15
16
|
.gem/ruby
|
16
|
-
lib/ruby/
|
17
17
|
bin/bundle
|
18
|
-
|
18
|
+
rdebug-ide
|
19
|
+
]
|
19
20
|
|
20
|
-
if ::
|
21
|
-
|
22
|
-
|
21
|
+
@backtrace_filters << RbConfig::CONFIG['rubyarchdir'] if RbConfig::CONFIG['rubyarchdir']
|
22
|
+
@backtrace_filters << RbConfig::CONFIG['rubylibdir'] if RbConfig::CONFIG['rubylibdir']
|
23
|
+
|
24
|
+
@backtrace_filters << 'org/jruby/' if ::Cucumber::JRUBY
|
23
25
|
|
24
26
|
BACKTRACE_FILTER_PATTERNS = Regexp.new(@backtrace_filters.join('|'))
|
25
27
|
|
@@ -31,7 +33,7 @@ module Cucumber
|
|
31
33
|
def exception
|
32
34
|
return @exception if ::Cucumber.use_full_backtrace
|
33
35
|
|
34
|
-
pwd_pattern = /#{::Regexp.escape(::Dir.pwd)}\//m
|
36
|
+
pwd_pattern = /#{::Regexp.escape(::Dir.pwd)}\//m # rubocop:disable Style/RegexpLiteral
|
35
37
|
backtrace = @exception.backtrace.map { |line| line.gsub(pwd_pattern, './') }
|
36
38
|
|
37
39
|
filtered = (backtrace || []).reject do |line|
|
@@ -41,7 +43,7 @@ module Cucumber
|
|
41
43
|
if ::ENV['CUCUMBER_TRUNCATE_OUTPUT']
|
42
44
|
# Strip off file locations
|
43
45
|
filtered = filtered.map do |line|
|
44
|
-
line =~ /(.*):in `/ ?
|
46
|
+
line =~ /(.*):in `/ ? Regexp.last_match(1) : line
|
45
47
|
end
|
46
48
|
end
|
47
49
|
|
@@ -1,5 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
# rubocop:disable Metrics/ModuleLength
|
4
|
+
|
3
5
|
require 'cucumber/formatter/ansicolor'
|
4
6
|
require 'cucumber/formatter/duration'
|
5
7
|
require 'cucumber/gherkin/i18n'
|
@@ -32,7 +34,7 @@ module Cucumber
|
|
32
34
|
|
33
35
|
def format_step(keyword, step_match, status, source_indent)
|
34
36
|
comment = if source_indent
|
35
|
-
c = ('# ' + step_match.location.to_s)
|
37
|
+
c = indent(('# ' + step_match.location.to_s), source_indent)
|
36
38
|
format_string(c, :comment)
|
37
39
|
else
|
38
40
|
''
|
@@ -46,7 +48,7 @@ module Cucumber
|
|
46
48
|
def format_string(o, status)
|
47
49
|
fmt = format_for(status)
|
48
50
|
o.to_s.split("\n").map do |line|
|
49
|
-
if Proc
|
51
|
+
if Proc == fmt.class
|
50
52
|
fmt.call(line)
|
51
53
|
else
|
52
54
|
fmt % line
|
@@ -54,10 +56,6 @@ module Cucumber
|
|
54
56
|
end.join("\n")
|
55
57
|
end
|
56
58
|
|
57
|
-
def print_steps(status)
|
58
|
-
print_elements(runtime.steps(status), status, 'steps')
|
59
|
-
end
|
60
|
-
|
61
59
|
def print_elements(elements, status, kind)
|
62
60
|
return if elements.empty?
|
63
61
|
|
@@ -101,36 +99,40 @@ module Cucumber
|
|
101
99
|
@io.puts(format_string(string, status))
|
102
100
|
end
|
103
101
|
|
104
|
-
def exception_message_string(e,
|
102
|
+
def exception_message_string(e, indent_amount)
|
105
103
|
message = "#{e.message} (#{e.class})".dup.force_encoding('UTF-8')
|
106
104
|
message = linebreaks(message, ENV['CUCUMBER_TRUNCATE_OUTPUT'].to_i)
|
107
105
|
|
108
|
-
"#{message}\n#{e.backtrace.join("\n")}"
|
106
|
+
indent("#{message}\n#{e.backtrace.join("\n")}", indent_amount)
|
109
107
|
end
|
110
108
|
|
111
109
|
# http://blade.nagaokaut.ac.jp/cgi-bin/scat.rb/ruby/ruby-talk/10655
|
112
|
-
def linebreaks(
|
113
|
-
return
|
114
|
-
|
110
|
+
def linebreaks(msg, max)
|
111
|
+
return msg unless max && max > 0
|
112
|
+
msg.gsub(/.{1,#{max}}(?:\s|\Z)/) { ($& + 5.chr).gsub(/\n\005/, "\n").gsub(/\005/, "\n") }.rstrip
|
115
113
|
end
|
116
114
|
|
117
|
-
def collect_snippet_data(test_step,
|
115
|
+
def collect_snippet_data(test_step, ast_lookup)
|
118
116
|
# collect snippet data for undefined steps
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
117
|
+
keyword = ast_lookup.snippet_step_keyword(test_step)
|
118
|
+
@snippets_input << Console::SnippetData.new(keyword, test_step)
|
119
|
+
end
|
120
|
+
|
121
|
+
def collect_undefined_parameter_type_names(undefined_parameter_type)
|
122
|
+
@undefined_parameter_types << undefined_parameter_type.type_name
|
124
123
|
end
|
125
124
|
|
126
125
|
def print_snippets(options)
|
127
126
|
return unless options[:snippets]
|
128
|
-
return if runtime.steps(:undefined).empty?
|
129
127
|
|
130
128
|
snippet_text_proc = lambda do |step_keyword, step_name, multiline_arg|
|
131
|
-
|
129
|
+
snippet_text(step_keyword, step_name, multiline_arg)
|
130
|
+
end
|
131
|
+
do_print_snippets(snippet_text_proc) unless @snippets_input.empty?
|
132
|
+
|
133
|
+
@undefined_parameter_types.map do |type_name|
|
134
|
+
do_print_undefined_parameter_type_snippet(type_name)
|
132
135
|
end
|
133
|
-
do_print_snippets(snippet_text_proc)
|
134
136
|
end
|
135
137
|
|
136
138
|
def do_print_snippets(snippet_text_proc)
|
@@ -146,10 +148,14 @@ module Cucumber
|
|
146
148
|
@io.flush
|
147
149
|
end
|
148
150
|
|
149
|
-
def print_passing_wip(
|
150
|
-
return unless
|
151
|
-
|
152
|
-
|
151
|
+
def print_passing_wip(config, passed_test_cases, ast_lookup)
|
152
|
+
return unless config.wip?
|
153
|
+
messages = passed_test_cases.map do |test_case|
|
154
|
+
scenario_source = ast_lookup.scenario_source(test_case)
|
155
|
+
keyword = scenario_source.type == :Scenario ? scenario_source.scenario.keyword : scenario_source.scenario_outline.keyword
|
156
|
+
linebreaks("#{test_case.location.on_line(test_case.location.lines.max)}:in `#{keyword}: #{test_case.name}'", ENV['CUCUMBER_TRUNCATE_OUTPUT'].to_i)
|
157
|
+
end
|
158
|
+
do_print_passing_wip(messages)
|
153
159
|
end
|
154
160
|
|
155
161
|
def do_print_passing_wip(passed_messages)
|
@@ -161,59 +167,55 @@ module Cucumber
|
|
161
167
|
end
|
162
168
|
end
|
163
169
|
|
164
|
-
def
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
# activate this feature
|
170
|
-
def puts(*messages)
|
171
|
-
if @delayed_messages
|
172
|
-
@delayed_messages += messages
|
173
|
-
else
|
174
|
-
if @io
|
175
|
-
@io.puts
|
176
|
-
messages.each do |message|
|
177
|
-
@io.puts(format_string(message, :tag))
|
178
|
-
end
|
179
|
-
@io.flush
|
180
|
-
end
|
181
|
-
end
|
182
|
-
end
|
183
|
-
|
184
|
-
def print_messages
|
185
|
-
@delayed_messages.each { |message| print_message(message) }
|
186
|
-
empty_messages
|
187
|
-
end
|
188
|
-
|
189
|
-
def print_table_row_messages
|
190
|
-
return if @delayed_messages.empty?
|
191
|
-
@io.print(format_string(@delayed_messages.join(', '), :tag).indent(2))
|
192
|
-
@io.flush
|
193
|
-
empty_messages
|
194
|
-
end
|
195
|
-
|
196
|
-
def print_message(message)
|
197
|
-
@io.puts(format_string(message, :tag).indent(@indent))
|
170
|
+
def attach(src, media_type)
|
171
|
+
return unless media_type == 'text/x.cucumber.log+plain'
|
172
|
+
return unless @io
|
173
|
+
@io.puts
|
174
|
+
@io.puts(format_string(src, :tag))
|
198
175
|
@io.flush
|
199
176
|
end
|
200
177
|
|
201
|
-
def empty_messages
|
202
|
-
@delayed_messages = []
|
203
|
-
end
|
204
|
-
|
205
178
|
def print_profile_information
|
206
179
|
return if @options[:skip_profile_information] || @options[:profiles].nil? || @options[:profiles].empty?
|
207
180
|
do_print_profile_information(@options[:profiles])
|
208
181
|
end
|
209
182
|
|
210
183
|
def do_print_profile_information(profiles)
|
211
|
-
profiles_sentence = profiles.size == 1
|
212
|
-
|
184
|
+
profiles_sentence = if profiles.size == 1
|
185
|
+
profiles.first
|
186
|
+
else
|
187
|
+
"#{profiles[0...-1].join(', ')} and #{profiles.last}"
|
188
|
+
end
|
213
189
|
|
214
190
|
@io.puts "Using the #{profiles_sentence} profile#{'s' if profiles.size > 1}..."
|
215
191
|
end
|
216
192
|
|
193
|
+
def do_print_undefined_parameter_type_snippet(type_name)
|
194
|
+
camelized = type_name.split(/_|-/).collect(&:capitalize).join
|
195
|
+
|
196
|
+
@io.puts [
|
197
|
+
"The parameter #{type_name} is not defined. You can define a new one with:",
|
198
|
+
'',
|
199
|
+
'ParameterType(',
|
200
|
+
" name: '#{type_name}',",
|
201
|
+
' regexp: /some regexp here/,',
|
202
|
+
" type: #{camelized},",
|
203
|
+
' # The transformer takes as many arguments as there are capture groups in the regexp,',
|
204
|
+
' # or just one if there are none.',
|
205
|
+
" transformer: ->(s) { #{camelized}.new(s) }",
|
206
|
+
')',
|
207
|
+
''
|
208
|
+
].join("\n")
|
209
|
+
end
|
210
|
+
|
211
|
+
def indent(string, padding)
|
212
|
+
if padding >= 0
|
213
|
+
string.gsub(/^/, ' ' * padding)
|
214
|
+
else
|
215
|
+
string.gsub(/^ {0,#{-padding}}/, '')
|
216
|
+
end
|
217
|
+
end
|
218
|
+
|
217
219
|
private
|
218
220
|
|
219
221
|
FORMATS = Hash.new { |hash, format| hash[format] = method(format).to_proc }
|
@@ -225,10 +227,6 @@ module Cucumber
|
|
225
227
|
fmt
|
226
228
|
end
|
227
229
|
|
228
|
-
def hook?(test_step)
|
229
|
-
not test_step.source.last.respond_to?(:actual_keyword)
|
230
|
-
end
|
231
|
-
|
232
230
|
def element_messages(elements, status)
|
233
231
|
elements.map do |element|
|
234
232
|
if status == :failed
|
@@ -249,9 +247,11 @@ module Cucumber
|
|
249
247
|
class SnippetData
|
250
248
|
attr_reader :actual_keyword, :step
|
251
249
|
def initialize(actual_keyword, step)
|
252
|
-
@actual_keyword
|
250
|
+
@actual_keyword = actual_keyword
|
251
|
+
@step = step
|
253
252
|
end
|
254
253
|
end
|
255
254
|
end
|
256
255
|
end
|
257
256
|
end
|
257
|
+
# rubocop:enable Metrics/ModuleLength
|
@@ -29,15 +29,10 @@ module Cucumber
|
|
29
29
|
end
|
30
30
|
|
31
31
|
def status_counts(summary)
|
32
|
-
counts = Core::Test::Result::TYPES.map
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
count > 0
|
37
|
-
end.map do |status, count|
|
38
|
-
format_string("#{count} #{status}", status)
|
39
|
-
end
|
40
|
-
"(#{counts.join(", ")})" if counts.any?
|
32
|
+
counts = Core::Test::Result::TYPES.map { |status| [status, summary.total(status)] }
|
33
|
+
counts = counts.select { |_status, count| count > 0 }
|
34
|
+
counts = counts.map { |status, count| format_string("#{count} #{status}", status) }
|
35
|
+
"(#{counts.join(', ')})" if counts.any?
|
41
36
|
end
|
42
37
|
end
|
43
38
|
end
|
@@ -5,7 +5,7 @@ module Cucumber
|
|
5
5
|
class ConsoleIssues
|
6
6
|
include Console
|
7
7
|
|
8
|
-
def initialize(config)
|
8
|
+
def initialize(config, ast_lookup = AstLookup.new(config))
|
9
9
|
@previous_test_case = nil
|
10
10
|
@issues = Hash.new { |h, k| h[k] = [] }
|
11
11
|
@config = config
|
@@ -18,6 +18,7 @@ module Cucumber
|
|
18
18
|
@issues[:failed].delete(event.test_case)
|
19
19
|
end
|
20
20
|
end
|
21
|
+
@ast_lookup = ast_lookup
|
21
22
|
end
|
22
23
|
|
23
24
|
def to_s
|
@@ -35,8 +36,10 @@ module Cucumber
|
|
35
36
|
def scenario_listing(type, test_cases)
|
36
37
|
return [] if test_cases.empty?
|
37
38
|
[format_string("#{type_heading(type)} Scenarios:", type)] + test_cases.map do |test_case|
|
38
|
-
|
39
|
-
|
39
|
+
scenario_source = @ast_lookup.scenario_source(test_case)
|
40
|
+
keyword = scenario_source.type == :Scenario ? scenario_source.scenario.keyword : scenario_source.scenario_outline.keyword
|
41
|
+
source = @config.source? ? format_string(" # #{keyword}: #{test_case.name}", :comment) : ''
|
42
|
+
format_string("cucumber #{profiles_string}#{test_case.location.file}:#{test_case.location.lines.max}", type) + source
|
40
43
|
end
|
41
44
|
end
|
42
45
|
|
@@ -22,8 +22,10 @@ module Cucumber
|
|
22
22
|
def exception(*) end
|
23
23
|
|
24
24
|
def duration(duration, *)
|
25
|
-
duration.tap { |
|
25
|
+
duration.tap { |dur| @result_duration = dur.nanoseconds / 10**9.0 }
|
26
26
|
end
|
27
|
+
|
28
|
+
def attach(*) end
|
27
29
|
end
|
28
30
|
end
|
29
31
|
end
|