cucumber 2.0.0.rc.5 → 4.0.0.rc.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (527) hide show
  1. checksums.yaml +5 -5
  2. data/{History.md → CHANGELOG.md} +700 -264
  3. data/CONTRIBUTING.md +26 -23
  4. data/LICENSE +1 -1
  5. data/README.md +25 -8
  6. data/bin/cucumber +3 -3
  7. data/lib/autotest/cucumber.rb +2 -0
  8. data/lib/autotest/cucumber_mixin.rb +45 -48
  9. data/lib/autotest/cucumber_rails.rb +2 -0
  10. data/lib/autotest/cucumber_rails_rspec.rb +2 -0
  11. data/lib/autotest/cucumber_rails_rspec2.rb +2 -0
  12. data/lib/autotest/cucumber_rspec.rb +2 -0
  13. data/lib/autotest/cucumber_rspec2.rb +2 -0
  14. data/lib/autotest/discover.rb +4 -2
  15. data/lib/cucumber.rb +3 -7
  16. data/lib/cucumber/cli/configuration.rb +23 -106
  17. data/lib/cucumber/cli/main.rb +45 -25
  18. data/lib/cucumber/cli/options.rb +410 -229
  19. data/lib/cucumber/cli/profile_loader.rb +64 -38
  20. data/lib/cucumber/cli/rerun_file.rb +31 -0
  21. data/lib/cucumber/configuration.rb +240 -8
  22. data/lib/cucumber/constantize.rb +9 -10
  23. data/lib/cucumber/core_ext/string.rb +3 -1
  24. data/lib/cucumber/deprecate.rb +30 -0
  25. data/lib/cucumber/encoding.rb +3 -1
  26. data/lib/cucumber/errors.rb +9 -8
  27. data/lib/cucumber/events.rb +40 -0
  28. data/lib/cucumber/events/gherkin_source_parsed.rb +11 -0
  29. data/lib/cucumber/events/gherkin_source_read.rb +14 -0
  30. data/lib/cucumber/events/step_activated.rb +26 -0
  31. data/lib/cucumber/events/step_definition_registered.rb +20 -0
  32. data/lib/cucumber/events/test_case_finished.rb +14 -0
  33. data/lib/cucumber/events/test_case_started.rb +11 -0
  34. data/lib/cucumber/events/test_run_finished.rb +11 -0
  35. data/lib/cucumber/events/test_run_started.rb +14 -0
  36. data/lib/cucumber/events/test_step_finished.rb +14 -0
  37. data/lib/cucumber/events/test_step_started.rb +11 -0
  38. data/lib/cucumber/file_specs.rb +8 -6
  39. data/lib/cucumber/filters.rb +4 -0
  40. data/lib/cucumber/filters/activate_steps.rb +48 -7
  41. data/lib/cucumber/filters/apply_after_hooks.rb +2 -0
  42. data/lib/cucumber/filters/apply_after_step_hooks.rb +2 -0
  43. data/lib/cucumber/filters/apply_around_hooks.rb +2 -0
  44. data/lib/cucumber/filters/apply_before_hooks.rb +2 -0
  45. data/lib/cucumber/filters/broadcast_test_run_started_event.rb +28 -0
  46. data/lib/cucumber/filters/gated_receiver.rb +2 -2
  47. data/lib/cucumber/filters/prepare_world.rb +7 -13
  48. data/lib/cucumber/filters/quit.rb +4 -7
  49. data/lib/cucumber/filters/randomizer.rb +10 -5
  50. data/lib/cucumber/filters/retry.rb +32 -0
  51. data/lib/cucumber/filters/tag_limits.rb +6 -5
  52. data/lib/cucumber/filters/tag_limits/test_case_index.rb +2 -2
  53. data/lib/cucumber/filters/tag_limits/verifier.rb +5 -7
  54. data/lib/cucumber/formatter/ansicolor.rb +39 -45
  55. data/lib/cucumber/formatter/ast_lookup.rb +165 -0
  56. data/lib/cucumber/formatter/backtrace_filter.rb +54 -0
  57. data/lib/cucumber/formatter/console.rb +93 -115
  58. data/lib/cucumber/formatter/console_counts.rb +39 -0
  59. data/lib/cucumber/formatter/console_issues.rb +61 -0
  60. data/lib/cucumber/formatter/duration.rb +3 -1
  61. data/lib/cucumber/formatter/duration_extractor.rb +31 -0
  62. data/lib/cucumber/formatter/fail_fast.rb +23 -0
  63. data/lib/cucumber/formatter/fanout.rb +4 -3
  64. data/lib/cucumber/formatter/ignore_missing_messages.rb +3 -4
  65. data/lib/cucumber/formatter/interceptor.rb +13 -14
  66. data/lib/cucumber/formatter/io.rb +10 -5
  67. data/lib/cucumber/formatter/json.rb +310 -8
  68. data/lib/cucumber/formatter/junit.rb +143 -173
  69. data/lib/cucumber/formatter/pretty.rb +359 -154
  70. data/lib/cucumber/formatter/progress.rb +78 -72
  71. data/lib/cucumber/formatter/rerun.rb +34 -17
  72. data/lib/cucumber/formatter/stepdefs.rb +3 -2
  73. data/lib/cucumber/formatter/steps.rb +7 -7
  74. data/lib/cucumber/formatter/summary.rb +50 -20
  75. data/lib/cucumber/formatter/unicode.rb +17 -18
  76. data/lib/cucumber/formatter/usage.rb +61 -54
  77. data/lib/cucumber/gherkin/data_table_parser.rb +44 -0
  78. data/lib/cucumber/gherkin/formatter/ansi_escapes.rb +97 -0
  79. data/lib/cucumber/gherkin/formatter/escaping.rb +19 -0
  80. data/lib/cucumber/gherkin/i18n.rb +17 -0
  81. data/lib/cucumber/gherkin/steps_parser.rb +42 -0
  82. data/lib/cucumber/{rb_support/rb_dsl.rb → glue/dsl.rb} +47 -26
  83. data/lib/cucumber/glue/hook.rb +51 -0
  84. data/lib/cucumber/glue/invoke_in_world.rb +65 -0
  85. data/lib/cucumber/glue/proto_world.rb +219 -0
  86. data/lib/cucumber/glue/registry_and_more.rb +192 -0
  87. data/lib/cucumber/glue/snippet.rb +205 -0
  88. data/lib/cucumber/glue/step_definition.rb +142 -0
  89. data/lib/cucumber/glue/world_factory.rb +21 -0
  90. data/lib/cucumber/hooks.rb +34 -21
  91. data/lib/cucumber/load_path.rb +2 -0
  92. data/lib/cucumber/multiline_argument.rb +10 -24
  93. data/lib/cucumber/multiline_argument/data_table.rb +157 -253
  94. data/lib/cucumber/multiline_argument/data_table/diff_matrices.rb +154 -0
  95. data/lib/cucumber/multiline_argument/doc_string.rb +3 -2
  96. data/lib/cucumber/platform.rb +7 -10
  97. data/lib/cucumber/project_initializer.rb +4 -3
  98. data/lib/cucumber/rake/task.rb +37 -29
  99. data/lib/cucumber/rspec/disable_option_parser.rb +11 -8
  100. data/lib/cucumber/rspec/doubles.rb +3 -1
  101. data/lib/cucumber/running_test_case.rb +15 -70
  102. data/lib/cucumber/runtime.rb +83 -81
  103. data/lib/cucumber/runtime/after_hooks.rb +10 -7
  104. data/lib/cucumber/runtime/before_hooks.rb +12 -6
  105. data/lib/cucumber/runtime/for_programming_languages.rb +14 -44
  106. data/lib/cucumber/runtime/step_hooks.rb +8 -4
  107. data/lib/cucumber/runtime/support_code.rb +46 -160
  108. data/lib/cucumber/runtime/user_interface.rb +15 -24
  109. data/lib/cucumber/step_definition_light.rb +10 -6
  110. data/lib/cucumber/step_definitions.rb +6 -4
  111. data/lib/cucumber/step_match.rb +58 -38
  112. data/lib/cucumber/step_match_search.rb +67 -0
  113. data/lib/cucumber/term/ansicolor.rb +40 -39
  114. data/lib/cucumber/unit.rb +2 -0
  115. data/lib/cucumber/version +1 -0
  116. data/{spec → lib}/simplecov_setup.rb +3 -1
  117. metadata +257 -621
  118. data/.coveralls.yml +0 -1
  119. data/.rspec +0 -1
  120. data/.ruby-gemset +0 -1
  121. data/.travis.yml +0 -18
  122. data/.yardopts +0 -1
  123. data/Gemfile +0 -11
  124. data/Rakefile +0 -22
  125. data/cucumber.gemspec +0 -50
  126. data/cucumber.yml +0 -20
  127. data/examples/i18n/README.textile +0 -3
  128. data/examples/i18n/Rakefile +0 -33
  129. data/examples/i18n/ar/Rakefile +0 -6
  130. data/examples/i18n/ar/features/addition.feature +0 -17
  131. data/examples/i18n/ar/features/step_definitions/calculator_steps.rb +0 -24
  132. data/examples/i18n/ar/lib/calculator.rb +0 -11
  133. data/examples/i18n/bg/Rakefile +0 -6
  134. data/examples/i18n/bg/features/addition.feature +0 -12
  135. data/examples/i18n/bg/features/consecutive_calculations.feature +0 -19
  136. data/examples/i18n/bg/features/division.feature +0 -17
  137. data/examples/i18n/bg/features/step_definitions/calculator_steps.rb +0 -24
  138. data/examples/i18n/bg/features/support/env.rb +0 -5
  139. data/examples/i18n/bg/features/support/world.rb +0 -8
  140. data/examples/i18n/bg/lib/calculator.rb +0 -24
  141. data/examples/i18n/ca/Rakefile +0 -6
  142. data/examples/i18n/ca/features/step_definitions/calculator_steps.rb +0 -21
  143. data/examples/i18n/ca/features/suma.feature +0 -17
  144. data/examples/i18n/ca/lib/calculadora.rb +0 -16
  145. data/examples/i18n/cs/Rakefile +0 -6
  146. data/examples/i18n/cs/features/addition.feature +0 -17
  147. data/examples/i18n/cs/features/division.feature +0 -11
  148. data/examples/i18n/cs/features/step_definitions/calculator_steps.rb +0 -24
  149. data/examples/i18n/cs/lib/calculator.rb +0 -14
  150. data/examples/i18n/da/Rakefile +0 -6
  151. data/examples/i18n/da/features/sammenlaegning.feature +0 -18
  152. data/examples/i18n/da/features/step_definitions/lommeregner_steps.rb +0 -24
  153. data/examples/i18n/da/lib/lommeregner.rb +0 -11
  154. data/examples/i18n/de/Rakefile +0 -6
  155. data/examples/i18n/de/features/addition.feature +0 -17
  156. data/examples/i18n/de/features/division.feature +0 -10
  157. data/examples/i18n/de/features/step_definitions/calculator_steps.rb +0 -24
  158. data/examples/i18n/de/lib/calculator.rb +0 -14
  159. data/examples/i18n/el/Rakefile +0 -6
  160. data/examples/i18n/el/features/addition.feature +0 -17
  161. data/examples/i18n/el/features/division.feature +0 -10
  162. data/examples/i18n/el/features/step_definitions/calculator_steps.rb +0 -24
  163. data/examples/i18n/el/lib/calculator.rb +0 -14
  164. data/examples/i18n/en-lol/Rakefile +0 -4
  165. data/examples/i18n/en-lol/features/step_definitions/cucumbrz_steps.rb +0 -16
  166. data/examples/i18n/en-lol/features/stuffing.feature +0 -8
  167. data/examples/i18n/en-lol/features/support/env.rb +0 -7
  168. data/examples/i18n/en-lol/lib/basket.rb +0 -12
  169. data/examples/i18n/en-lol/lib/belly.rb +0 -11
  170. data/examples/i18n/en/Rakefile +0 -6
  171. data/examples/i18n/en/features/addition.feature +0 -17
  172. data/examples/i18n/en/features/division.feature +0 -10
  173. data/examples/i18n/en/features/step_definitions/calculator_steps.rb +0 -24
  174. data/examples/i18n/en/lib/calculator.rb +0 -14
  175. data/examples/i18n/eo/Rakefile +0 -6
  176. data/examples/i18n/eo/features/adicio.feature +0 -17
  177. data/examples/i18n/eo/features/divido.feature +0 -10
  178. data/examples/i18n/eo/features/step_definitions/calculator_steps.rb +0 -24
  179. data/examples/i18n/eo/lib/calculator.rb +0 -14
  180. data/examples/i18n/es/Rakefile +0 -6
  181. data/examples/i18n/es/features/adicion.feature +0 -17
  182. data/examples/i18n/es/features/step_definitions/calculador_steps.rb +0 -21
  183. data/examples/i18n/es/lib/calculador.rb +0 -14
  184. data/examples/i18n/et/Rakefile +0 -6
  185. data/examples/i18n/et/features/jagamine.feature +0 -10
  186. data/examples/i18n/et/features/liitmine.feature +0 -17
  187. data/examples/i18n/et/features/step_definitions/kalkulaator_steps.rb +0 -24
  188. data/examples/i18n/et/lib/kalkulaator.rb +0 -14
  189. data/examples/i18n/fi/Rakefile +0 -6
  190. data/examples/i18n/fi/features/jakolasku.feature +0 -10
  191. data/examples/i18n/fi/features/step_definitions/laskin_steps.rb +0 -24
  192. data/examples/i18n/fi/features/yhteenlasku.feature +0 -17
  193. data/examples/i18n/fi/lib/laskin.rb +0 -14
  194. data/examples/i18n/fr/Rakefile +0 -8
  195. data/examples/i18n/fr/features/addition.feature +0 -18
  196. data/examples/i18n/fr/features/addition2.feature +0 -17
  197. data/examples/i18n/fr/features/step_definitions/calculatrice_steps.rb +0 -32
  198. data/examples/i18n/fr/features/support/env.rb +0 -5
  199. data/examples/i18n/fr/lib/calculatrice.rb +0 -10
  200. data/examples/i18n/he/Rakefile +0 -6
  201. data/examples/i18n/he/features/addition.feature +0 -17
  202. data/examples/i18n/he/features/division.feature +0 -10
  203. data/examples/i18n/he/features/step_definitions/calculator_steps.rb +0 -24
  204. data/examples/i18n/he/lib/calculator.rb +0 -15
  205. data/examples/i18n/hi/Rakefile +0 -6
  206. data/examples/i18n/hi/features/addition.feature +0 -16
  207. data/examples/i18n/hi/features/division.feature +0 -10
  208. data/examples/i18n/hi/features/step_definitions/calculator_steps.rb +0 -24
  209. data/examples/i18n/hi/lib/calculator.rb +0 -15
  210. data/examples/i18n/ht/Rakefile +0 -6
  211. data/examples/i18n/ht/features/adisyon.feature +0 -17
  212. data/examples/i18n/ht/features/divizyon.feature +0 -10
  213. data/examples/i18n/ht/features/step_definitions/kalkilatris_steps.rb +0 -25
  214. data/examples/i18n/ht/lib/kalkilatris.rb +0 -14
  215. data/examples/i18n/hu/Rakefile +0 -6
  216. data/examples/i18n/hu/features/osszeadas.feature +0 -17
  217. data/examples/i18n/hu/features/osztas.feature +0 -10
  218. data/examples/i18n/hu/features/step_definitions/calculator_steps.rb +0 -25
  219. data/examples/i18n/hu/lib/calculator.rb +0 -14
  220. data/examples/i18n/id/Rakefile +0 -6
  221. data/examples/i18n/id/features/addition.feature +0 -17
  222. data/examples/i18n/id/features/division.feature +0 -10
  223. data/examples/i18n/id/features/step_definitions/calculator_steps.rb +0 -24
  224. data/examples/i18n/id/lib/calculator.rb +0 -14
  225. data/examples/i18n/it/Rakefile +0 -6
  226. data/examples/i18n/it/features/somma.feature +0 -11
  227. data/examples/i18n/it/features/step_definitions/calcolatrice_steps.rb +0 -24
  228. data/examples/i18n/it/lib/calcolatrice.rb +0 -11
  229. data/examples/i18n/ja/Rakefile +0 -6
  230. data/examples/i18n/ja/features/addition.feature +0 -17
  231. data/examples/i18n/ja/features/division.feature +0 -10
  232. data/examples/i18n/ja/features/step_definitions/calculator_steps.rb +0 -19
  233. data/examples/i18n/ja/features/support/env.rb +0 -5
  234. data/examples/i18n/ja/lib/calculator.rb +0 -14
  235. data/examples/i18n/ko/Rakefile +0 -6
  236. data/examples/i18n/ko/features/addition.feature +0 -17
  237. data/examples/i18n/ko/features/division.feature +0 -11
  238. data/examples/i18n/ko/features/step_definitions/calculator_steps.rb +0 -24
  239. data/examples/i18n/ko/lib/calculator.rb +0 -14
  240. data/examples/i18n/lt/Rakefile +0 -6
  241. data/examples/i18n/lt/features/addition.feature +0 -17
  242. data/examples/i18n/lt/features/division.feature +0 -10
  243. data/examples/i18n/lt/features/step_definitions/calculator_steps.rb +0 -24
  244. data/examples/i18n/lt/lib/calculator.rb +0 -14
  245. data/examples/i18n/lv/Rakefile +0 -6
  246. data/examples/i18n/lv/features/addition.feature +0 -17
  247. data/examples/i18n/lv/features/division.feature +0 -10
  248. data/examples/i18n/lv/features/step_definitions/calculator_steps.rb +0 -24
  249. data/examples/i18n/lv/lib/calculator.rb +0 -14
  250. data/examples/i18n/no/Rakefile +0 -6
  251. data/examples/i18n/no/features/step_definitions/kalkulator_steps.rb +0 -17
  252. data/examples/i18n/no/features/summering.feature +0 -19
  253. data/examples/i18n/no/features/support/env.rb +0 -6
  254. data/examples/i18n/no/lib/kalkulator.rb +0 -11
  255. data/examples/i18n/pl/Rakefile +0 -6
  256. data/examples/i18n/pl/features/addition.feature +0 -17
  257. data/examples/i18n/pl/features/division.feature +0 -10
  258. data/examples/i18n/pl/features/step_definitions/calculator_steps.rb +0 -24
  259. data/examples/i18n/pl/features/support/env.rb +0 -5
  260. data/examples/i18n/pl/lib/calculator.rb +0 -14
  261. data/examples/i18n/pt/Rakefile +0 -6
  262. data/examples/i18n/pt/features/adicao.feature +0 -11
  263. data/examples/i18n/pt/features/step_definitions/calculadora_steps.rb +0 -20
  264. data/examples/i18n/pt/features/support/env.rb +0 -5
  265. data/examples/i18n/pt/lib/calculadora.rb +0 -10
  266. data/examples/i18n/ro/Rakefile +0 -6
  267. data/examples/i18n/ro/features/adunare.feature +0 -12
  268. data/examples/i18n/ro/features/step_definitions/calculator_steps.rb +0 -21
  269. data/examples/i18n/ro/lib/calculator.rb +0 -11
  270. data/examples/i18n/ru/Rakefile +0 -6
  271. data/examples/i18n/ru/features/addition.feature +0 -11
  272. data/examples/i18n/ru/features/consecutive_calculations.feature +0 -17
  273. data/examples/i18n/ru/features/division.feature +0 -16
  274. data/examples/i18n/ru/features/step_definitions/calculator_steps.rb +0 -19
  275. data/examples/i18n/ru/features/support/env.rb +0 -5
  276. data/examples/i18n/ru/features/support/world.rb +0 -8
  277. data/examples/i18n/ru/lib/calculator.rb +0 -24
  278. data/examples/i18n/sk/Rakefile +0 -6
  279. data/examples/i18n/sk/features/addition.feature +0 -17
  280. data/examples/i18n/sk/features/division.feature +0 -10
  281. data/examples/i18n/sk/features/step_definitions/calculator_steps.rb +0 -24
  282. data/examples/i18n/sk/lib/calculator.rb +0 -14
  283. data/examples/i18n/sr-Cyrl/Rakefile +0 -6
  284. data/examples/i18n/sr-Cyrl/features/sabiranje.feature +0 -18
  285. data/examples/i18n/sr-Cyrl/features/step_definitions/calculator_steps.rb +0 -20
  286. data/examples/i18n/sr-Cyrl/features/support/env.rb +0 -5
  287. data/examples/i18n/sr-Cyrl/lib/calculator.rb +0 -12
  288. data/examples/i18n/sr-Latn/Rakefile +0 -6
  289. data/examples/i18n/sr-Latn/features/sabiranje.feature +0 -18
  290. data/examples/i18n/sr-Latn/features/step_definitions/calculator_steps.rb +0 -24
  291. data/examples/i18n/sr-Latn/lib/calculator.rb +0 -12
  292. data/examples/i18n/sv/Rakefile +0 -6
  293. data/examples/i18n/sv/features/step_definitions/kalkulator_steps.rb +0 -24
  294. data/examples/i18n/sv/features/summering.feature +0 -18
  295. data/examples/i18n/sv/lib/kalkulator.rb +0 -11
  296. data/examples/i18n/tr/Rakefile +0 -6
  297. data/examples/i18n/tr/features/bolme.feature +0 -10
  298. data/examples/i18n/tr/features/step_definitions/hesap_makinesi_adimlari.rb +0 -24
  299. data/examples/i18n/tr/features/toplama.feature +0 -18
  300. data/examples/i18n/tr/lib/hesap_makinesi.rb +0 -15
  301. data/examples/i18n/uk/Rakefile +0 -6
  302. data/examples/i18n/uk/features/addition.feature +0 -11
  303. data/examples/i18n/uk/features/consecutive_calculations.feature +0 -17
  304. data/examples/i18n/uk/features/division.feature +0 -16
  305. data/examples/i18n/uk/features/step_definitions/calculator_steps.rb +0 -19
  306. data/examples/i18n/uk/features/support/env.rb +0 -5
  307. data/examples/i18n/uk/features/support/world.rb +0 -8
  308. data/examples/i18n/uk/lib/calculator.rb +0 -24
  309. data/examples/i18n/uz/Rakefile +0 -6
  310. data/examples/i18n/uz/features/addition.feature +0 -10
  311. data/examples/i18n/uz/features/consecutive_calculations.feature +0 -17
  312. data/examples/i18n/uz/features/division.feature +0 -17
  313. data/examples/i18n/uz/features/step_definitions/calculator_steps.rb +0 -19
  314. data/examples/i18n/uz/features/support/env.rb +0 -5
  315. data/examples/i18n/uz/features/support/world.rb +0 -8
  316. data/examples/i18n/uz/lib/calculator.rb +0 -24
  317. data/examples/i18n/zh-CN/Rakefile +0 -4
  318. data/examples/i18n/zh-CN/features/addition.feature +0 -18
  319. data/examples/i18n/zh-CN/features/step_definitions/calculator_steps.rb +0 -26
  320. data/examples/i18n/zh-CN/lib/calculator.rb +0 -10
  321. data/examples/i18n/zh-TW/Rakefile +0 -4
  322. data/examples/i18n/zh-TW/features/addition.feature +0 -17
  323. data/examples/i18n/zh-TW/features/division.feature +0 -11
  324. data/examples/i18n/zh-TW/features/step_definitions/calculator_steps.rb +0 -24
  325. data/examples/i18n/zh-TW/lib/calculator.rb +0 -14
  326. data/examples/rspec_doubles/Rakefile +0 -4
  327. data/examples/rspec_doubles/features/mocking.feature +0 -9
  328. data/examples/rspec_doubles/features/step_definitions/calvin_steps.rb +0 -19
  329. data/examples/rspec_doubles/features/support/env.rb +0 -12
  330. data/examples/sinatra/README.textile +0 -13
  331. data/examples/sinatra/Rakefile +0 -6
  332. data/examples/sinatra/app.rb +0 -14
  333. data/examples/sinatra/features/add.feature +0 -11
  334. data/examples/sinatra/features/step_definitions/add_steps.rb +0 -15
  335. data/examples/sinatra/features/support/env.rb +0 -10
  336. data/examples/sinatra/views/add.erb +0 -7
  337. data/examples/sinatra/views/layout.erb +0 -8
  338. data/examples/tcl/README.textile +0 -11
  339. data/examples/tcl/Rakefile +0 -6
  340. data/examples/tcl/features/fibonnacci.feature +0 -17
  341. data/examples/tcl/features/step_definitions/fib_steps.rb +0 -7
  342. data/examples/tcl/features/support/env.rb +0 -6
  343. data/examples/tcl/src/fib.tcl +0 -3
  344. data/examples/test_unit/Gemfile +0 -4
  345. data/examples/test_unit/Rakefile +0 -6
  346. data/examples/test_unit/features/step_definitions/test_unit_steps.rb +0 -20
  347. data/examples/test_unit/features/test_unit.feature +0 -9
  348. data/examples/watir/README.textile +0 -16
  349. data/examples/watir/Rakefile +0 -12
  350. data/examples/watir/cucumber.yml +0 -1
  351. data/examples/watir/features/search.feature +0 -12
  352. data/examples/watir/features/step_definitions/search_steps.rb +0 -26
  353. data/examples/watir/features/support/env.rb +0 -35
  354. data/examples/watir/features/support/screenshots.rb +0 -47
  355. data/features/docs/api/list_step_defs_as_json.feature +0 -50
  356. data/features/docs/api/run_cli_main_with_existing_runtime.feature +0 -29
  357. data/features/docs/cli/backtraces.feature +0 -36
  358. data/features/docs/cli/dry_run.feature +0 -68
  359. data/features/docs/cli/exclude_files.feature +0 -19
  360. data/features/docs/cli/execute_with_tag_filter.feature +0 -117
  361. data/features/docs/cli/randomize.feature +0 -46
  362. data/features/docs/cli/require.feature +0 -27
  363. data/features/docs/cli/run_scenarios_matching_name.feature +0 -104
  364. data/features/docs/cli/run_specific_scenarios.feature +0 -130
  365. data/features/docs/cli/showing_differences.feature +0 -43
  366. data/features/docs/cli/specifying_multiple_formatters.feature +0 -44
  367. data/features/docs/cli/strict_mode.feature +0 -64
  368. data/features/docs/defining_steps/nested_steps.feature +0 -178
  369. data/features/docs/defining_steps/nested_steps_i18n.feature +0 -34
  370. data/features/docs/defining_steps/nested_steps_with_second_arg.feature +0 -73
  371. data/features/docs/defining_steps/one_line_step_definitions.feature +0 -65
  372. data/features/docs/defining_steps/printing_messages.feature +0 -148
  373. data/features/docs/defining_steps/skip_scenario.feature +0 -62
  374. data/features/docs/defining_steps/snippets.feature +0 -56
  375. data/features/docs/defining_steps/table_diffing.feature +0 -50
  376. data/features/docs/defining_steps/transforms.feature +0 -63
  377. data/features/docs/exception_in_after_hook.feature +0 -126
  378. data/features/docs/exception_in_after_step_hook.feature +0 -102
  379. data/features/docs/exception_in_around_hook.feature +0 -80
  380. data/features/docs/exception_in_before_hook.feature +0 -96
  381. data/features/docs/extending_cucumber/custom_filter.feature +0 -29
  382. data/features/docs/extending_cucumber/custom_formatter.feature +0 -94
  383. data/features/docs/formatters/debug_formatter.feature +0 -47
  384. data/features/docs/formatters/formatter_step_file_colon_line.feature +0 -44
  385. data/features/docs/formatters/html_formatter.feature +0 -90
  386. data/features/docs/formatters/json_formatter.feature +0 -760
  387. data/features/docs/formatters/junit_formatter.feature +0 -316
  388. data/features/docs/formatters/pretty_formatter.feature +0 -73
  389. data/features/docs/formatters/progress_formatter.feature +0 -31
  390. data/features/docs/formatters/rerun_formatter.feature +0 -136
  391. data/features/docs/formatters/usage_formatter.feature +0 -102
  392. data/features/docs/getting_started.feature +0 -27
  393. data/features/docs/gherkin/background.feature +0 -559
  394. data/features/docs/gherkin/doc_strings.feature +0 -74
  395. data/features/docs/gherkin/expand_option_for_outlines.feature +0 -46
  396. data/features/docs/gherkin/language_from_header.feature +0 -32
  397. data/features/docs/gherkin/language_help.feature +0 -37
  398. data/features/docs/gherkin/outlines.feature +0 -158
  399. data/features/docs/gherkin/unicode_table.feature +0 -32
  400. data/features/docs/gherkin/using_descriptions.feature +0 -88
  401. data/features/docs/gherkin/using_star_notation.feature +0 -37
  402. data/features/docs/iso-8859-1.feature +0 -6
  403. data/features/docs/post_configuration_hook.feature +0 -49
  404. data/features/docs/profiles.feature +0 -120
  405. data/features/docs/rake_task.feature +0 -141
  406. data/features/docs/raketask.feature +0 -44
  407. data/features/docs/wire_protocol/erb_configuration.feature +0 -56
  408. data/features/docs/wire_protocol/handle_unexpected_response.feature +0 -30
  409. data/features/docs/wire_protocol/invoke_message.feature +0 -216
  410. data/features/docs/wire_protocol/readme.md +0 -26
  411. data/features/docs/wire_protocol/snippets_message.feature +0 -51
  412. data/features/docs/wire_protocol/step_matches_message.feature +0 -81
  413. data/features/docs/wire_protocol/table_diffing.feature +0 -126
  414. data/features/docs/wire_protocol/tags.feature +0 -87
  415. data/features/docs/wire_protocol/timeouts.feature +0 -64
  416. data/features/docs/work_in_progress.feature +0 -154
  417. data/features/docs/writing_support_code/after_hooks.feature +0 -56
  418. data/features/docs/writing_support_code/around_hooks.feature +0 -260
  419. data/features/docs/writing_support_code/before_hook.feature +0 -65
  420. data/features/docs/writing_support_code/hook_order.feature +0 -61
  421. data/features/docs/writing_support_code/load_path.feature +0 -17
  422. data/features/docs/writing_support_code/state.feature +0 -32
  423. data/features/docs/writing_support_code/tagged_hooks.feature +0 -95
  424. data/features/lib/step_definitions/aruba_steps.rb +0 -22
  425. data/features/lib/step_definitions/cucumber_steps.rb +0 -50
  426. data/features/lib/step_definitions/iso-8859-1_steps.rb +0 -12
  427. data/features/lib/step_definitions/json_steps.rb +0 -7
  428. data/features/lib/step_definitions/junit_steps.rb +0 -13
  429. data/features/lib/step_definitions/language_steps.rb +0 -8
  430. data/features/lib/step_definitions/profile_steps.rb +0 -15
  431. data/features/lib/step_definitions/ruby_steps.rb +0 -3
  432. data/features/lib/step_definitions/wire_steps.rb +0 -58
  433. data/features/lib/support/env.rb +0 -21
  434. data/features/lib/support/fake_wire_server.rb +0 -80
  435. data/features/lib/support/feature_factory.rb +0 -67
  436. data/features/lib/support/normalise_output.rb +0 -33
  437. data/gem_tasks/contributors.rake +0 -15
  438. data/gem_tasks/cov.rake +0 -5
  439. data/gem_tasks/cucumber.rake +0 -25
  440. data/gem_tasks/downloads.rb +0 -7
  441. data/gem_tasks/environment.rake +0 -7
  442. data/gem_tasks/examples.rake +0 -11
  443. data/gem_tasks/fix_cr_lf.rake +0 -10
  444. data/gem_tasks/flog.rake +0 -4
  445. data/gem_tasks/rspec.rake +0 -6
  446. data/gem_tasks/sass.rake +0 -4
  447. data/gem_tasks/stats +0 -16
  448. data/gem_tasks/versions.txt +0 -74
  449. data/lib/cucumber/ast.rb +0 -13
  450. data/lib/cucumber/core_ext/instance_exec.rb +0 -70
  451. data/lib/cucumber/core_ext/proc.rb +0 -36
  452. data/lib/cucumber/formatter/cucumber.css +0 -285
  453. data/lib/cucumber/formatter/cucumber.sass +0 -247
  454. data/lib/cucumber/formatter/debug.rb +0 -35
  455. data/lib/cucumber/formatter/gherkin_formatter_adapter.rb +0 -204
  456. data/lib/cucumber/formatter/gpretty.rb +0 -24
  457. data/lib/cucumber/formatter/html.rb +0 -674
  458. data/lib/cucumber/formatter/jquery-min.js +0 -154
  459. data/lib/cucumber/formatter/json_pretty.rb +0 -10
  460. data/lib/cucumber/formatter/legacy_api/adapter.rb +0 -1060
  461. data/lib/cucumber/formatter/legacy_api/ast.rb +0 -376
  462. data/lib/cucumber/formatter/legacy_api/results.rb +0 -51
  463. data/lib/cucumber/formatter/legacy_api/runtime_facade.rb +0 -30
  464. data/lib/cucumber/language_support.rb +0 -30
  465. data/lib/cucumber/language_support/language_methods.rb +0 -71
  466. data/lib/cucumber/rb_support/rb_hook.rb +0 -18
  467. data/lib/cucumber/rb_support/rb_language.rb +0 -160
  468. data/lib/cucumber/rb_support/rb_step_definition.rb +0 -131
  469. data/lib/cucumber/rb_support/rb_transform.rb +0 -59
  470. data/lib/cucumber/rb_support/rb_world.rb +0 -142
  471. data/lib/cucumber/rb_support/regexp_argument_matcher.rb +0 -21
  472. data/lib/cucumber/rb_support/snippet.rb +0 -149
  473. data/lib/cucumber/wire_support/configuration.rb +0 -38
  474. data/lib/cucumber/wire_support/connection.rb +0 -61
  475. data/lib/cucumber/wire_support/request_handler.rb +0 -32
  476. data/lib/cucumber/wire_support/wire_exception.rb +0 -32
  477. data/lib/cucumber/wire_support/wire_language.rb +0 -68
  478. data/lib/cucumber/wire_support/wire_packet.rb +0 -34
  479. data/lib/cucumber/wire_support/wire_protocol.rb +0 -43
  480. data/lib/cucumber/wire_support/wire_protocol/requests.rb +0 -133
  481. data/lib/cucumber/wire_support/wire_step_definition.rb +0 -19
  482. data/spec/cucumber/cli/configuration_spec.rb +0 -503
  483. data/spec/cucumber/cli/main_spec.rb +0 -105
  484. data/spec/cucumber/cli/options_spec.rb +0 -335
  485. data/spec/cucumber/cli/profile_loader_spec.rb +0 -46
  486. data/spec/cucumber/configuration_spec.rb +0 -23
  487. data/spec/cucumber/constantize_spec.rb +0 -19
  488. data/spec/cucumber/core_ext/proc_spec.rb +0 -69
  489. data/spec/cucumber/file_specs_spec.rb +0 -60
  490. data/spec/cucumber/filters/activate_steps_spec.rb +0 -57
  491. data/spec/cucumber/filters/gated_receiver_spec.rb +0 -47
  492. data/spec/cucumber/filters/tag_limits/test_case_index_spec.rb +0 -38
  493. data/spec/cucumber/filters/tag_limits/verifier_spec.rb +0 -57
  494. data/spec/cucumber/filters/tag_limits_spec.rb +0 -59
  495. data/spec/cucumber/formatter/ansicolor_spec.rb +0 -32
  496. data/spec/cucumber/formatter/debug_spec.rb +0 -64
  497. data/spec/cucumber/formatter/duration_spec.rb +0 -22
  498. data/spec/cucumber/formatter/html_spec.rb +0 -459
  499. data/spec/cucumber/formatter/interceptor_spec.rb +0 -136
  500. data/spec/cucumber/formatter/junit_spec.rb +0 -252
  501. data/spec/cucumber/formatter/legacy_api/adapter_spec.rb +0 -2090
  502. data/spec/cucumber/formatter/pretty_spec.rb +0 -823
  503. data/spec/cucumber/formatter/progress_spec.rb +0 -85
  504. data/spec/cucumber/formatter/rerun_spec.rb +0 -107
  505. data/spec/cucumber/formatter/spec_helper.rb +0 -67
  506. data/spec/cucumber/hooks_spec.rb +0 -30
  507. data/spec/cucumber/multiline_argument/data_table_spec.rb +0 -514
  508. data/spec/cucumber/project_initializer_spec.rb +0 -87
  509. data/spec/cucumber/rake/forked_spec.rb +0 -53
  510. data/spec/cucumber/rb_support/rb_language_spec.rb +0 -279
  511. data/spec/cucumber/rb_support/rb_step_definition_spec.rb +0 -170
  512. data/spec/cucumber/rb_support/rb_transform_spec.rb +0 -46
  513. data/spec/cucumber/rb_support/regexp_argument_matcher_spec.rb +0 -22
  514. data/spec/cucumber/rb_support/snippet_spec.rb +0 -137
  515. data/spec/cucumber/running_test_case_spec.rb +0 -83
  516. data/spec/cucumber/runtime/for_programming_languages_spec.rb +0 -36
  517. data/spec/cucumber/runtime/support_code_spec.rb +0 -123
  518. data/spec/cucumber/runtime_spec.rb +0 -46
  519. data/spec/cucumber/sell_cucumbers.feature +0 -19
  520. data/spec/cucumber/step_match_spec.rb +0 -79
  521. data/spec/cucumber/wire_support/configuration_spec.rb +0 -64
  522. data/spec/cucumber/wire_support/connection_spec.rb +0 -64
  523. data/spec/cucumber/wire_support/wire_exception_spec.rb +0 -50
  524. data/spec/cucumber/wire_support/wire_language_spec.rb +0 -46
  525. data/spec/cucumber/wire_support/wire_packet_spec.rb +0 -44
  526. data/spec/cucumber/world/pending_spec.rb +0 -46
  527. data/spec/spec_helper.rb +0 -29
@@ -1,17 +1,16 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'builder'
4
+ require 'cucumber/formatter/backtrace_filter'
2
5
  require 'cucumber/formatter/io'
3
6
  require 'cucumber/formatter/interceptor'
4
7
  require 'fileutils'
8
+ require 'cucumber/formatter/ast_lookup'
5
9
 
6
10
  module Cucumber
7
11
  module Formatter
8
12
  # The formatter used for <tt>--format junit</tt>
9
13
  class Junit
10
-
11
- # TODO: remove coupling to types
12
- AST_SCENARIO_OUTLINE = Core::Ast::ScenarioOutline
13
- AST_EXAMPLE_ROW = LegacyApi::Ast::ExampleTableRow
14
-
15
14
  include Io
16
15
 
17
16
  class UnNamedFeatureError < StandardError
@@ -20,166 +19,150 @@ module Cucumber
20
19
  end
21
20
  end
22
21
 
23
- def initialize(runtime, io, options)
24
- @reportdir = ensure_dir(io, "junit")
25
- @options = options
22
+ def initialize(config)
23
+ @ast_lookup = AstLookup.new(config)
24
+ config.on_event :test_case_started, &method(:on_test_case_started)
25
+ config.on_event :test_case_finished, &method(:on_test_case_finished)
26
+ config.on_event :test_step_finished, &method(:on_test_step_finished)
27
+ config.on_event :test_run_finished, &method(:on_test_run_finished)
28
+ @reportdir = ensure_dir(config.out_stream, 'junit')
29
+ @config = config
30
+ @features_data = Hash.new do |h, k|
31
+ h[k] = {
32
+ feature: nil,
33
+ failures: 0,
34
+ errors: 0,
35
+ tests: 0,
36
+ skipped: 0,
37
+ time: 0,
38
+ builder: Builder::XmlMarkup.new(indent: 2)
39
+ }
40
+ end
26
41
  end
27
42
 
28
- def before_feature(feature)
29
- @current_feature = feature
30
- @failures = @errors = @tests = @skipped = 0
31
- @builder = Builder::XmlMarkup.new( :indent => 2 )
32
- @time = 0
43
+ def on_test_case_started(event)
44
+ test_case = event.test_case
45
+ start_feature(test_case) unless same_feature_as_previous_test_case?(test_case)
46
+ @failing_test_step = nil
33
47
  # In order to fill out <system-err/> and <system-out/>, we need to
34
48
  # intercept the $stderr and $stdout
35
49
  @interceptedout = Interceptor::Pipe.wrap(:stdout)
36
50
  @interceptederr = Interceptor::Pipe.wrap(:stderr)
37
51
  end
38
52
 
39
- def before_feature_element(feature_element)
40
- @in_examples = AST_SCENARIO_OUTLINE === feature_element
41
- @steps_start = Time.now
42
- end
53
+ def on_test_step_finished(event)
54
+ test_step, result = *event.attributes
55
+ return if @failing_test_step
43
56
 
44
- def after_feature(feature)
45
- @testsuite = Builder::XmlMarkup.new( :indent => 2 )
46
- @testsuite.instruct!
47
- @testsuite.testsuite(
48
- :failures => @failures,
49
- :errors => @errors,
50
- :skipped => @skipped,
51
- :tests => @tests,
52
- :time => "%.6f" % @time,
53
- :name => @feature_name ) do
54
- @testsuite << @builder.target!
55
- @testsuite.tag!('system-out') do
56
- @testsuite.cdata! strip_control_chars(@interceptedout.buffer.join)
57
- end
58
- @testsuite.tag!('system-err') do
59
- @testsuite.cdata! strip_control_chars(@interceptederr.buffer.join)
60
- end
61
- end
57
+ @failing_test_step = test_step unless result.ok?(@config.strict)
58
+ end
62
59
 
63
- write_file(feature_result_filename(feature.file), @testsuite.target!)
60
+ def on_test_case_finished(event)
61
+ test_case, result = *event.attributes
62
+ result = result.with_filtered_backtrace(Cucumber::Formatter::BacktraceFilter)
63
+ test_case_name = NameBuilder.new(test_case, @ast_lookup)
64
+ scenario = test_case_name.scenario_name
65
+ scenario_designation = "#{scenario}#{test_case_name.name_suffix}"
66
+ output = create_output_string(test_case, scenario, result, test_case_name.row_name)
67
+ build_testcase(result, scenario_designation, output)
64
68
 
65
69
  Interceptor::Pipe.unwrap! :stdout
66
70
  Interceptor::Pipe.unwrap! :stderr
67
71
  end
68
72
 
69
- def before_background(*args)
70
- @in_background = true
73
+ def on_test_run_finished(_event)
74
+ @features_data.each { |_file, data| end_feature(data) }
71
75
  end
72
76
 
73
- def after_background(*args)
74
- @in_background = false
75
- end
76
-
77
- def feature_name(keyword, name)
78
- raise UnNamedFeatureError.new(@current_feature.file) if name.empty?
79
- lines = name.split(/\r?\n/)
80
- @feature_name = lines[0]
81
- end
77
+ private
82
78
 
83
- def scenario_name(keyword, name, file_colon_line, source_indent)
84
- @scenario = (name.nil? || name == "") ? "Unnamed scenario" : name.split("\n")[0]
85
- @output = "#{keyword}: #{@scenario}\n\n"
79
+ def same_feature_as_previous_test_case?(test_case)
80
+ @current_feature_data && @current_feature_data[:uri] == test_case.location.file
86
81
  end
87
82
 
88
- def before_steps(steps)
83
+ def start_feature(test_case)
84
+ uri = test_case.location.file
85
+ feature = @ast_lookup.gherkin_document(uri).feature
86
+ raise UnNamedFeatureError, uri if feature.name.empty?
87
+ @current_feature_data = @features_data[uri]
88
+ @current_feature_data[:uri] = uri unless @current_feature_data[:uri]
89
+ @current_feature_data[:feature] = feature unless @current_feature_data[:feature]
89
90
  end
90
91
 
91
- def after_steps(steps)
92
- return if @in_background || @in_examples
93
-
94
- duration = Time.now - @steps_start
95
- if steps.failed?
96
- steps.each { |step| @output += "#{step.keyword}#{step.name}\n" }
97
- @output += "\nMessage:\n"
92
+ def end_feature(feature_data)
93
+ @testsuite = Builder::XmlMarkup.new(indent: 2)
94
+ @testsuite.instruct!
95
+ @testsuite.testsuite(
96
+ failures: feature_data[:failures],
97
+ errors: feature_data[:errors],
98
+ skipped: feature_data[:skipped],
99
+ tests: feature_data[:tests],
100
+ time: format('%<time>.6f', time: feature_data[:time]),
101
+ name: feature_data[:feature].name
102
+ ) do
103
+ @testsuite << feature_data[:builder].target!
98
104
  end
99
- build_testcase(duration, steps.status, steps.exception)
100
- end
101
-
102
- def before_examples(*args)
103
- @header_row = true
104
- @in_examples = true
105
- end
106
105
 
107
- def after_examples(*args)
108
- @in_examples = false
109
- end
110
-
111
- def before_table_row(table_row)
112
- return unless @in_examples
113
-
114
- @table_start = Time.now
115
- end
116
-
117
- def after_table_row(table_row)
118
- return unless @in_examples and AST_EXAMPLE_ROW === table_row
119
- duration = Time.now - @table_start
120
- unless @header_row
121
- name_suffix = " (outline example : #{table_row.name})"
122
- if table_row.failed?
123
- @output += "Example row: #{table_row.name}\n"
124
- @output += "\nMessage:\n"
106
+ write_file(feature_result_filename(feature_data[:uri]), @testsuite.target!)
107
+ end
108
+
109
+ def create_output_string(test_case, scenario, result, row_name) # rubocop:disable Metrics/PerceivedComplexity
110
+ scenario_source = @ast_lookup.scenario_source(test_case)
111
+ keyword = scenario_source.type == :Scenario ? scenario_source.scenario.keyword : scenario_source.scenario_outline.keyword
112
+ output = "#{keyword}: #{scenario}\n\n"
113
+ return output if result.ok?(@config.strict)
114
+ if scenario_source.type == :Scenario
115
+ if @failing_test_step
116
+ if @failing_test_step.hook?
117
+ output += "#{@failing_test_step.text} at #{@failing_test_step.location}\n"
118
+ else
119
+ step_source = @ast_lookup.step_source(@failing_test_step).step
120
+ output += "#{step_source.keyword}#{@failing_test_step.text}\n"
121
+ end
122
+ else # An Around hook has failed
123
+ output += "Around hook\n"
125
124
  end
126
- build_testcase(duration, table_row.status, table_row.exception, name_suffix)
127
- end
128
-
129
- @header_row = false if @header_row
130
- end
131
-
132
- def before_test_case(test_case)
133
- if @options[:expand] and test_case.keyword == "Scenario Outline"
134
- @exception = nil
125
+ else
126
+ output += "Example row: #{row_name}\n"
135
127
  end
136
- end
137
-
138
- def after_step_result(keyword, step_match, multiline_arg, status, exception, source_indent, background, file_colon_line)
139
- if @options[:expand] and @in_examples
140
- if not @exception and exception
141
- @exception = exception
128
+ output + "\nMessage:\n"
129
+ end
130
+
131
+ def build_testcase(result, scenario_designation, output)
132
+ duration = ResultBuilder.new(result).test_case_duration
133
+ @current_feature_data[:time] += duration
134
+ classname = @current_feature_data[:feature].name
135
+ name = scenario_designation
136
+
137
+ @current_feature_data[:builder].testcase(classname: classname, name: name, time: format('%<duration>.6f', duration: duration)) do
138
+ if !result.passed? && result.ok?(@config.strict)
139
+ @current_feature_data[:builder].skipped
140
+ @current_feature_data[:skipped] += 1
141
+ elsif !result.passed?
142
+ status = result.to_sym
143
+ exception = get_backtrace_object(result)
144
+ @current_feature_data[:builder].failure(message: "#{status} #{name}", type: status) do
145
+ @current_feature_data[:builder].cdata! output
146
+ @current_feature_data[:builder].cdata!(format_exception(exception)) if exception
147
+ end
148
+ @current_feature_data[:failures] += 1
142
149
  end
143
- end
144
- end
145
-
146
- def after_test_case(test_case, result)
147
- if @options[:expand] and test_case.keyword == "Scenario Outline"
148
- test_case_name = NameBuilder.new(test_case)
149
- @scenario = test_case_name.outline_name
150
- @output = "#{test_case.keyword}: #{@scenario}\n\n"
151
- if result.failed?
152
- @output += "Example row: #{test_case_name.row_name}\n"
153
- @output += "\nMessage:\n"
150
+ @current_feature_data[:builder].tag!('system-out') do
151
+ @current_feature_data[:builder].cdata! strip_control_chars(@interceptedout.buffer_string)
152
+ end
153
+ @current_feature_data[:builder].tag!('system-err') do
154
+ @current_feature_data[:builder].cdata! strip_control_chars(@interceptederr.buffer_string)
154
155
  end
155
- test_case_result = ResultBuilder.new(result)
156
- build_testcase(test_case_result.test_case_duration, test_case_result.status, @exception, test_case_name.name_suffix)
157
156
  end
157
+ @current_feature_data[:tests] += 1
158
158
  end
159
159
 
160
- private
161
-
162
- def build_testcase(duration, status, exception = nil, suffix = "")
163
- @time += duration
164
- classname = @feature_name
165
- name = "#{@scenario}#{suffix}"
166
- pending = [:pending, :undefined].include?(status) && (!@options[:strict])
167
-
168
- @builder.testcase(:classname => classname, :name => name, :time => "%.6f" % duration) do
169
- if status == :skipped || pending
170
- @builder.skipped
171
- @skipped += 1
172
- elsif status != :passed
173
- @builder.failure(:message => "#{status.to_s} #{name}", :type => status.to_s) do
174
- @builder.cdata! @output
175
- @builder.cdata!(format_exception(exception)) if exception
176
- end
177
- @failures += 1
178
- end
179
- @builder.tag!('system-out')
180
- @builder.tag!('system-err')
160
+ def get_backtrace_object(result)
161
+ if result.failed?
162
+ result.exception
163
+ elsif result.backtrace
164
+ result
181
165
  end
182
- @tests += 1
183
166
  end
184
167
 
185
168
  def format_exception(exception)
@@ -191,7 +174,7 @@ module Cucumber
191
174
  end
192
175
 
193
176
  def basename(feature_file)
194
- File.basename(feature_file.gsub(/[\\\/]/, '-'), '.feature')
177
+ File.basename(feature_file.gsub(/[\\\/]/, '-'), '.feature') # rubocop:disable Style/RegexpLiteral
195
178
  end
196
179
 
197
180
  def write_file(feature_filename, data)
@@ -202,74 +185,61 @@ module Cucumber
202
185
  def strip_control_chars(cdata)
203
186
  cdata.scan(/[[:print:]\t\n\r]/).join
204
187
  end
205
-
206
188
  end
207
189
 
208
190
  class NameBuilder
209
- attr_reader :outline_name, :name_suffix, :row_name
210
-
211
- def initialize(test_case)
212
- test_case.describe_source_to self
213
- end
214
-
215
- def feature(*)
216
- self
191
+ attr_reader :scenario_name, :name_suffix, :row_name
192
+
193
+ def initialize(test_case, ast_lookup)
194
+ @name_suffix = ''
195
+ @row_name = ''
196
+ scenario_source = ast_lookup.scenario_source(test_case)
197
+ if scenario_source.type == :Scenario
198
+ scenario(scenario_source.scenario)
199
+ else
200
+ scenario_outline(scenario_source.scenario_outline)
201
+ examples_table_row(scenario_source.row)
202
+ end
217
203
  end
218
204
 
219
- def scenario(*)
220
- self
205
+ def scenario(scenario)
206
+ @scenario_name = scenario.name.empty? ? 'Unnamed scenario' : scenario.name
221
207
  end
222
208
 
223
209
  def scenario_outline(outline)
224
- @outline_name = outline.name
225
- self
226
- end
227
-
228
- def examples_table(*)
229
- self
210
+ @scenario_name = outline.name.empty? ? 'Unnamed scenario outline' : outline.name
230
211
  end
231
212
 
232
213
  def examples_table_row(row)
233
- @row_name = '| ' + row.values.join(' | ') + ' |'
214
+ @row_name = '| ' + row.cells.map(&:value).join(' | ') + ' |'
234
215
  @name_suffix = " (outline example : #{@row_name})"
235
- self
236
216
  end
237
217
  end
238
218
 
239
219
  class ResultBuilder
240
- attr_reader :status, :test_case_duration
220
+ attr_reader :test_case_duration
241
221
  def initialize(result)
242
222
  @test_case_duration = 0
243
223
  result.describe_to(self)
244
224
  end
245
225
 
246
- def passed
247
- @status = :passed
248
- end
226
+ def passed(*) end
249
227
 
250
- def failed
251
- @status = :failed
252
- end
228
+ def failed(*) end
253
229
 
254
- def undefined
255
- @status = :undefined
256
- end
230
+ def undefined(*) end
257
231
 
258
- def skipped
259
- @status = :skipped
260
- end
232
+ def skipped(*) end
261
233
 
262
- def pending(*)
263
- @status = :pending
264
- end
234
+ def pending(*) end
265
235
 
266
- def exception(*)
267
- end
236
+ def exception(*) end
268
237
 
269
238
  def duration(duration, *)
270
- duration.tap { |duration| @test_case_duration = duration.nanoseconds / 10 ** 9.0 }
239
+ duration.tap { |dur| @test_case_duration = dur.nanoseconds / 10**9.0 }
271
240
  end
272
- end
273
241
 
242
+ def embed(*) end
243
+ end
274
244
  end
275
245
  end
@@ -1,248 +1,453 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'fileutils'
4
+ require 'gherkin/dialect'
2
5
  require 'cucumber/formatter/console'
3
6
  require 'cucumber/formatter/io'
4
- require 'gherkin/formatter/escaping'
7
+ require 'cucumber/gherkin/formatter/escaping'
8
+ require 'cucumber/formatter/console_counts'
9
+ require 'cucumber/formatter/console_issues'
10
+ require 'cucumber/formatter/duration_extractor'
11
+ require 'cucumber/formatter/backtrace_filter'
12
+ require 'cucumber/formatter/ast_lookup'
5
13
 
6
14
  module Cucumber
7
15
  module Formatter
8
16
  # The formatter used for <tt>--format pretty</tt> (the default formatter).
9
17
  #
10
- # This formatter prints features to plain text - exactly how they were parsed,
11
- # just prettier. That means with proper indentation and alignment of table columns.
18
+ # This formatter prints the result of the feature executions to plain text - exactly how they were parsed.
12
19
  #
13
20
  # If the output is STDOUT (and not a file), there are bright colours to watch too.
14
21
  #
15
- class Pretty
22
+ class Pretty # rubocop:disable Metrics/ClassLength
16
23
  include FileUtils
17
24
  include Console
18
25
  include Io
19
- include Gherkin::Formatter::Escaping
20
- attr_writer :indent
21
- attr_reader :runtime
22
-
23
- def initialize(runtime, path_or_io, options)
24
- @runtime, @io, @options = runtime, ensure_io(path_or_io, "pretty"), options
25
- @exceptions = []
26
- @indent = 0
27
- @prefixes = options[:prefixes] || {}
28
- @delayed_messages = []
29
- @previous_step_keyword = nil
26
+ include Cucumber::Gherkin::Formatter::Escaping
27
+ attr_reader :config, :options
28
+ private :config, :options
29
+ attr_reader :current_feature_uri, :current_scenario_outline, :current_examples, :current_test_case
30
+ private :current_feature_uri, :current_scenario_outline, :current_examples, :current_test_case
31
+ attr_reader :in_scenario_outline, :print_background_steps
32
+ private :in_scenario_outline, :print_background_steps
33
+
34
+ def initialize(config)
35
+ @io = ensure_io(config.out_stream)
36
+ @config = config
37
+ @options = config.to_hash
30
38
  @snippets_input = []
39
+ @total_duration = 0
40
+ @exceptions = []
41
+ @gherkin_sources = {}
42
+ @step_matches = {}
43
+ @ast_lookup = AstLookup.new(config)
44
+ @counts = ConsoleCounts.new(config)
45
+ @issues = ConsoleIssues.new(config, @ast_lookup)
46
+ @first_feature = true
47
+ @current_feature_uri = ''
48
+ @current_scenario_outline = nil
49
+ @current_examples = nil
50
+ @current_test_case = nil
51
+ @in_scenario_outline = false
52
+ @print_background_steps = false
53
+ @test_step_output = []
54
+ @passed_test_cases = []
55
+ @source_indent = 0
56
+ @next_comment_to_be_printed = 0
57
+ config.on_event :gherkin_source_read, &method(:on_gherkin_source_read)
58
+ config.on_event :step_activated, &method(:on_step_activated)
59
+ config.on_event :test_case_started, &method(:on_test_case_started)
60
+ config.on_event :test_step_started, &method(:on_test_step_started)
61
+ config.on_event :test_step_finished, &method(:on_test_step_finished)
62
+ config.on_event :test_case_finished, &method(:on_test_case_finished)
63
+ config.on_event :test_run_finished, &method(:on_test_run_finished)
64
+ end
65
+
66
+ def on_gherkin_source_read(event)
67
+ @gherkin_sources[event.path] = event.body
68
+ end
69
+
70
+ def on_step_activated(event)
71
+ test_step, step_match = *event.attributes
72
+ @step_matches[test_step.to_s] = step_match
73
+ end
74
+
75
+ def on_test_case_started(event)
76
+ if !same_feature_as_previous_test_case?(event.test_case.location)
77
+ if first_feature?
78
+ @first_feature = false
79
+ print_profile_information
80
+ else
81
+ print_comments(gherkin_source.split("\n").length, 0)
82
+ @io.puts
83
+ end
84
+ @current_feature_uri = event.test_case.location.file
85
+ @exceptions = []
86
+ print_feature_data
87
+ if feature_has_background?
88
+ print_background_data
89
+ @print_background_steps = true
90
+ @in_scenario_outline = false
91
+ end
92
+ else
93
+ @print_background_steps = false
94
+ end
95
+ @current_test_case = event.test_case
96
+ print_step_header(current_test_case) unless print_background_steps
97
+ end
98
+
99
+ def on_test_step_started(event)
100
+ return if event.test_step.hook?
101
+ print_step_header(current_test_case) if first_step_after_printing_background_steps?(event.test_step)
102
+ end
103
+
104
+ def on_test_step_finished(event)
105
+ collect_snippet_data(event.test_step, @ast_lookup) if event.result.undefined?
106
+ return if in_scenario_outline && !options[:expand]
107
+ exception_to_be_printed = find_exception_to_be_printed(event.result)
108
+ print_step_data(event.test_step, event.result) if print_step_data?(event, exception_to_be_printed)
109
+ print_step_output
110
+ return unless exception_to_be_printed
111
+ print_exception(exception_to_be_printed, event.result.to_sym, 6)
112
+ @exceptions << exception_to_be_printed
113
+ end
114
+
115
+ def on_test_case_finished(event)
116
+ @total_duration += DurationExtractor.new(event.result).result_duration
117
+ @passed_test_cases << event.test_case if config.wip? && event.result.passed?
118
+ if in_scenario_outline && !options[:expand]
119
+ print_row_data(event.test_case, event.result)
120
+ else
121
+ exception_to_be_printed = find_exception_to_be_printed(event.result)
122
+ return unless exception_to_be_printed
123
+ print_exception(exception_to_be_printed, event.result.to_sym, 6)
124
+ @exceptions << exception_to_be_printed
125
+ end
31
126
  end
32
127
 
33
- def before_features(features)
34
- print_profile_information
128
+ def on_test_run_finished(_event)
129
+ print_comments(gherkin_source.split("\n").length, 0) unless current_feature_uri.empty?
130
+ @io.puts
131
+ print_summary
35
132
  end
36
133
 
37
- def after_features(features)
38
- print_summary(features)
134
+ def puts(*messages)
135
+ messages.each do |message|
136
+ @test_step_output.push message
137
+ end
39
138
  end
40
139
 
41
- def before_feature(feature)
42
- @exceptions = []
43
- @indent = 0
140
+ private
141
+
142
+ def find_exception_to_be_printed(result)
143
+ return nil if result.ok?(options[:strict])
144
+ result = result.with_filtered_backtrace(Cucumber::Formatter::BacktraceFilter)
145
+ exception = result.failed? ? result.exception : result
146
+ return nil if @exceptions.include?(exception)
147
+ exception
44
148
  end
45
149
 
46
- def comment_line(comment_line)
47
- @io.puts(comment_line.indent(@indent))
48
- @io.flush
150
+ def calculate_source_indent(test_case)
151
+ scenario = scenario_source(test_case).scenario
152
+ @source_indent = calculate_source_indent_for_ast_node(scenario)
49
153
  end
50
154
 
51
- def after_tags(tags)
52
- if @indent == 1
53
- @io.puts
54
- @io.flush
155
+ def calculate_source_indent_for_ast_node(ast_node)
156
+ indent = 4 + ast_node.keyword.length
157
+ indent += 1 + ast_node.name.length
158
+ ast_node.steps.each do |step|
159
+ step_indent = 5 + step.keyword.length + step.text.length
160
+ indent = step_indent if step_indent > indent
55
161
  end
162
+ indent
56
163
  end
57
164
 
58
- def tag_name(tag_name)
59
- tag = format_string(tag_name, :tag).indent(@indent)
60
- @io.print(tag)
61
- @io.flush
62
- @indent = 1
165
+ def calculate_source_indent_for_expanded_test_case(test_case, scenario_keyword, expanded_name)
166
+ indent = 7 + scenario_keyword.length
167
+ indent += 2 + expanded_name.length
168
+ test_case.test_steps.each do |step|
169
+ if !step.hook? && step.location.lines.max >= test_case.location.lines.max
170
+ step_indent = 9 + test_step_keyword(step).length + step.text.length
171
+ indent = step_indent if step_indent > indent
172
+ end
173
+ end
174
+ indent
63
175
  end
64
176
 
65
- def feature_name(keyword, name)
66
- @io.puts("#{keyword}: #{name}")
67
- @io.puts
68
- @io.flush
177
+ def print_step_output
178
+ @test_step_output.each { |message| @io.puts(format_string(message, :tag).indent(6)) }
179
+ @test_step_output = []
69
180
  end
70
181
 
71
- def before_feature_element(feature_element)
72
- @indent = 2
73
- @scenario_indent = 2
182
+ def first_feature?
183
+ @first_feature
74
184
  end
75
185
 
76
- def after_feature_element(feature_element)
77
- print_messages
78
- @io.puts
79
- @io.flush
186
+ def same_feature_as_previous_test_case?(location)
187
+ location.file == current_feature_uri
80
188
  end
81
189
 
82
- def before_background(background)
83
- @indent = 2
84
- @scenario_indent = 2
85
- @in_background = true
190
+ def feature_has_background?
191
+ feature_children = gherkin_document.feature.children
192
+ return false if feature_children.empty?
193
+ !feature_children.first.background.nil?
86
194
  end
87
195
 
88
- def after_background(background)
89
- print_messages
90
- @in_background = nil
91
- @io.puts
92
- @io.flush
196
+ def print_step_header(test_case)
197
+ if from_scenario_outline?(test_case)
198
+ @in_scenario_outline = true
199
+ unless same_outline_as_previous_test_case?(test_case)
200
+ @current_scenario_outline = scenario_source(test_case).scenario_outline
201
+ @io.puts
202
+ print_outline_data(current_scenario_outline)
203
+ end
204
+ unless same_examples_as_previous_test_case?(test_case)
205
+ @current_examples = scenario_source(test_case).examples
206
+ @io.puts
207
+ print_examples_data(current_examples)
208
+ end
209
+ print_expanded_row_data(current_test_case) if options[:expand]
210
+ else
211
+ @in_scenario_outline = false
212
+ @current_scenario_outline = nil
213
+ @current_examples = nil
214
+ @io.puts
215
+ @source_indent = calculate_source_indent(current_test_case)
216
+ print_scenario_data(test_case)
217
+ end
93
218
  end
94
219
 
95
- def background_name(keyword, name, file_colon_line, source_indent)
96
- print_feature_element_name(keyword, name, file_colon_line, source_indent)
220
+ def same_outline_as_previous_test_case?(test_case)
221
+ scenario_source(test_case).scenario_outline == current_scenario_outline
97
222
  end
98
223
 
99
- def before_examples_array(examples_array)
100
- @indent = 4
101
- @io.puts
102
- @visiting_first_example_name = true
224
+ def same_examples_as_previous_test_case?(test_case)
225
+ scenario_source(test_case).examples == current_examples
226
+ end
227
+
228
+ def from_scenario_outline?(test_case)
229
+ scenario = scenario_source(test_case)
230
+ scenario.type != :Scenario
231
+ end
232
+
233
+ def first_step_after_printing_background_steps?(test_step)
234
+ return false unless print_background_steps
235
+ return false unless test_step.location.lines.max >= current_test_case.location.lines.max
236
+ @print_background_steps = false
237
+ true
103
238
  end
104
239
 
105
- def examples_name(keyword, name)
106
- @io.puts unless @visiting_first_example_name
107
- @visiting_first_example_name = false
108
- names = name.strip.empty? ? [name.strip] : name.split("\n")
109
- @io.puts(" #{keyword}: #{names[0]}")
110
- names[1..-1].each {|s| @io.puts " #{s}" } unless names.empty?
240
+ def print_feature_data
241
+ feature = gherkin_document.feature
242
+ print_language_comment(feature.location.line)
243
+ print_comments(feature.location.line, 0)
244
+ print_tags(feature.tags, 0)
245
+ print_feature_line(feature)
246
+ print_description(feature.description)
111
247
  @io.flush
112
- @indent = 6
113
- @scenario_indent = 6
114
248
  end
115
249
 
116
- def before_outline_table(outline_table)
117
- @table = outline_table
250
+ def print_language_comment(feature_line)
251
+ gherkin_source.split("\n")[0..feature_line].each do |line|
252
+ @io.puts(format_string(line, :comment)) if /# *language *:/ =~ line
253
+ end
118
254
  end
119
255
 
120
- def after_outline_table(outline_table)
121
- @table = nil
122
- @indent = 4
256
+ def print_comments(up_to_line, indent)
257
+ comments = gherkin_document.comments
258
+ return if comments.empty? || comments.length <= @next_comment_to_be_printed
259
+ comments[@next_comment_to_be_printed..-1].each do |comment|
260
+ if comment.location.line <= up_to_line
261
+ @io.puts(format_string(comment.text.strip, :comment).indent(indent))
262
+ @next_comment_to_be_printed += 1
263
+ end
264
+ break if @next_comment_to_be_printed >= comments.length
265
+ end
123
266
  end
124
267
 
125
- def scenario_name(keyword, name, file_colon_line, source_indent)
126
- print_feature_element_name(keyword, name, file_colon_line, source_indent)
268
+ def print_tags(tags, indent)
269
+ return if !tags || tags.empty?
270
+ @io.puts(tags.map { |tag| format_string(tag.name, :tag) }.join(' ').indent(indent))
127
271
  end
128
272
 
129
- def before_step(step)
130
- @current_step = step
131
- @indent = 6
132
- print_messages
273
+ def print_feature_line(feature)
274
+ print_keyword_name(feature.keyword, feature.name, 0)
133
275
  end
134
276
 
135
- def before_step_result(keyword, step_match, multiline_arg, status, exception, source_indent, background, file_colon_line)
136
- @hide_this_step = false
137
- if exception
138
- if @exceptions.include?(exception)
139
- @hide_this_step = true
140
- return
141
- end
142
- @exceptions << exception
277
+ def print_keyword_name(keyword, name, indent, location = nil)
278
+ line = "#{keyword}:"
279
+ line += " #{name}"
280
+ @io.print(line.indent(indent))
281
+ if location && options[:source]
282
+ line_comment = format_string("# #{location}", :comment).indent(@source_indent - line.length - indent)
283
+ @io.print(line_comment)
143
284
  end
144
- if status != :failed && @in_background ^ background
145
- @hide_this_step = true
146
- return
285
+ @io.puts
286
+ end
287
+
288
+ def print_description(description)
289
+ return unless description
290
+ description.split("\n").each do |line|
291
+ @io.puts(line)
147
292
  end
148
- @status = status
149
293
  end
150
294
 
151
- def step_name(keyword, step_match, status, source_indent, background, file_colon_line)
152
- return if @hide_this_step
153
- source_indent = nil unless @options[:source]
154
- name_to_report = format_step(keyword, step_match, status, source_indent)
155
- @io.puts(name_to_report.indent(@scenario_indent + 2))
156
- print_messages
295
+ def print_background_data
296
+ @io.puts
297
+ background = gherkin_document.feature.children.first.background
298
+ @source_indent = calculate_source_indent_for_ast_node(background) if options[:source]
299
+ print_comments(background.location.line, 2)
300
+ print_background_line(background)
301
+ print_description(background.description)
302
+ @io.flush
303
+ end
304
+
305
+ def print_background_line(background)
306
+ print_keyword_name(background.keyword, background.name, 2, "#{current_feature_uri}:#{background.location.line}")
157
307
  end
158
308
 
159
- def doc_string(string)
160
- return if @options[:no_multiline] || @hide_this_step
161
- s = %{"""\n#{string}\n"""}.indent(@indent)
162
- s = s.split("\n").map{|l| l =~ /^\s+$/ ? '' : l}.join("\n")
163
- @io.puts(format_string(s, @current_step.status))
309
+ def print_scenario_data(test_case)
310
+ scenario = scenario_source(test_case).scenario
311
+ print_comments(scenario.location.line, 2)
312
+ print_tags(scenario.tags, 2)
313
+ print_scenario_line(scenario, test_case.location)
314
+ print_description(scenario.description)
164
315
  @io.flush
165
316
  end
166
317
 
167
- def exception(exception, status)
168
- return if @hide_this_step
169
- print_messages
170
- print_exception(exception, status, @indent)
318
+ def print_scenario_line(scenario, location = nil)
319
+ print_keyword_name(scenario.keyword, scenario.name, 2, location)
320
+ end
321
+
322
+ def print_step_data?(event, exception_to_be_printed)
323
+ !event.test_step.hook? && (
324
+ print_background_steps ||
325
+ event.test_step.location.lines.max >= current_test_case.location.lines.max ||
326
+ exception_to_be_printed
327
+ )
328
+ end
329
+
330
+ def print_step_data(test_step, result)
331
+ base_indent = options[:expand] && in_scenario_outline ? 8 : 4
332
+ step_keyword = test_step_keyword(test_step)
333
+ indent = options[:source] ? @source_indent - step_keyword.length - test_step.text.length - base_indent : nil
334
+ print_comments(test_step.location.lines.max, base_indent)
335
+ name_to_report = format_step(step_keyword, @step_matches.fetch(test_step.to_s) { NoStepMatch.new(test_step, test_step.text) }, result.to_sym, indent)
336
+ @io.puts(name_to_report.indent(base_indent))
337
+ print_multiline_argument(test_step, result, base_indent + 2) unless options[:no_multiline]
171
338
  @io.flush
172
339
  end
173
340
 
174
- def before_multiline_arg(multiline_arg)
175
- return if @options[:no_multiline] || @hide_this_step
176
- @table = multiline_arg
341
+ def test_step_keyword(test_step)
342
+ step = step_source(test_step).step
343
+ step.keyword
177
344
  end
178
345
 
179
- def after_multiline_arg(multiline_arg)
180
- @table = nil
346
+ def step_source(test_step)
347
+ @ast_lookup.step_source(test_step)
181
348
  end
182
349
 
183
- def before_table_row(table_row)
184
- return if !@table || @hide_this_step
185
- @col_index = 0
186
- @io.print ' |'.indent(@indent-2)
350
+ def scenario_source(test_case)
351
+ @ast_lookup.scenario_source(test_case)
187
352
  end
188
353
 
189
- def after_table_row(table_row)
190
- return if !@table || @hide_this_step
191
- print_table_row_messages
192
- @io.puts
193
- if table_row.exception && !@exceptions.include?(table_row.exception)
194
- print_exception(table_row.exception, table_row.status, @indent)
195
- end
354
+ def gherkin_source
355
+ @gherkin_sources[current_feature_uri]
196
356
  end
197
357
 
198
- def after_table_cell(cell)
199
- return unless @table
200
- @col_index += 1
358
+ def gherkin_document
359
+ @ast_lookup.gherkin_document(current_feature_uri)
201
360
  end
202
361
 
203
- def table_cell_value(value, status)
204
- return if !@table || @hide_this_step
205
- status ||= @status || :passed
206
- width = @table.col_width(@col_index)
207
- cell_text = escape_cell(value.to_s || '')
208
- padded = cell_text + (' ' * (width - cell_text.unpack('U*').length))
209
- prefix = cell_prefix(status)
210
- @io.print(' ' + format_string("#{prefix}#{padded}", status) + ::Cucumber::Term::ANSIColor.reset(" |"))
211
- @io.flush
362
+ def print_multiline_argument(test_step, result, indent)
363
+ step = step_source(test_step).step
364
+ if !step.doc_string.nil?
365
+ print_doc_string(step.doc_string.content, result.to_sym, indent)
366
+ elsif !step.data_table.nil?
367
+ print_data_table(step.data_table, result.to_sym, indent)
368
+ end
212
369
  end
213
370
 
214
- def before_test_case(test_case)
215
- @previous_step_keyword = nil
371
+ def print_data_table(data_table, status, indent)
372
+ data_table.rows.each do |row|
373
+ print_comments(row.location.line, indent)
374
+ @io.puts format_string(gherkin_source.split("\n")[row.location.line - 1].strip, status).indent(indent)
375
+ end
216
376
  end
217
377
 
218
- def after_test_step(test_step, result)
219
- collect_snippet_data(test_step, result)
378
+ def print_outline_data(scenario_outline) # rubocop:disable Metrics/AbcSize
379
+ print_comments(scenario_outline.location.line, 2)
380
+ print_tags(scenario_outline.tags, 2)
381
+ @source_indent = calculate_source_indent_for_ast_node(scenario_outline) if options[:source]
382
+ print_scenario_line(scenario_outline, "#{current_feature_uri}:#{scenario_outline.location.line}")
383
+ print_description(scenario_outline.description)
384
+ scenario_outline.steps.each do |step|
385
+ print_comments(step.location.line, 4)
386
+ step_line = " #{step.keyword}#{step.text}"
387
+ @io.print(format_string(step_line, :skipped))
388
+ if options[:source]
389
+ comment_line = format_string("# #{current_feature_uri}:#{step.location.line}", :comment)
390
+ @io.print(comment_line.indent(@source_indent - step_line.length))
391
+ end
392
+ @io.puts
393
+ next if options[:no_multiline]
394
+ print_doc_string(step.doc_string.content, :skipped, 6) unless step.doc_string.nil?
395
+ print_data_table(step.data_table, :skipped, 6) unless step.data_table.nil?
396
+ end
397
+ @io.flush
220
398
  end
221
399
 
222
- private
400
+ def print_doc_string(content, status, indent)
401
+ s = %("""\n#{content}\n""").indent(indent)
402
+ s = s.split("\n").map { |l| l =~ /^\s+$/ ? '' : l }.join("\n")
403
+ @io.puts(format_string(s, status))
404
+ end
223
405
 
224
- def print_feature_element_name(keyword, name, file_colon_line, source_indent)
225
- @io.puts if @scenario_indent == 6
226
- names = name.empty? ? [name] : name.split("\n")
227
- line = "#{keyword}: #{names[0]}".indent(@scenario_indent)
228
- @io.print(line)
229
- if @options[:source]
230
- line_comment = "# #{file_colon_line}".indent(source_indent)
231
- @io.print(format_string(line_comment, :comment))
406
+ def print_examples_data(examples)
407
+ print_comments(examples.location.line, 4)
408
+ print_tags(examples.tags, 4)
409
+ print_keyword_name(examples.keyword, examples.name, 4)
410
+ print_description(examples.description)
411
+ unless options[:expand]
412
+ print_comments(examples.table_header.location.line, 6)
413
+ @io.puts(gherkin_source.split("\n")[examples.table_header.location.line - 1].strip.indent(6))
232
414
  end
415
+ @io.flush
416
+ end
417
+
418
+ def print_row_data(test_case, result)
419
+ print_comments(test_case.location.lines.max, 6)
420
+ @io.print(format_string(gherkin_source.split("\n")[test_case.location.lines.max - 1].strip, result.to_sym).indent(6))
421
+ @io.print(format_string(@test_step_output.join(', '), :tag).indent(2)) unless @test_step_output.empty?
422
+ @test_step_output = []
233
423
  @io.puts
234
- names[1..-1].each {|s| @io.puts " #{s}"}
424
+ if result.failed? || result.pending?
425
+ result = result.with_filtered_backtrace(Cucumber::Formatter::BacktraceFilter)
426
+ exception = result.failed? ? result.exception : result
427
+ unless @exceptions.include?(exception)
428
+ print_exception(exception, result.to_sym, 6)
429
+ @exceptions << exception
430
+ end
431
+ end
235
432
  @io.flush
236
433
  end
237
434
 
238
- def cell_prefix(status)
239
- @prefixes[status]
435
+ def print_expanded_row_data(test_case)
436
+ feature = gherkin_document.feature
437
+ language_code = feature.language || 'en'
438
+ language = ::Gherkin::Dialect.for(language_code)
439
+ scenario_keyword = language.scenario_keywords[0]
440
+ row = scenario_source(test_case).row
441
+ expanded_name = '| ' + row.cells.map(&:value).join(' | ') + ' |'
442
+ @source_indent = calculate_source_indent_for_expanded_test_case(test_case, scenario_keyword, expanded_name)
443
+ @io.puts
444
+ print_keyword_name(scenario_keyword, expanded_name, 6, test_case.location)
240
445
  end
241
446
 
242
- def print_summary(features)
243
- print_stats(features, @options)
244
- print_snippets(@options)
245
- print_passing_wip(@options)
447
+ def print_summary
448
+ print_statistics(@total_duration, config, @counts, @issues)
449
+ print_snippets(options)
450
+ print_passing_wip(config, @passed_test_cases, @ast_lookup)
246
451
  end
247
452
  end
248
453
  end