cuke_modeler 2.1.0 → 3.4.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.
Files changed (146) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +53 -1
  3. data/README.md +14 -16
  4. data/cuke_modeler.gemspec +33 -21
  5. data/lib/cuke_modeler.rb +2 -1
  6. data/lib/cuke_modeler/adapters/gherkin_10_adapter.rb +13 -0
  7. data/lib/cuke_modeler/adapters/gherkin_11_adapter.rb +13 -0
  8. data/lib/cuke_modeler/adapters/gherkin_12_adapter.rb +13 -0
  9. data/lib/cuke_modeler/adapters/gherkin_13_adapter.rb +13 -0
  10. data/lib/cuke_modeler/adapters/gherkin_14_adapter.rb +13 -0
  11. data/lib/cuke_modeler/adapters/gherkin_15_adapter.rb +13 -0
  12. data/lib/cuke_modeler/adapters/gherkin_9_adapter.rb +356 -4
  13. data/lib/cuke_modeler/containing.rb +41 -89
  14. data/lib/cuke_modeler/described.rb +40 -1
  15. data/lib/cuke_modeler/models/background.rb +12 -12
  16. data/lib/cuke_modeler/models/cell.rb +14 -8
  17. data/lib/cuke_modeler/models/comment.rb +6 -6
  18. data/lib/cuke_modeler/models/directory.rb +14 -18
  19. data/lib/cuke_modeler/models/doc_string.rb +11 -8
  20. data/lib/cuke_modeler/models/example.rb +64 -46
  21. data/lib/cuke_modeler/models/feature.rb +38 -20
  22. data/lib/cuke_modeler/models/feature_file.rb +6 -8
  23. data/lib/cuke_modeler/models/model.rb +2 -1
  24. data/lib/cuke_modeler/models/outline.rb +20 -15
  25. data/lib/cuke_modeler/models/row.rb +11 -8
  26. data/lib/cuke_modeler/models/rule.rb +101 -0
  27. data/lib/cuke_modeler/models/scenario.rb +18 -13
  28. data/lib/cuke_modeler/models/step.rb +41 -19
  29. data/lib/cuke_modeler/models/table.rb +10 -7
  30. data/lib/cuke_modeler/models/tag.rb +10 -6
  31. data/lib/cuke_modeler/named.rb +5 -1
  32. data/lib/cuke_modeler/nested.rb +22 -18
  33. data/lib/cuke_modeler/parsed.rb +8 -0
  34. data/lib/cuke_modeler/parsing.rb +55 -74
  35. data/lib/cuke_modeler/sourceable.rb +8 -0
  36. data/lib/cuke_modeler/stepped.rb +8 -0
  37. data/lib/cuke_modeler/taggable.rb +9 -1
  38. data/lib/cuke_modeler/version.rb +1 -1
  39. data/testing/cucumber/features/modeling/feature_modeling.feature +28 -7
  40. data/testing/cucumber/features/modeling/feature_output.feature +45 -23
  41. data/testing/cucumber/features/modeling/rule_modeling.feature +108 -0
  42. data/testing/cucumber/features/modeling/rule_output.feature +111 -0
  43. metadata +68 -161
  44. data/.gitignore +0 -18
  45. data/.simplecov +0 -7
  46. data/.travis.yml +0 -81
  47. data/Gemfile +0 -44
  48. data/Rakefile +0 -73
  49. data/appveyor.yml +0 -88
  50. data/lib/cuke_modeler/adapters/gherkin_2_adapter.rb +0 -274
  51. data/lib/cuke_modeler/adapters/gherkin_3_adapter.rb +0 -297
  52. data/lib/cuke_modeler/adapters/gherkin_4_adapter.rb +0 -309
  53. data/lib/cuke_modeler/adapters/gherkin_5_adapter.rb +0 -12
  54. data/lib/cuke_modeler/adapters/gherkin_6_adapter.rb +0 -310
  55. data/lib/cuke_modeler/adapters/gherkin_7_adapter.rb +0 -307
  56. data/lib/cuke_modeler/adapters/gherkin_8_adapter.rb +0 -12
  57. data/testing/cucumber/step_definitions/action_steps.rb +0 -13
  58. data/testing/cucumber/step_definitions/background_steps.rb +0 -1
  59. data/testing/cucumber/step_definitions/directory_steps.rb +0 -6
  60. data/testing/cucumber/step_definitions/doc_string_steps.rb +0 -1
  61. data/testing/cucumber/step_definitions/feature_file_steps.rb +0 -16
  62. data/testing/cucumber/step_definitions/feature_steps.rb +0 -7
  63. data/testing/cucumber/step_definitions/modeling_steps.rb +0 -49
  64. data/testing/cucumber/step_definitions/setup_steps.rb +0 -32
  65. data/testing/cucumber/step_definitions/step_steps.rb +0 -3
  66. data/testing/cucumber/step_definitions/table_steps.rb +0 -1
  67. data/testing/cucumber/step_definitions/tag_steps.rb +0 -3
  68. data/testing/cucumber/step_definitions/verification_steps.rb +0 -181
  69. data/testing/cucumber/support/env.rb +0 -30
  70. data/testing/dialect_helper.rb +0 -48
  71. data/testing/file_helper.rb +0 -47
  72. data/testing/gemfiles/gherkin2.gemfile +0 -33
  73. data/testing/gemfiles/gherkin3.gemfile +0 -26
  74. data/testing/gemfiles/gherkin4.gemfile +0 -27
  75. data/testing/gemfiles/gherkin5.gemfile +0 -27
  76. data/testing/gemfiles/gherkin6.gemfile +0 -10
  77. data/testing/gemfiles/gherkin7.gemfile +0 -9
  78. data/testing/gemfiles/gherkin8.gemfile +0 -9
  79. data/testing/gemfiles/gherkin9.gemfile +0 -9
  80. data/testing/helper_methods.rb +0 -23
  81. data/testing/rspec/spec/integration/adapters/gherkin_2_adapter_spec.rb +0 -166
  82. data/testing/rspec/spec/integration/adapters/gherkin_3_adapter_spec.rb +0 -166
  83. data/testing/rspec/spec/integration/adapters/gherkin_4_adapter_spec.rb +0 -165
  84. data/testing/rspec/spec/integration/adapters/gherkin_5_adapter_spec.rb +0 -165
  85. data/testing/rspec/spec/integration/adapters/gherkin_6_adapter_spec.rb +0 -159
  86. data/testing/rspec/spec/integration/adapters/gherkin_7_adapter_spec.rb +0 -162
  87. data/testing/rspec/spec/integration/adapters/gherkin_8_adapter_spec.rb +0 -162
  88. data/testing/rspec/spec/integration/adapters/gherkin_9_adapter_spec.rb +0 -162
  89. data/testing/rspec/spec/integration/models/background_integration_spec.rb +0 -438
  90. data/testing/rspec/spec/integration/models/cell_integration_spec.rb +0 -338
  91. data/testing/rspec/spec/integration/models/comment_integration_spec.rb +0 -180
  92. data/testing/rspec/spec/integration/models/directory_integration_spec.rb +0 -218
  93. data/testing/rspec/spec/integration/models/doc_string_integration_spec.rb +0 -398
  94. data/testing/rspec/spec/integration/models/example_integration_spec.rb +0 -753
  95. data/testing/rspec/spec/integration/models/feature_file_integration_spec.rb +0 -276
  96. data/testing/rspec/spec/integration/models/feature_integration_spec.rb +0 -655
  97. data/testing/rspec/spec/integration/models/model_integration_spec.rb +0 -15
  98. data/testing/rspec/spec/integration/models/outline_integration_spec.rb +0 -619
  99. data/testing/rspec/spec/integration/models/row_integration_spec.rb +0 -303
  100. data/testing/rspec/spec/integration/models/scenario_integration_spec.rb +0 -475
  101. data/testing/rspec/spec/integration/models/step_integration_spec.rb +0 -573
  102. data/testing/rspec/spec/integration/models/table_integration_spec.rb +0 -333
  103. data/testing/rspec/spec/integration/models/tag_integration_spec.rb +0 -271
  104. data/testing/rspec/spec/integration/nested_integration_spec.rb +0 -91
  105. data/testing/rspec/spec/integration/parsing_integration_spec.rb +0 -143
  106. data/testing/rspec/spec/integration/shared/models_integration_specs.rb +0 -18
  107. data/testing/rspec/spec/spec_helper.rb +0 -129
  108. data/testing/rspec/spec/unit/cuke_modeler_unit_spec.rb +0 -25
  109. data/testing/rspec/spec/unit/described_unit_spec.rb +0 -23
  110. data/testing/rspec/spec/unit/models/background_unit_spec.rb +0 -83
  111. data/testing/rspec/spec/unit/models/cell_unit_spec.rb +0 -68
  112. data/testing/rspec/spec/unit/models/comment_unit_spec.rb +0 -68
  113. data/testing/rspec/spec/unit/models/directory_unit_spec.rb +0 -127
  114. data/testing/rspec/spec/unit/models/doc_string_unit_spec.rb +0 -100
  115. data/testing/rspec/spec/unit/models/example_unit_spec.rb +0 -133
  116. data/testing/rspec/spec/unit/models/feature_file_unit_spec.rb +0 -125
  117. data/testing/rspec/spec/unit/models/feature_unit_spec.rb +0 -157
  118. data/testing/rspec/spec/unit/models/model_unit_spec.rb +0 -15
  119. data/testing/rspec/spec/unit/models/outline_unit_spec.rb +0 -117
  120. data/testing/rspec/spec/unit/models/row_unit_spec.rb +0 -68
  121. data/testing/rspec/spec/unit/models/scenario_unit_spec.rb +0 -86
  122. data/testing/rspec/spec/unit/models/step_unit_spec.rb +0 -109
  123. data/testing/rspec/spec/unit/models/table_unit_spec.rb +0 -77
  124. data/testing/rspec/spec/unit/models/tag_unit_spec.rb +0 -68
  125. data/testing/rspec/spec/unit/named_unit_spec.rb +0 -23
  126. data/testing/rspec/spec/unit/nested_unit_spec.rb +0 -43
  127. data/testing/rspec/spec/unit/parsed_unit_spec.rb +0 -27
  128. data/testing/rspec/spec/unit/parsing_unit_spec.rb +0 -54
  129. data/testing/rspec/spec/unit/shared/bare_bones_models_unit_specs.rb +0 -14
  130. data/testing/rspec/spec/unit/shared/containing_models_unit_specs.rb +0 -127
  131. data/testing/rspec/spec/unit/shared/described_models_unit_specs.rb +0 -38
  132. data/testing/rspec/spec/unit/shared/keyworded_models_unit_specs.rb +0 -58
  133. data/testing/rspec/spec/unit/shared/models_unit_specs.rb +0 -15
  134. data/testing/rspec/spec/unit/shared/named_models_unit_specs.rb +0 -39
  135. data/testing/rspec/spec/unit/shared/nested_models_unit_specs.rb +0 -51
  136. data/testing/rspec/spec/unit/shared/parsed_models_unit_specs.rb +0 -39
  137. data/testing/rspec/spec/unit/shared/prepopulated_models_unit_specs.rb +0 -18
  138. data/testing/rspec/spec/unit/shared/sourced_models_unit_specs.rb +0 -39
  139. data/testing/rspec/spec/unit/shared/stepped_models_unit_specs.rb +0 -46
  140. data/testing/rspec/spec/unit/shared/stringifiable_models_unit_specs.rb +0 -18
  141. data/testing/rspec/spec/unit/shared/tagged_models_unit_specs.rb +0 -72
  142. data/testing/rspec/spec/unit/sourceable_unit_spec.rb +0 -27
  143. data/testing/rspec/spec/unit/stepped_unit_spec.rb +0 -23
  144. data/testing/rspec/spec/unit/taggable_unit_spec.rb +0 -69
  145. data/testing/test_languages.json +0 -45
  146. data/todo.txt +0 -25
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 1e45af46cbcc095433e4bf40fd868daed964bbbc53dac06d3d3de6ca6a4c2db1
4
- data.tar.gz: 36e8660c3f4e993732113f79979168ac59b1bd33c35e8722fc06abdf4e54ce85
3
+ metadata.gz: ab8068e33b09b4e508b0cd53f77c954052796fbbd33686070cbf6e65cd616378
4
+ data.tar.gz: 57849a93b03a7d6d0cc56548e4e6482eb61557c4b42a68950c82e3b43542e482
5
5
  SHA512:
6
- metadata.gz: 39109727b23ffbed1015bc7d690367a10c8f8a88e12d099ba973fcd722cfc5fd0f15e9c5dfeef4a61793850b75a46feb9933f1bba66845cb7ccd72c323d26c36
7
- data.tar.gz: 11848c5307c7da2a0669e80dfef83cf535ba1259e3c04d05897b12ab5057760419f239ff7eaaebbc37fec33e7610cc11c771f81128d05f7a779794ad70770978
6
+ metadata.gz: 3077bf8a1a1defacc92d371db01044c2ea5c7ef44dc7a194079a1e8721b4e8dad598eb4705acc4882d04a5934c27a70103a7eca2795f45ea9f5bebad37d24e7d
7
+ data.tar.gz: 0a9e2fbae81285bad22f59ef8626fd020be7516a824292fd08387eb38f34d1bc185f41d71084c1be84f331574c7d65145fa3e37ae28d8585ab5369f25adfffa5
@@ -8,6 +8,53 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
8
8
 
9
9
  Nothing yet...
10
10
 
11
+ ## [3.4.0] - 2020-09-02
12
+
13
+ ### Added
14
+ - `Feature#has_background?` and `Rule#has_background?` now both have a more conventional name via the alias `#background?`
15
+
16
+ ## [3.3.0] - 2020-08-15
17
+
18
+ ### Added
19
+ - Support added for more versions of the `cucumber-gherkin` gem
20
+ - 15.x
21
+
22
+ ## [3.2.0] - 2020-07-27
23
+
24
+ ### Added
25
+ - The `Rule` keyword is now a modeled element.
26
+
27
+ ### Deprecated
28
+ - `Feature#test_case_count` will be removed on the next major release. It's a random analysis method in what is
29
+ otherwise a purely abstraction layer library. The [CQL](https://github.com/enkessler/cql) gem is better suited to such tasks.
30
+
31
+ ## [3.1.0] - 2020-06-28
32
+
33
+ ### Added
34
+ - Support added for more versions of the `cucumber-gherkin` gem
35
+ - 14.x
36
+
37
+ ### Fixed
38
+ - Text is converted to UTF-8 encoding before being passed to the underlying Gherkin gem. This is due to UTF-8 being
39
+ the only encoding supported by Gherkin. The `gherkin` gem did the conversion automatically and so this conversion
40
+ was not necessary previously but the `cucumber-gherkin` gem does not do any automatic conversion.
41
+
42
+ ## [3.0.0] - 2020-06-08
43
+
44
+ ### Changed
45
+ - This gem now wraps the `cucumber-gherkin` gem instead of the `gherkin` gem, now that `cucumber-gherkin` has superseded `gherkin`.
46
+ - Support for versions of Ruby earlier than 2.3 has been dropped due to that being the minimum required version of Ruby required by the `cucumber-gherkin` gem.
47
+ - When using the parsing functionality provided by this gem, the standardized AST returned when parsing Gherkin text is now returned directly as a Hash instead of also being wrapped in an array. The array was an artifact of basing the AST on the earliest versions of `gherkin` that were supported.
48
+ - No longer including every file in the Git repository as part of the gem. Only the files needed for using the gem (and the informative ones like the README) will be packaged into the released gem.
49
+
50
+ ### Added
51
+ - Support added for more versions of the `cucumber-gherkin` gem
52
+ - 13.x
53
+ - 12.x
54
+ - 11.x
55
+ - 10.x
56
+ - 9.x
57
+
11
58
  ## [2.1.0] - 2020-05-27
12
59
 
13
60
  ### Added
@@ -287,7 +334,12 @@ Nothing yet...
287
334
  - Initial release
288
335
 
289
336
 
290
- [Unreleased]: https://github.com/enkessler/cuke_modeler/compare/v2.1.0...HEAD
337
+ [Unreleased]: https://github.com/enkessler/cuke_modeler/compare/v3.4.0...HEAD
338
+ [3.4.0]: https://github.com/enkessler/cuke_modeler/compare/v3.3.0...v3.4.0
339
+ [3.3.0]: https://github.com/enkessler/cuke_modeler/compare/v3.2.0...v3.3.0
340
+ [3.2.0]: https://github.com/enkessler/cuke_modeler/compare/v3.1.0...v3.2.0
341
+ [3.1.0]: https://github.com/enkessler/cuke_modeler/compare/v3.0.0...v3.1.0
342
+ [3.0.0]: https://github.com/enkessler/cuke_modeler/compare/v2.1.0...v3.0.0
291
343
  [2.1.0]: https://github.com/enkessler/cuke_modeler/compare/v2.0.0...v2.1.0
292
344
  [2.0.0]: https://github.com/enkessler/cuke_modeler/compare/v1.5.1...v2.0.0
293
345
  [1.5.1]: https://github.com/enkessler/cuke_modeler/compare/v1.5.0...v1.5.1
data/README.md CHANGED
@@ -8,11 +8,11 @@ User stuff:
8
8
  [![Yard Docs](http://img.shields.io/badge/Documentation-API-blue.svg)](https://www.rubydoc.info/gems/cuke_modeler)
9
9
 
10
10
  Developer stuff:
11
- [![Build Status](https://travis-ci.org/enkessler/cuke_modeler.svg?branch=dev)](https://travis-ci.org/enkessler/cuke_modeler)
12
- [![Build status](https://ci.appveyor.com/api/projects/status/is8xqvoqn3pjh9l0/branch/dev?svg=true)](https://ci.appveyor.com/project/enkessler/cuke-modeler/branch/dev)
13
- [![Coverage Status](https://coveralls.io/repos/github/enkessler/cuke_modeler/badge.svg?branch=dev)](https://coveralls.io/github/enkessler/cuke_modeler?branch=dev)
11
+ [![Build Status](https://travis-ci.org/enkessler/cuke_modeler.svg)](https://travis-ci.org/enkessler/cuke_modeler)
12
+ [![Build status](https://ci.appveyor.com/api/projects/status/is8xqvoqn3pjh9l0?svg=true)](https://ci.appveyor.com/project/enkessler/cuke-modeler)
13
+ [![Coverage Status](https://coveralls.io/repos/github/enkessler/cuke_modeler/badge.svg)](https://coveralls.io/github/enkessler/cuke_modeler)
14
14
  [![Maintainability](https://api.codeclimate.com/v1/badges/83986d8f7a918fed9707/maintainability)](https://codeclimate.com/github/enkessler/cuke_modeler/maintainability)
15
- [![Inline docs](http://inch-ci.org/github/enkessler/cuke_modeler.svg?branch=dev)](https://inch-ci.org/github/enkessler/cuke_modeler?branch=dev)
15
+ [![Inline docs](http://inch-ci.org/github/enkessler/cuke_modeler.svg)](https://inch-ci.org/github/enkessler/cuke_modeler)
16
16
 
17
17
  ---
18
18
 
@@ -21,7 +21,7 @@ Developer stuff:
21
21
  There comes a time in every programmer's adventures with Cucumber when they
22
22
  want to do Really Cool Stuff with their tests. This usually necessitates
23
23
  scanning all of their feature files and playing with the output. While the
24
- **[gherkin](https://github.com/cucumber/gherkin)** gem does a fine job of parsing feature files, reading or even manipulating
24
+ **[cucumber-gherkin](https://github.com/cucumber/cucumber/tree/master/gherkin)** gem does a fine job of parsing feature files, reading or even manipulating
25
25
  the resulting Abstract Syntax Tree is not always fun. **cuke_modeler** comes to
26
26
  the rescue by providing a modeling layer that is easier to work with.
27
27
 
@@ -82,13 +82,14 @@ and setting their attributes afterward.
82
82
  One could, if so inclined, use this method to dynamically edit or even create
83
83
  an entire test suite!
84
84
 
85
- For more information on the different models and how to use them, see the
85
+ For more information on the different models (which more or less have the same relation
86
+ to each other as described in the AST [here](https://github.com/cucumber/cucumber/tree/master/gherkin#ast)) and how to use them, see the
86
87
  [documentation](https://app.cucumber.pro/projects/cuke_modeler).
87
88
 
88
89
  ## Modeling dialects other than English
89
90
 
90
91
  The modeling functionality provided by this gem will work with any dialect that
91
- is supported by the **gherkin** gem. For modeling at the feature level or higher,
92
+ is supported by the **cucumber-gherkin** gem. For modeling at the feature level or higher,
92
93
  no additional effort is needed because the `# language` header at the top of a
93
94
  feature already indicates that a non-default dialect is being used.
94
95
 
@@ -126,13 +127,10 @@ that feature is ultimately run with SpecFlow (Cucumber for C#), Lettuce
126
127
  * [cuke_linter](https://github.com/enkessler/cuke_linter)
127
128
 
128
129
 
129
- ## Contributing
130
+ ## Development and Contributing
130
131
 
131
- 1. Fork it
132
- 2. Create your feature branch (**off of the development branch**)
133
- `git checkout -b my-new-feature`
134
- 3. Commit your changes
135
- `git commit -am 'Add some feature'`
136
- 4. Push to the branch
137
- `git push origin my-new-feature`
138
- 5. Create new Pull Request
132
+ See [CONTRIBUTING.md](https://github.com/enkessler/cuke_modeler/blob/master/CONTRIBUTING.md)
133
+
134
+ ## License
135
+
136
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
@@ -1,35 +1,47 @@
1
- # coding: utf-8
2
- lib = File.expand_path('../lib', __FILE__)
1
+ lib = File.expand_path('lib', __dir__)
3
2
  $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
3
  require 'cuke_modeler/version'
5
4
 
6
5
  Gem::Specification.new do |spec|
7
- spec.name = "cuke_modeler"
6
+ spec.name = 'cuke_modeler'
8
7
  spec.version = CukeModeler::VERSION
9
- spec.authors = ["Eric Kessler"]
10
- spec.email = ["morrow748@gmail.com"]
11
- spec.summary = %q{A gem providing functionality to model Gherkin based test suites.}
12
- spec.description = %q{This gem facilitates modeling a test suite that is written in Gherkin (e.g. Cucumber, SpecFlow, Lettuce, etc.). It does this by providing an abstraction layer on top of the Abstract Syntax Tree that the 'gherkin' gem generates when parsing features, as well as providing models for feature files and directories in order to be able to have a fully traversable model tree of a test suite's structure. These models can then be analyzed or manipulated more easily than the underlying AST layer.}
8
+ spec.authors = ['Eric Kessler']
9
+ spec.email = ['morrow748@gmail.com']
10
+ spec.summary = 'A gem providing functionality to model Gherkin based test suites.'
11
+ spec.description = ['This gem facilitates modeling a test suite that is written in Gherkin (e.g. Cucumber, ',
12
+ 'SpecFlow, Lettuce, etc.). It does this by providing an abstraction layer on top of the ',
13
+ "Abstract Syntax Tree that the 'cucumber-gherkin' gem generates when parsing features, ",
14
+ 'as well as providing models for feature files and directories in order to be able to ',
15
+ "have a fully traversable model tree of a test suite's structure. These models can then ",
16
+ 'be analyzed or manipulated more easily than the underlying AST layer.'].join("\n")
13
17
  spec.homepage = 'https://github.com/enkessler/cuke_modeler'
14
- spec.license = "MIT"
18
+ spec.license = 'MIT'
15
19
 
16
- spec.files = `git ls-files -z`.split("\x0")
17
- spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
18
- spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
19
- spec.require_paths = ["lib"]
20
+ # Specify which files should be added to the gem when it is released.
21
+ # The `git ls-files -z` loads the files in the RubyGem that have been added into git.
22
+ spec.files = Dir.chdir(File.expand_path('', __dir__)) do
23
+ source_controlled_files = `git ls-files -z`.split("\x0")
24
+ source_controlled_files.keep_if { |file| file =~ %r{^(lib|testing/cucumber/features)} }
25
+ source_controlled_files + ['README.md', 'LICENSE.txt', 'CHANGELOG.md', 'cuke_modeler.gemspec']
26
+ end
27
+ spec.require_paths = ['lib']
20
28
 
21
- spec.required_ruby_version = '>= 1.8.7', '< 3.0'
29
+ spec.required_ruby_version = '>= 2.3', '< 3.0'
22
30
 
23
- spec.add_runtime_dependency 'gherkin', '< 10.0'
24
- spec.add_runtime_dependency('json', '>= 1.0', '< 3.0')
25
- spec.add_runtime_dependency('multi_json', '~> 1.0')
31
+ spec.add_runtime_dependency 'cucumber-gherkin', '< 16.0'
26
32
 
27
33
  spec.add_development_dependency 'bundler', '< 3.0'
28
- spec.add_development_dependency "rake", '< 13.0.0'
29
- spec.add_development_dependency 'cucumber', '< 5.0.0'
30
- spec.add_development_dependency 'rspec', '~> 3.0'
31
- spec.add_development_dependency 'simplecov', '< 1.0.0'
32
- spec.add_development_dependency 'racatt', '~> 1.0'
33
34
  spec.add_development_dependency 'coveralls', '< 1.0.0'
35
+ # Cucumber 4.x is the earliest version to use cucumber-gherkin
36
+ spec.add_development_dependency 'cucumber', '>= 4.0.0', '< 6.0.0'
37
+ spec.add_development_dependency 'racatt', '~> 1.0'
34
38
  spec.add_development_dependency 'rainbow', '< 4.0.0'
39
+ spec.add_development_dependency 'rake', '< 14.0.0'
40
+ spec.add_development_dependency 'rspec', '~> 3.0'
41
+ # RuboCop drops Ruby 2.3 support after this version and we need to maintain Ruby 2.3 compatibility when writing code
42
+ # for this gem
43
+ spec.add_development_dependency 'rubocop', '< 0.82.0'
44
+ # Coveralls gem does not support any newer version than this
45
+ spec.add_development_dependency 'simplecov', '<= 0.16.1'
46
+ spec.add_development_dependency 'test-unit', '< 4.0.0'
35
47
  end
@@ -4,7 +4,7 @@ module CukeModeler
4
4
  end
5
5
 
6
6
 
7
- require "cuke_modeler/version"
7
+ require 'cuke_modeler/version'
8
8
 
9
9
  require 'cuke_modeler/parsing'
10
10
  require 'cuke_modeler/containing'
@@ -19,6 +19,7 @@ require 'cuke_modeler/models/model'
19
19
  require 'cuke_modeler/models/feature_file'
20
20
  require 'cuke_modeler/models/directory'
21
21
  require 'cuke_modeler/models/feature'
22
+ require 'cuke_modeler/models/rule'
22
23
  require 'cuke_modeler/models/background'
23
24
  require 'cuke_modeler/models/scenario'
24
25
  require 'cuke_modeler/models/outline'
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 10.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin10Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 11.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin11Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 12.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin12Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 13.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin13Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 14.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin14Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ require_relative 'gherkin_9_adapter'
2
+
3
+
4
+ module CukeModeler
5
+
6
+ # NOT A PART OF THE PUBLIC API
7
+ # An adapter that can convert the output of version 15.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
9
+
10
+ class Gherkin15Adapter < Gherkin9Adapter
11
+
12
+ end
13
+ end
@@ -1,12 +1,364 @@
1
- require_relative 'gherkin_7_adapter'
1
+ # Some things just aren't going to get better due to the inherent complexity of the AST
2
+ # rubocop:disable Metrics/ClassLength, Metrics/AbcSize, Metrics/MethodLength
2
3
 
3
4
  module CukeModeler
4
5
 
5
6
  # NOT A PART OF THE PUBLIC API
6
- # An adapter that can convert the output of version 9.x of the *gherkin* gem into input that is consumable by this gem.
7
+ # An adapter that can convert the output of version 9.x of the *cucumber-gherkin* gem into input that is consumable
8
+ # by this gem.
7
9
 
8
- class Gherkin9Adapter < Gherkin7Adapter
10
+ class Gherkin9Adapter
9
11
 
10
- end
12
+ # Adapts the given AST into the shape that this gem expects
13
+ def adapt(ast)
14
+ adapted_ast = {}
15
+
16
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
17
+ save_original_data(adapted_ast, ast)
18
+ clear_child_elements(adapted_ast, [[:feature],
19
+ [:comments]])
20
+
21
+ adapted_ast['comments'] = adapt_comments(ast)
22
+ adapted_ast['feature'] = adapt_feature(ast[:feature])
23
+
24
+ adapted_ast
25
+ end
26
+
27
+ # Adapts the AST sub-tree that is rooted at the given feature node.
28
+ def adapt_feature(feature_ast)
29
+ return nil unless feature_ast
30
+
31
+ adapted_feature = {}
32
+
33
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
34
+ save_original_data(adapted_feature, feature_ast)
35
+ clear_child_elements(adapted_feature, [[:tags],
36
+ [:children]])
37
+
38
+ adapted_feature['keyword'] = feature_ast[:keyword]
39
+ adapted_feature['name'] = feature_ast[:name]
40
+ adapted_feature['description'] = feature_ast[:description] || ''
41
+ adapted_feature['line'] = feature_ast[:location][:line]
42
+
43
+ adapted_feature['elements'] = adapt_child_elements(feature_ast)
44
+ adapted_feature['tags'] = adapt_tags(feature_ast)
45
+
46
+ adapted_feature
47
+ end
48
+
49
+ # Adapts the AST sub-tree that is rooted at the given background node.
50
+ def adapt_background(background_ast)
51
+ adapted_background = {}
52
+
53
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
54
+ save_original_data(adapted_background, background_ast)
55
+ clear_child_elements(adapted_background, [[:background, :steps]])
56
+
57
+ adapted_background['type'] = 'Background'
58
+ adapted_background['keyword'] = background_ast[:background][:keyword]
59
+ adapted_background['name'] = background_ast[:background][:name]
60
+ adapted_background['description'] = background_ast[:background][:description] || ''
61
+ adapted_background['line'] = background_ast[:background][:location][:line]
62
+
63
+ adapted_background['steps'] = adapt_steps(background_ast[:background])
64
+
65
+ adapted_background
66
+ end
67
+
68
+ # Adapts the AST sub-tree that is rooted at the given rule node.
69
+ def adapt_rule(rule_ast)
70
+ adapted_rule = {}
71
+
72
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
73
+ save_original_data(adapted_rule, rule_ast)
74
+ clear_child_elements(adapted_rule, [[:rule, :children]])
75
+
76
+ adapted_rule['type'] = 'Rule'
77
+ adapted_rule['keyword'] = rule_ast[:rule][:keyword]
78
+ adapted_rule['name'] = rule_ast[:rule][:name]
79
+ adapted_rule['description'] = rule_ast[:rule][:description] || ''
80
+ adapted_rule['line'] = rule_ast[:rule][:location][:line]
81
+
82
+ adapted_rule['elements'] = adapt_child_elements(rule_ast[:rule])
83
+
84
+ adapted_rule
85
+ end
86
+
87
+ # Adapts the AST sub-tree that is rooted at the given scenario node.
88
+ def adapt_scenario(test_ast)
89
+ adapted_scenario = {}
90
+
91
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
92
+ save_original_data(adapted_scenario, test_ast)
93
+ clear_child_elements(adapted_scenario, [[:scenario, :tags],
94
+ [:scenario, :steps]])
95
+
96
+ adapted_scenario['type'] = 'Scenario'
97
+ adapted_scenario['keyword'] = test_ast[:scenario][:keyword]
98
+ adapted_scenario['name'] = test_ast[:scenario][:name]
99
+ adapted_scenario['description'] = test_ast[:scenario][:description] || ''
100
+ adapted_scenario['line'] = test_ast[:scenario][:location][:line]
101
+
102
+ adapted_scenario['tags'] = adapt_tags(test_ast[:scenario])
103
+ adapted_scenario['steps'] = adapt_steps(test_ast[:scenario])
104
+
105
+ adapted_scenario
106
+ end
107
+
108
+ # Adapts the AST sub-tree that is rooted at the given outline node.
109
+ def adapt_outline(test_ast)
110
+ adapted_outline = {}
111
+
112
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
113
+ save_original_data(adapted_outline, test_ast)
114
+ clear_child_elements(adapted_outline, [[:scenario, :tags],
115
+ [:scenario, :steps],
116
+ [:scenario, :examples]])
117
+
118
+ adapted_outline['type'] = 'ScenarioOutline'
119
+ adapted_outline['keyword'] = test_ast[:scenario][:keyword]
120
+ adapted_outline['name'] = test_ast[:scenario][:name]
121
+ adapted_outline['description'] = test_ast[:scenario][:description] || ''
122
+ adapted_outline['line'] = test_ast[:scenario][:location][:line]
123
+
124
+ adapted_outline['tags'] = adapt_tags(test_ast[:scenario])
125
+ adapted_outline['steps'] = adapt_steps(test_ast[:scenario])
126
+ adapted_outline['examples'] = adapt_examples(test_ast[:scenario])
127
+
128
+ adapted_outline
129
+ end
130
+
131
+ # Adapts the AST sub-tree that is rooted at the given example node.
132
+ def adapt_example(example_ast)
133
+ adapted_example = {}
134
+
135
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
136
+ save_original_data(adapted_example, example_ast)
137
+ clear_child_elements(adapted_example, [[:tags],
138
+ [:table_header],
139
+ [:table_body]])
140
+
141
+ adapted_example['keyword'] = example_ast[:keyword]
142
+ adapted_example['name'] = example_ast[:name]
143
+ adapted_example['line'] = example_ast[:location][:line]
144
+ adapted_example['description'] = example_ast[:description] || ''
145
+
146
+ adapted_example['rows'] = []
147
+ adapted_example['rows'] << adapt_table_row(example_ast[:table_header]) if example_ast[:table_header]
148
+
149
+ example_ast[:table_body]&.each do |row|
150
+ adapted_example['rows'] << adapt_table_row(row)
151
+ end
152
+
153
+ adapted_example['tags'] = adapt_tags(example_ast)
154
+
155
+ adapted_example
156
+ end
157
+
158
+ # Adapts the AST sub-tree that is rooted at the given tag node.
159
+ def adapt_tag(tag_ast)
160
+ adapted_tag = {}
161
+
162
+ # Saving off the original data
163
+ save_original_data(adapted_tag, tag_ast)
164
+
165
+ adapted_tag['name'] = tag_ast[:name]
166
+ adapted_tag['line'] = tag_ast[:location][:line]
167
+
168
+ adapted_tag
169
+ end
170
+
171
+ # Adapts the AST sub-tree that is rooted at the given comment node.
172
+ def adapt_comment(comment_ast)
173
+ adapted_comment = {}
174
+
175
+ # Saving off the original data
176
+ save_original_data(adapted_comment, comment_ast)
177
+
178
+ adapted_comment['text'] = comment_ast[:text]
179
+ adapted_comment['line'] = comment_ast[:location][:line]
180
+
181
+ adapted_comment
182
+ end
183
+
184
+ # Adapts the AST sub-tree that is rooted at the given step node.
185
+ def adapt_step(step_ast)
186
+ adapted_step = {}
187
+
188
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
189
+ save_original_data(adapted_step, step_ast)
190
+ clear_child_elements(adapted_step, [[:data_table],
191
+ [:doc_string]])
11
192
 
193
+ adapted_step['keyword'] = step_ast[:keyword]
194
+ adapted_step['name'] = step_ast[:text]
195
+ adapted_step['line'] = step_ast[:location][:line]
196
+
197
+ if step_ast[:doc_string]
198
+ adapted_step['doc_string'] = adapt_doc_string(step_ast[:doc_string])
199
+ elsif step_ast[:data_table]
200
+ adapted_step['table'] = adapt_step_table(step_ast[:data_table])
201
+ end
202
+
203
+ adapted_step
204
+ end
205
+
206
+ # Adapts the AST sub-tree that is rooted at the given doc string node.
207
+ def adapt_doc_string(doc_string_ast)
208
+ adapted_doc_string = {}
209
+
210
+ # Saving off the original data
211
+ save_original_data(adapted_doc_string, doc_string_ast)
212
+
213
+ adapted_doc_string['value'] = doc_string_ast[:content]
214
+ adapted_doc_string['content_type'] = doc_string_ast[:media_type]
215
+ adapted_doc_string['line'] = doc_string_ast[:location][:line]
216
+
217
+ adapted_doc_string
218
+ end
219
+
220
+ # Adapts the AST sub-tree that is rooted at the given table node.
221
+ def adapt_step_table(step_table_ast)
222
+ adapted_step_table = {}
223
+
224
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
225
+ save_original_data(adapted_step_table, step_table_ast)
226
+ clear_child_elements(adapted_step_table, [[:rows]])
227
+
228
+ adapted_step_table['rows'] = []
229
+ step_table_ast[:rows].each do |row|
230
+ adapted_step_table['rows'] << adapt_table_row(row)
231
+ end
232
+ adapted_step_table['line'] = step_table_ast[:location][:line]
233
+
234
+ adapted_step_table
235
+ end
236
+
237
+ # Adapts the AST sub-tree that is rooted at the given row node.
238
+ def adapt_table_row(table_row_ast)
239
+ adapted_table_row = {}
240
+
241
+ # Saving off the original data and removing parsed data for child elements in order to avoid duplicating data
242
+ save_original_data(adapted_table_row, table_row_ast)
243
+ clear_child_elements(adapted_table_row, [[:cells]])
244
+
245
+ adapted_table_row['line'] = table_row_ast[:location][:line]
246
+
247
+ adapted_table_row['cells'] = []
248
+ table_row_ast[:cells].each do |row|
249
+ adapted_table_row['cells'] << adapt_table_cell(row)
250
+ end
251
+
252
+ adapted_table_row
253
+ end
254
+
255
+ # Adapts the AST sub-tree that is rooted at the given cell node.
256
+ def adapt_table_cell(cell_ast)
257
+ adapted_cell = {}
258
+
259
+ # Saving off the original data
260
+ save_original_data(adapted_cell, cell_ast)
261
+
262
+ adapted_cell['value'] = cell_ast[:value]
263
+ adapted_cell['line'] = cell_ast[:location][:line]
264
+
265
+ adapted_cell
266
+ end
267
+
268
+
269
+ private
270
+
271
+
272
+ def adapt_comments(file_ast)
273
+ return [] unless file_ast[:comments]
274
+
275
+ file_ast[:comments].map { |comment| adapt_comment(comment) }
276
+ end
277
+
278
+ def adapt_tags(element_ast)
279
+ return [] unless element_ast[:tags]
280
+
281
+ element_ast[:tags].map { |tag| adapt_tag(tag) }
282
+ end
283
+
284
+ def adapt_steps(element_ast)
285
+ return [] unless element_ast[:steps]
286
+
287
+ element_ast[:steps].map { |step| adapt_step(step) }
288
+ end
289
+
290
+ def adapt_examples(element_ast)
291
+ return [] unless element_ast[:examples]
292
+
293
+ element_ast[:examples].map { |example| adapt_example(example) }
294
+ end
295
+
296
+ def adapt_child_elements(element_ast)
297
+ return [] unless element_ast[:children]
298
+
299
+ adapted_children = []
300
+
301
+ element_ast[:children].each do |child_element|
302
+ adapted_children << if child_element[:background]
303
+ adapt_background(child_element)
304
+ elsif child_element[:rule]
305
+ adapt_rule(child_element)
306
+ else
307
+ adapt_test(child_element)
308
+ end
309
+ end
310
+
311
+ adapted_children
312
+ end
313
+
314
+ def adapt_test(test_ast)
315
+ if (test_node?(test_ast) && test_has_examples?(test_ast)) ||
316
+ (test_node?(test_ast) && test_uses_outline_keyword?(test_ast))
317
+
318
+ adapt_outline(test_ast)
319
+ elsif test_node?(test_ast)
320
+ adapt_scenario(test_ast)
321
+ else
322
+ raise(ArgumentError, "Unknown test type with keys: #{test_ast.keys}")
323
+ end
324
+ end
325
+
326
+ def save_original_data(adapted_ast, raw_ast)
327
+ adapted_ast['cuke_modeler_parsing_data'] = Marshal.load(Marshal.dump(raw_ast))
328
+ end
329
+
330
+ def clear_child_elements(ast, child_paths)
331
+ child_paths.each do |traversal_path|
332
+ if ast['cuke_modeler_parsing_data'].dig(*traversal_path)
333
+ bury(ast['cuke_modeler_parsing_data'], traversal_path, nil)
334
+ end
335
+ end
336
+ end
337
+
338
+ def bury(hash, traversal_path, value)
339
+ keys = *traversal_path
340
+
341
+ current = hash
342
+ (keys.count - 1).times do |index|
343
+ current = hash[keys[index]]
344
+ end
345
+
346
+ current[keys.last] = value
347
+ end
348
+
349
+ def test_node?(ast_node)
350
+ ast_node[:scenario]
351
+ end
352
+
353
+ def test_has_examples?(ast_node)
354
+ ast_node[:scenario][:examples]
355
+ end
356
+
357
+ def test_uses_outline_keyword?(test_ast)
358
+ Parsing.dialects[Parsing.dialect]['scenarioOutline'].include?(test_ast[:scenario][:keyword])
359
+ end
360
+
361
+ end
12
362
  end
363
+
364
+ # rubocop:enable Metrics/ClassLength, Metrics/AbcSize, Metrics/MethodLength