gherkin 2.12.2 → 3.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (333) hide show
  1. checksums.yaml +4 -4
  2. data/.travis.yml +2 -15
  3. data/CONTRIBUTING.md +23 -0
  4. data/Gemfile +0 -3
  5. data/LICENSE +18 -17
  6. data/Makefile +65 -0
  7. data/README.md +9 -268
  8. data/Rakefile +17 -19
  9. data/bin/gherkin-generate-ast +21 -0
  10. data/bin/gherkin-generate-pickles +25 -0
  11. data/bin/gherkin-generate-tokens +12 -0
  12. data/gherkin-ruby.razor +218 -0
  13. data/gherkin.gemspec +22 -76
  14. data/lib/gherkin/ast_builder.rb +243 -0
  15. data/lib/gherkin/ast_node.rb +30 -0
  16. data/lib/gherkin/dialect.rb +58 -0
  17. data/lib/gherkin/errors.rb +45 -0
  18. data/lib/gherkin/gherkin-languages.json +2968 -0
  19. data/lib/gherkin/gherkin_line.rb +95 -0
  20. data/lib/gherkin/parser.rb +1910 -0
  21. data/lib/gherkin/pickles/compiler.rb +164 -0
  22. data/lib/gherkin/token.rb +18 -0
  23. data/lib/gherkin/token_formatter_builder.rb +39 -0
  24. data/lib/gherkin/token_matcher.rb +169 -0
  25. data/lib/gherkin/token_scanner.rb +40 -0
  26. data/spec/capture_warnings.rb +68 -0
  27. data/spec/coverage.rb +10 -0
  28. data/spec/gherkin/parser_spec.rb +146 -0
  29. metadata +60 -567
  30. data/.gitattributes +0 -2
  31. data/.mailmap +0 -2
  32. data/.rbenv-gemsets +0 -1
  33. data/.rspec +0 -1
  34. data/.ruby-gemset +0 -1
  35. data/.ruby-version +0 -1
  36. data/.yardopts +0 -5
  37. data/History.md +0 -881
  38. data/build_native_gems.sh +0 -7
  39. data/cucumber.yml +0 -5
  40. data/examples/parse_and_output_json.rb +0 -19
  41. data/ext/gherkin_lexer_ar/extconf.rb +0 -6
  42. data/ext/gherkin_lexer_ar/gherkin_lexer_ar.c +0 -1432
  43. data/ext/gherkin_lexer_bg/extconf.rb +0 -6
  44. data/ext/gherkin_lexer_bg/gherkin_lexer_bg.c +0 -1655
  45. data/ext/gherkin_lexer_bm/extconf.rb +0 -6
  46. data/ext/gherkin_lexer_bm/gherkin_lexer_bm.c +0 -1516
  47. data/ext/gherkin_lexer_ca/extconf.rb +0 -6
  48. data/ext/gherkin_lexer_ca/gherkin_lexer_ca.c +0 -1581
  49. data/ext/gherkin_lexer_cs/extconf.rb +0 -6
  50. data/ext/gherkin_lexer_cs/gherkin_lexer_cs.c +0 -1515
  51. data/ext/gherkin_lexer_cy_gb/extconf.rb +0 -6
  52. data/ext/gherkin_lexer_cy_gb/gherkin_lexer_cy_gb.c +0 -1282
  53. data/ext/gherkin_lexer_da/extconf.rb +0 -6
  54. data/ext/gherkin_lexer_da/gherkin_lexer_da.c +0 -1298
  55. data/ext/gherkin_lexer_de/extconf.rb +0 -6
  56. data/ext/gherkin_lexer_de/gherkin_lexer_de.c +0 -1425
  57. data/ext/gherkin_lexer_el/extconf.rb +0 -6
  58. data/ext/gherkin_lexer_el/gherkin_lexer_el.c +0 -1919
  59. data/ext/gherkin_lexer_en/extconf.rb +0 -6
  60. data/ext/gherkin_lexer_en/gherkin_lexer_en.c +0 -1418
  61. data/ext/gherkin_lexer_en_au/extconf.rb +0 -6
  62. data/ext/gherkin_lexer_en_au/gherkin_lexer_en_au.c +0 -1765
  63. data/ext/gherkin_lexer_en_lol/extconf.rb +0 -6
  64. data/ext/gherkin_lexer_en_lol/gherkin_lexer_en_lol.c +0 -1177
  65. data/ext/gherkin_lexer_en_old/extconf.rb +0 -6
  66. data/ext/gherkin_lexer_en_old/gherkin_lexer_en_old.c +0 -1309
  67. data/ext/gherkin_lexer_en_pirate/extconf.rb +0 -6
  68. data/ext/gherkin_lexer_en_pirate/gherkin_lexer_en_pirate.c +0 -1471
  69. data/ext/gherkin_lexer_en_scouse/extconf.rb +0 -6
  70. data/ext/gherkin_lexer_en_scouse/gherkin_lexer_en_scouse.c +0 -1634
  71. data/ext/gherkin_lexer_en_tx/extconf.rb +0 -6
  72. data/ext/gherkin_lexer_en_tx/gherkin_lexer_en_tx.c +0 -1265
  73. data/ext/gherkin_lexer_eo/extconf.rb +0 -6
  74. data/ext/gherkin_lexer_eo/gherkin_lexer_eo.c +0 -1241
  75. data/ext/gherkin_lexer_es/extconf.rb +0 -6
  76. data/ext/gherkin_lexer_es/gherkin_lexer_es.c +0 -1399
  77. data/ext/gherkin_lexer_et/extconf.rb +0 -6
  78. data/ext/gherkin_lexer_et/gherkin_lexer_et.c +0 -1236
  79. data/ext/gherkin_lexer_fa/extconf.rb +0 -6
  80. data/ext/gherkin_lexer_fa/gherkin_lexer_fa.c +0 -1475
  81. data/ext/gherkin_lexer_fi/extconf.rb +0 -6
  82. data/ext/gherkin_lexer_fi/gherkin_lexer_fi.c +0 -1215
  83. data/ext/gherkin_lexer_fr/extconf.rb +0 -6
  84. data/ext/gherkin_lexer_fr/gherkin_lexer_fr.c +0 -1493
  85. data/ext/gherkin_lexer_gl/extconf.rb +0 -6
  86. data/ext/gherkin_lexer_gl/gherkin_lexer_gl.c +0 -1374
  87. data/ext/gherkin_lexer_he/extconf.rb +0 -6
  88. data/ext/gherkin_lexer_he/gherkin_lexer_he.c +0 -1374
  89. data/ext/gherkin_lexer_hi/extconf.rb +0 -6
  90. data/ext/gherkin_lexer_hi/gherkin_lexer_hi.c +0 -1848
  91. data/ext/gherkin_lexer_hr/extconf.rb +0 -6
  92. data/ext/gherkin_lexer_hr/gherkin_lexer_hr.c +0 -1323
  93. data/ext/gherkin_lexer_hu/extconf.rb +0 -6
  94. data/ext/gherkin_lexer_hu/gherkin_lexer_hu.c +0 -1376
  95. data/ext/gherkin_lexer_id/extconf.rb +0 -6
  96. data/ext/gherkin_lexer_id/gherkin_lexer_id.c +0 -1208
  97. data/ext/gherkin_lexer_is/extconf.rb +0 -6
  98. data/ext/gherkin_lexer_is/gherkin_lexer_is.c +0 -1376
  99. data/ext/gherkin_lexer_it/extconf.rb +0 -6
  100. data/ext/gherkin_lexer_it/gherkin_lexer_it.c +0 -1340
  101. data/ext/gherkin_lexer_ja/extconf.rb +0 -6
  102. data/ext/gherkin_lexer_ja/gherkin_lexer_ja.c +0 -1699
  103. data/ext/gherkin_lexer_kn/extconf.rb +0 -6
  104. data/ext/gherkin_lexer_kn/gherkin_lexer_kn.c +0 -1965
  105. data/ext/gherkin_lexer_ko/extconf.rb +0 -6
  106. data/ext/gherkin_lexer_ko/gherkin_lexer_ko.c +0 -1360
  107. data/ext/gherkin_lexer_lt/extconf.rb +0 -6
  108. data/ext/gherkin_lexer_lt/gherkin_lexer_lt.c +0 -1296
  109. data/ext/gherkin_lexer_lu/extconf.rb +0 -6
  110. data/ext/gherkin_lexer_lu/gherkin_lexer_lu.c +0 -1389
  111. data/ext/gherkin_lexer_lv/extconf.rb +0 -6
  112. data/ext/gherkin_lexer_lv/gherkin_lexer_lv.c +0 -1425
  113. data/ext/gherkin_lexer_nl/extconf.rb +0 -6
  114. data/ext/gherkin_lexer_nl/gherkin_lexer_nl.c +0 -1368
  115. data/ext/gherkin_lexer_no/extconf.rb +0 -6
  116. data/ext/gherkin_lexer_no/gherkin_lexer_no.c +0 -1311
  117. data/ext/gherkin_lexer_pa/extconf.rb +0 -6
  118. data/ext/gherkin_lexer_pa/gherkin_lexer_pa.c +0 -2160
  119. data/ext/gherkin_lexer_pl/extconf.rb +0 -6
  120. data/ext/gherkin_lexer_pl/gherkin_lexer_pl.c +0 -1738
  121. data/ext/gherkin_lexer_pt/extconf.rb +0 -6
  122. data/ext/gherkin_lexer_pt/gherkin_lexer_pt.c +0 -1711
  123. data/ext/gherkin_lexer_ro/extconf.rb +0 -6
  124. data/ext/gherkin_lexer_ro/gherkin_lexer_ro.c +0 -1427
  125. data/ext/gherkin_lexer_ru/extconf.rb +0 -6
  126. data/ext/gherkin_lexer_ru/gherkin_lexer_ru.c +0 -2071
  127. data/ext/gherkin_lexer_sk/extconf.rb +0 -6
  128. data/ext/gherkin_lexer_sk/gherkin_lexer_sk.c +0 -1739
  129. data/ext/gherkin_lexer_sr_cyrl/extconf.rb +0 -6
  130. data/ext/gherkin_lexer_sr_cyrl/gherkin_lexer_sr_cyrl.c +0 -2112
  131. data/ext/gherkin_lexer_sr_latn/extconf.rb +0 -6
  132. data/ext/gherkin_lexer_sr_latn/gherkin_lexer_sr_latn.c +0 -1567
  133. data/ext/gherkin_lexer_sv/extconf.rb +0 -6
  134. data/ext/gherkin_lexer_sv/gherkin_lexer_sv.c +0 -1324
  135. data/ext/gherkin_lexer_th/extconf.rb +0 -6
  136. data/ext/gherkin_lexer_th/gherkin_lexer_th.c +0 -2840
  137. data/ext/gherkin_lexer_tl/extconf.rb +0 -6
  138. data/ext/gherkin_lexer_tl/gherkin_lexer_tl.c +0 -2021
  139. data/ext/gherkin_lexer_tr/extconf.rb +0 -6
  140. data/ext/gherkin_lexer_tr/gherkin_lexer_tr.c +0 -1347
  141. data/ext/gherkin_lexer_tt/extconf.rb +0 -6
  142. data/ext/gherkin_lexer_tt/gherkin_lexer_tt.c +0 -1936
  143. data/ext/gherkin_lexer_uk/extconf.rb +0 -6
  144. data/ext/gherkin_lexer_uk/gherkin_lexer_uk.c +0 -1941
  145. data/ext/gherkin_lexer_uz/extconf.rb +0 -6
  146. data/ext/gherkin_lexer_uz/gherkin_lexer_uz.c +0 -1651
  147. data/ext/gherkin_lexer_vi/extconf.rb +0 -6
  148. data/ext/gherkin_lexer_vi/gherkin_lexer_vi.c +0 -1458
  149. data/ext/gherkin_lexer_zh_cn/extconf.rb +0 -6
  150. data/ext/gherkin_lexer_zh_cn/gherkin_lexer_zh_cn.c +0 -1314
  151. data/ext/gherkin_lexer_zh_tw/extconf.rb +0 -6
  152. data/ext/gherkin_lexer_zh_tw/gherkin_lexer_zh_tw.c +0 -1307
  153. data/features/.cucumber/stepdefs.json +0 -244
  154. data/features/escaped_pipes.feature +0 -8
  155. data/features/feature_parser.feature +0 -237
  156. data/features/json_formatter.feature +0 -498
  157. data/features/json_parser.feature +0 -331
  158. data/features/native_lexer.feature +0 -19
  159. data/features/parser_with_native_lexer.feature +0 -205
  160. data/features/pretty_formatter.feature +0 -17
  161. data/features/step_definitions/eyeball_steps.rb +0 -3
  162. data/features/step_definitions/gherkin_steps.rb +0 -29
  163. data/features/step_definitions/json_formatter_steps.rb +0 -30
  164. data/features/step_definitions/json_parser_steps.rb +0 -21
  165. data/features/step_definitions/pretty_formatter_steps.rb +0 -85
  166. data/features/steps_parser.feature +0 -46
  167. data/features/support/env.rb +0 -41
  168. data/install_mingw_os_x.sh +0 -5
  169. data/js/.npmignore +0 -1
  170. data/js/lib/gherkin/lexer/.npmignore +0 -0
  171. data/lib/gherkin.rb +0 -2
  172. data/lib/gherkin/README.md +0 -7
  173. data/lib/gherkin/c_lexer.rb +0 -17
  174. data/lib/gherkin/formatter/ansi_escapes.rb +0 -97
  175. data/lib/gherkin/formatter/argument.rb +0 -16
  176. data/lib/gherkin/formatter/escaping.rb +0 -15
  177. data/lib/gherkin/formatter/filter_formatter.rb +0 -150
  178. data/lib/gherkin/formatter/hashable.rb +0 -25
  179. data/lib/gherkin/formatter/json_formatter.rb +0 -133
  180. data/lib/gherkin/formatter/line_filter.rb +0 -26
  181. data/lib/gherkin/formatter/model.rb +0 -281
  182. data/lib/gherkin/formatter/pretty_formatter.rb +0 -245
  183. data/lib/gherkin/formatter/regexp_filter.rb +0 -21
  184. data/lib/gherkin/formatter/step_printer.rb +0 -21
  185. data/lib/gherkin/formatter/tag_count_formatter.rb +0 -47
  186. data/lib/gherkin/formatter/tag_filter.rb +0 -19
  187. data/lib/gherkin/i18n.json +0 -786
  188. data/lib/gherkin/i18n.rb +0 -176
  189. data/lib/gherkin/json_parser.rb +0 -177
  190. data/lib/gherkin/lexer/ar.rb +0 -1170
  191. data/lib/gherkin/lexer/bg.rb +0 -1382
  192. data/lib/gherkin/lexer/bm.rb +0 -1250
  193. data/lib/gherkin/lexer/ca.rb +0 -1310
  194. data/lib/gherkin/lexer/cs.rb +0 -1246
  195. data/lib/gherkin/lexer/cy_gb.rb +0 -1032
  196. data/lib/gherkin/lexer/da.rb +0 -1048
  197. data/lib/gherkin/lexer/de.rb +0 -1166
  198. data/lib/gherkin/lexer/el.rb +0 -1628
  199. data/lib/gherkin/lexer/en.rb +0 -1156
  200. data/lib/gherkin/lexer/en_au.rb +0 -1486
  201. data/lib/gherkin/lexer/en_lol.rb +0 -934
  202. data/lib/gherkin/lexer/en_old.rb +0 -1037
  203. data/lib/gherkin/lexer/en_pirate.rb +0 -1210
  204. data/lib/gherkin/lexer/en_scouse.rb +0 -1362
  205. data/lib/gherkin/lexer/en_tx.rb +0 -1016
  206. data/lib/gherkin/lexer/encoding.rb +0 -41
  207. data/lib/gherkin/lexer/eo.rb +0 -995
  208. data/lib/gherkin/lexer/es.rb +0 -1140
  209. data/lib/gherkin/lexer/et.rb +0 -990
  210. data/lib/gherkin/lexer/fa.rb +0 -1214
  211. data/lib/gherkin/lexer/fi.rb +0 -969
  212. data/lib/gherkin/lexer/fr.rb +0 -1228
  213. data/lib/gherkin/lexer/gl.rb +0 -1117
  214. data/lib/gherkin/lexer/he.rb +0 -1118
  215. data/lib/gherkin/lexer/hi.rb +0 -1559
  216. data/lib/gherkin/lexer/hr.rb +0 -1066
  217. data/lib/gherkin/lexer/hu.rb +0 -1118
  218. data/lib/gherkin/lexer/i18n_lexer.rb +0 -48
  219. data/lib/gherkin/lexer/id.rb +0 -963
  220. data/lib/gherkin/lexer/is.rb +0 -1120
  221. data/lib/gherkin/lexer/it.rb +0 -1086
  222. data/lib/gherkin/lexer/ja.rb +0 -1418
  223. data/lib/gherkin/lexer/kn.rb +0 -1672
  224. data/lib/gherkin/lexer/ko.rb +0 -1102
  225. data/lib/gherkin/lexer/lt.rb +0 -1045
  226. data/lib/gherkin/lexer/lu.rb +0 -1132
  227. data/lib/gherkin/lexer/lv.rb +0 -1166
  228. data/lib/gherkin/lexer/nl.rb +0 -1115
  229. data/lib/gherkin/lexer/no.rb +0 -1060
  230. data/lib/gherkin/lexer/pa.rb +0 -1852
  231. data/lib/gherkin/lexer/pl.rb +0 -1457
  232. data/lib/gherkin/lexer/pt.rb +0 -1430
  233. data/lib/gherkin/lexer/ro.rb +0 -1164
  234. data/lib/gherkin/lexer/ru.rb +0 -1766
  235. data/lib/gherkin/lexer/sk.rb +0 -1452
  236. data/lib/gherkin/lexer/sr_cyrl.rb +0 -1803
  237. data/lib/gherkin/lexer/sr_latn.rb +0 -1294
  238. data/lib/gherkin/lexer/sv.rb +0 -1070
  239. data/lib/gherkin/lexer/th.rb +0 -2492
  240. data/lib/gherkin/lexer/tl.rb +0 -1726
  241. data/lib/gherkin/lexer/tr.rb +0 -1092
  242. data/lib/gherkin/lexer/tt.rb +0 -1643
  243. data/lib/gherkin/lexer/uk.rb +0 -1646
  244. data/lib/gherkin/lexer/uz.rb +0 -1376
  245. data/lib/gherkin/lexer/vi.rb +0 -1198
  246. data/lib/gherkin/lexer/zh_cn.rb +0 -1058
  247. data/lib/gherkin/lexer/zh_tw.rb +0 -1052
  248. data/lib/gherkin/listener/event.rb +0 -45
  249. data/lib/gherkin/listener/formatter_listener.rb +0 -150
  250. data/lib/gherkin/native.rb +0 -7
  251. data/lib/gherkin/native/java.rb +0 -72
  252. data/lib/gherkin/native/null.rb +0 -5
  253. data/lib/gherkin/native/therubyracer.rb +0 -41
  254. data/lib/gherkin/parser/meta.txt +0 -5
  255. data/lib/gherkin/parser/parser.rb +0 -166
  256. data/lib/gherkin/parser/root.txt +0 -11
  257. data/lib/gherkin/parser/steps.txt +0 -4
  258. data/lib/gherkin/platform.rb +0 -13
  259. data/lib/gherkin/rubify.rb +0 -31
  260. data/lib/gherkin/tag_expression.rb +0 -63
  261. data/ragel/lexer.c.rl.erb +0 -454
  262. data/ragel/lexer.java.rl.erb +0 -219
  263. data/ragel/lexer.js.rl.erb +0 -322
  264. data/ragel/lexer.rb.rl.erb +0 -179
  265. data/ragel/lexer_common.rl.erb +0 -50
  266. data/spec/gherkin/c_lexer_spec.rb +0 -22
  267. data/spec/gherkin/fixtures/1.feature +0 -8
  268. data/spec/gherkin/fixtures/comments_in_table.feature +0 -9
  269. data/spec/gherkin/fixtures/complex.feature +0 -45
  270. data/spec/gherkin/fixtures/complex.json +0 -139
  271. data/spec/gherkin/fixtures/complex_for_filtering.feature +0 -60
  272. data/spec/gherkin/fixtures/complex_with_tags.feature +0 -61
  273. data/spec/gherkin/fixtures/dos_line_endings.feature +0 -45
  274. data/spec/gherkin/fixtures/examples_with_only_header.feature +0 -14
  275. data/spec/gherkin/fixtures/hantu_pisang.feature +0 -35
  276. data/spec/gherkin/fixtures/i18n_fr.feature +0 -14
  277. data/spec/gherkin/fixtures/i18n_fr2.feature +0 -8
  278. data/spec/gherkin/fixtures/i18n_no.feature +0 -7
  279. data/spec/gherkin/fixtures/i18n_pt1.feature +0 -44
  280. data/spec/gherkin/fixtures/i18n_pt2.feature +0 -4
  281. data/spec/gherkin/fixtures/i18n_pt3.feature +0 -4
  282. data/spec/gherkin/fixtures/i18n_pt4.feature +0 -4
  283. data/spec/gherkin/fixtures/i18n_zh-CN.feature +0 -9
  284. data/spec/gherkin/fixtures/iso-8859-1.feature +0 -6
  285. data/spec/gherkin/fixtures/issue_145.feature +0 -22
  286. data/spec/gherkin/fixtures/scenario_outline_with_tags.feature +0 -13
  287. data/spec/gherkin/fixtures/scenario_without_steps.feature +0 -5
  288. data/spec/gherkin/fixtures/simple_with_comments.feature +0 -7
  289. data/spec/gherkin/fixtures/simple_with_tags.feature +0 -11
  290. data/spec/gherkin/fixtures/with_bom.feature +0 -3
  291. data/spec/gherkin/fixtures/with_bom_and_language_spec.feature +0 -4
  292. data/spec/gherkin/formatter/ansi_escapes_spec.rb +0 -32
  293. data/spec/gherkin/formatter/filter_formatter_spec.rb +0 -204
  294. data/spec/gherkin/formatter/json_formatter_spec.rb +0 -179
  295. data/spec/gherkin/formatter/model_spec.rb +0 -28
  296. data/spec/gherkin/formatter/pretty_formatter_spec.rb +0 -184
  297. data/spec/gherkin/formatter/spaces.feature +0 -9
  298. data/spec/gherkin/formatter/step_printer_spec.rb +0 -55
  299. data/spec/gherkin/formatter/tabs.feature +0 -9
  300. data/spec/gherkin/formatter/tag_count_formatter_spec.rb +0 -30
  301. data/spec/gherkin/i18n_spec.rb +0 -256
  302. data/spec/gherkin/java_lexer_spec.rb +0 -20
  303. data/spec/gherkin/json_parser_spec.rb +0 -165
  304. data/spec/gherkin/lexer/i18n_lexer_spec.rb +0 -65
  305. data/spec/gherkin/native_lexer_spec.rb +0 -29
  306. data/spec/gherkin/parser/parser_spec.rb +0 -16
  307. data/spec/gherkin/rubify_spec.rb +0 -23
  308. data/spec/gherkin/sexp_recorder.rb +0 -59
  309. data/spec/gherkin/shared/doc_string_group.rb +0 -161
  310. data/spec/gherkin/shared/encoding_group.rb +0 -48
  311. data/spec/gherkin/shared/lexer_group.rb +0 -589
  312. data/spec/gherkin/shared/row_group.rb +0 -123
  313. data/spec/gherkin/shared/tags_group.rb +0 -52
  314. data/spec/gherkin/tag_expression_spec.rb +0 -146
  315. data/spec/spec_helper.rb +0 -94
  316. data/tasks/apidoc.rake +0 -32
  317. data/tasks/bench.rake +0 -184
  318. data/tasks/bench/feature_builder.rb +0 -49
  319. data/tasks/bench/null_listener.rb +0 -4
  320. data/tasks/compile.rake +0 -83
  321. data/tasks/cucumber.rake +0 -23
  322. data/tasks/gems.rake +0 -46
  323. data/tasks/ikvm.rake +0 -126
  324. data/tasks/ragel_task.rb +0 -124
  325. data/tasks/release.rake +0 -34
  326. data/tasks/rspec.rake +0 -6
  327. data/tasks/yard/default/layout/html/bubble_32x32.png +0 -0
  328. data/tasks/yard/default/layout/html/bubble_48x48.png +0 -0
  329. data/tasks/yard/default/layout/html/footer.erb +0 -5
  330. data/tasks/yard/default/layout/html/index.erb +0 -1
  331. data/tasks/yard/default/layout/html/layout.erb +0 -25
  332. data/tasks/yard/default/layout/html/logo.erb +0 -1
  333. data/tasks/yard/default/layout/html/setup.rb +0 -4
data/Rakefile CHANGED
@@ -1,26 +1,24 @@
1
1
  # encoding: utf-8
2
- GHERKIN_VERSION = Gem::Specification.load(File.dirname(__FILE__) + '/gherkin.gemspec').version.version
3
2
  require 'rubygems'
4
- unless ENV['RUBY_CC_VERSION']
5
- require 'bundler'
6
- Bundler.setup
7
- Bundler::GemHelper.install_tasks
8
- end
9
- ENV['RUBYOPT'] = nil # Necessary to prevent Bundler from *&^%$#ing up rake-compiler.
3
+ require 'bundler'
4
+ Bundler::GemHelper.install_tasks
10
5
 
11
- require 'rake/clean'
6
+ $:.unshift File.expand_path("../lib", __FILE__)
12
7
 
13
- begin
14
- # Support Rake >= 0.9.0
15
- require 'rake/dsl_definition'
16
- include Rake::DSL
17
- rescue LoadError
8
+ require "rspec/core/rake_task"
9
+ RSpec::Core::RakeTask.new(:spec) do |t|
10
+ t.ruby_opts = %w[-r./spec/coverage -w]
11
+ t.rspec_opts = %w[--color]
18
12
  end
19
13
 
20
- $:.unshift(File.dirname(__FILE__) + '/lib')
21
-
22
- Dir['tasks/**/*.rake'].each { |rake| load File.expand_path(rake) }
14
+ require_relative 'spec/capture_warnings'
15
+ include CaptureWarnings
16
+ namespace :spec do
17
+ task :warnings do
18
+ report_warnings do
19
+ Rake::Task['spec'].invoke
20
+ end
21
+ end
22
+ end
23
23
 
24
- task :default => [:spec, :cucumber]
25
- task :spec => defined?(JRUBY_VERSION) ? :jar : :compile
26
- task :cucumber => defined?(JRUBY_VERSION) ? :jar : :compile
24
+ task default: ['spec:warnings']
@@ -0,0 +1,21 @@
1
+ #!/usr/bin/env ruby
2
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__),"../lib"))
3
+ require 'gherkin/parser'
4
+ require 'json'
5
+
6
+ parser = Gherkin::Parser.new
7
+ parser.stop_at_first_error = false
8
+ files = ARGV.any? ? ARGV : (STDIN.tty? ? [] : [STDIN])
9
+ start_time = Time.now
10
+ files.each do |file|
11
+ begin
12
+ File.open(file) do |io|
13
+ puts JSON.generate(parser.parse(io))
14
+ end
15
+ rescue Gherkin::ParserError => e
16
+ STDERR.puts e.message
17
+ exit 1
18
+ end
19
+ end
20
+ end_time = Time.now
21
+ STDERR.puts (end_time - start_time)*1000 if ENV['GHERKIN_PERF']
@@ -0,0 +1,25 @@
1
+ #!/usr/bin/env ruby
2
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__),"../lib"))
3
+ require 'gherkin/pickles/compiler'
4
+ require 'gherkin/parser'
5
+ require 'json'
6
+
7
+ compiler = Gherkin::Pickles::Compiler.new
8
+ parser = Gherkin::Parser.new
9
+ parser.stop_at_first_error = false
10
+ files = ARGV.any? ? ARGV : (STDIN.tty? ? [] : [STDIN])
11
+ start_time = Time.now
12
+ files.each do |file|
13
+ begin
14
+ File.open(file) do |io|
15
+ feature = parser.parse(io)
16
+ pickles = compiler.compile(feature, file);
17
+ puts JSON.generate(pickles)
18
+ end
19
+ rescue Gherkin::ParserError => e
20
+ STDERR.puts e.message
21
+ exit 1
22
+ end
23
+ end
24
+ end_time = Time.now
25
+ STDERR.puts (end_time - start_time)*1000 if ENV['GHERKIN_PERF']
@@ -0,0 +1,12 @@
1
+ #!/usr/bin/env ruby
2
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__),"../lib"))
3
+ require 'gherkin/parser'
4
+ require 'gherkin/token_formatter_builder'
5
+
6
+ parser = Gherkin::Parser.new(Gherkin::TokenFormatterBuilder.new)
7
+ files = ARGV + (STDIN.tty? ? [] : [STDIN])
8
+ files.each do |file|
9
+ File.open(file) do |io|
10
+ print parser.parse(io)
11
+ end
12
+ end
@@ -0,0 +1,218 @@
1
+ @using Berp;
2
+ @helper CallProduction(ProductionRule production)
3
+ {
4
+ switch(production.Type)
5
+ {
6
+ case ProductionRuleType.Start:
7
+ @: start_rule(context, :@production.RuleName);
8
+ break;
9
+ case ProductionRuleType.End:
10
+ @: end_rule(context, :@production.RuleName);
11
+ break;
12
+ case ProductionRuleType.Process:
13
+ @: build(context, token);
14
+ break;
15
+ }
16
+ }
17
+ @helper HandleParserError(IEnumerable<string> expectedTokens, State state)
18
+ {<text>
19
+ state_comment = "State: @state.Id - @Raw(state.Comment)"
20
+ token.detach
21
+ expected_tokens = ["@Raw(string.Join("\", \"", expectedTokens))"]
22
+ error = token.eof? ? UnexpectedEOFException.new(token, expected_tokens, state_comment) : UnexpectedTokenException.new(token, expected_tokens, state_comment)
23
+ raise error if (stop_at_first_error)
24
+ add_error(context, error)
25
+ return @state.Id</text>}
26
+ @helper MatchToken(TokenType tokenType)
27
+ {<text>match_@(tokenType)(context, token)</text>}
28
+ # This file is generated. Do not edit! Edit gherkin-ruby.razor instead.
29
+ require_relative 'ast_builder'
30
+ require_relative 'token_matcher'
31
+ require_relative 'token_scanner'
32
+ require_relative 'errors'
33
+
34
+ module Gherkin
35
+
36
+ RULE_TYPE = [
37
+ :None,
38
+ @foreach(var rule in Model.RuleSet.Where(r => !r.TempRule))
39
+ {<text> :@rule.Name.Replace("#", "_"), # @rule.ToString(true)
40
+ </text>}
41
+ ]
42
+
43
+ class ParserContext
44
+ attr_reader :token_scanner, :token_matcher, :token_queue, :errors
45
+
46
+ def initialize(token_scanner, token_matcher, token_queue, errors)
47
+ @@token_scanner = token_scanner
48
+ @@token_matcher = token_matcher
49
+ @@token_queue = token_queue
50
+ @@errors = errors
51
+ end
52
+ end
53
+
54
+ class @Model.ParserClassName
55
+ attr_accessor :stop_at_first_error
56
+
57
+ def initialize(ast_builder=AstBuilder.new)
58
+ @@ast_builder = ast_builder
59
+ end
60
+
61
+ def parse(token_scanner, token_matcher=TokenMatcher.new)
62
+ token_scanner = token_scanner.is_a?(TokenScanner) ? token_scanner : TokenScanner.new(token_scanner)
63
+
64
+ @@ast_builder.reset
65
+ token_matcher.reset
66
+ context = ParserContext.new(
67
+ token_scanner,
68
+ token_matcher,
69
+ [],
70
+ []
71
+ )
72
+
73
+ start_rule(context, :@Model.RuleSet.StartRule.Name);
74
+ state = 0
75
+ token = nil
76
+ begin
77
+ token = read_token(context)
78
+ state = match_token(state, token, context)
79
+ end until(token.eof?)
80
+
81
+ end_rule(context, :@Model.RuleSet.StartRule.Name)
82
+
83
+ raise CompositeParserException.new(context.errors) if context.errors.any?
84
+
85
+ get_result()
86
+ end
87
+
88
+ def build(context, token)
89
+ handle_ast_error(context) do
90
+ @@ast_builder.build(token)
91
+ end
92
+ end
93
+
94
+ def add_error(context, error)
95
+ context.errors.push(error)
96
+ raise CompositeParserException, context.errors if context.errors.length > 10
97
+ end
98
+
99
+ def start_rule(context, rule_type)
100
+ handle_ast_error(context) do
101
+ @@ast_builder.start_rule(rule_type)
102
+ end
103
+ end
104
+
105
+ def end_rule(context, rule_type)
106
+ handle_ast_error(context) do
107
+ @@ast_builder.end_rule(rule_type)
108
+ end
109
+ end
110
+
111
+ def get_result()
112
+ @@ast_builder.get_result
113
+ end
114
+
115
+ def read_token(context)
116
+ context.token_queue.any? ? context.token_queue.shift : context.token_scanner.read
117
+ end
118
+
119
+ @foreach(var rule in Model.RuleSet.TokenRules)
120
+ {<text>
121
+ def match_@(rule.Name.Replace("#", ""))( context, token)
122
+ @if (rule.Name != "#EOF")
123
+ {
124
+ @:return false if token.eof?
125
+ }
126
+ return handle_external_error(context, false) do
127
+ context.token_matcher.match_@(rule.Name.Replace("#", ""))(token)
128
+ end
129
+ end</text>
130
+ }
131
+
132
+ def match_token(state, token, context)
133
+ case state
134
+ @foreach(var state in Model.States.Values.Where(s => !s.IsEndState))
135
+ {
136
+ @:when @state.Id
137
+ @:match_token_at_@(state.Id)(token, context)
138
+ }
139
+ else
140
+ raise InvalidOperationException, "Unknown state: #{state}"
141
+ end
142
+ end
143
+
144
+ @foreach(var state in Model.States.Values.Where(s => !s.IsEndState))
145
+ {<text>
146
+ # @Raw(state.Comment)
147
+ def match_token_at_@(state.Id)(token, context)
148
+ @foreach(var transition in state.Transitions)
149
+ {
150
+ @:if @MatchToken(transition.TokenType)
151
+ if (transition.LookAheadHint != null)
152
+ {
153
+ @:if lookahead_@(transition.LookAheadHint.Id)(context, token)
154
+ }
155
+ foreach(var production in transition.Productions)
156
+ {
157
+ @CallProduction(production)
158
+ }
159
+ @:return @transition.TargetState
160
+ if (transition.LookAheadHint != null)
161
+ {
162
+ @:end
163
+ }
164
+ @:end
165
+ }
166
+ @HandleParserError(state.Transitions.Select(t => "#" + t.TokenType.ToString()).Distinct(), state)
167
+ end</text>
168
+ }
169
+
170
+ @foreach(var lookAheadHint in Model.RuleSet.LookAheadHints)
171
+ {
172
+ <text>
173
+ def lookahead_@(lookAheadHint.Id)(context, currentToken)
174
+ currentToken.detach
175
+ token = nil
176
+ queue = []
177
+ match = false
178
+ loop do
179
+ token = read_token(context)
180
+ token.detach
181
+ queue.push(token)
182
+
183
+ if (false @foreach(var tokenType in lookAheadHint.ExpectedTokens) {<text>|| @MatchToken(tokenType)</text>})
184
+ match = true
185
+ break
186
+ end
187
+
188
+ break unless (false @foreach(var tokenType in lookAheadHint.Skip) {<text>|| @MatchToken(tokenType)</text>})
189
+ end
190
+
191
+ context.token_queue.concat(queue)
192
+
193
+ return match
194
+ end
195
+ </text>
196
+ }
197
+
198
+ private
199
+
200
+ def handle_ast_error(context, &action)
201
+ handle_external_error(context, true, &action)
202
+ end
203
+
204
+ def handle_external_error(context, default_value, &action)
205
+ return action.call if stop_at_first_error
206
+
207
+ begin
208
+ return action.call
209
+ rescue CompositeParserException => e
210
+ e.errors.each { |error| add_error(context, error) }
211
+ rescue ParserException => e
212
+ add_error(context, e)
213
+ end
214
+ default_value
215
+ end
216
+
217
+ end
218
+ end
@@ -1,80 +1,26 @@
1
- # -*- encoding: utf-8 -*-
2
- $LOAD_PATH.unshift File.expand_path("../lib", __FILE__)
3
- require 'gherkin/platform'
4
-
1
+ # encoding: utf-8
5
2
  Gem::Specification.new do |s|
6
- s.name = "gherkin"
7
- # If the MINOR version is bumped (MAJOR.MINOR.PATCH), verify that the change is ok:
8
- #
9
- # * Comment out the cucumber dependency below
10
- # * Uncomment the cucumber location in Gemfile
11
- # * Bump cucumber's gherkin dependency to the new version
12
- # * bundle update
13
- # * bundle exec rake
14
- #
15
- # Repeat these step for cucumber (swap `cucumber` with `gherkin`).
16
- #
17
- # When both are building OK, do a `bundle exec rake install` in both cucumber and gherkin projects, revert the changes in the first 2 steps
18
- # and release both projects. Do this for both ruby 1.9.3, ruby 2.0.0 and jruby.
19
- #
20
- s.version = Gherkin::VERSION
21
- s.authors = ["Mike Sassak", "Gregory Hnatiuk", "Aslak Hellesøy"]
22
- s.description = "A fast Gherkin lexer/parser based on the Ragel State Machine Compiler."
23
- s.summary = "#{s.name}-#{s.version}"
24
- s.email = "cukes@googlegroups.com"
25
- s.homepage = "http://github.com/cucumber/gherkin"
26
- s.license = 'MIT'
27
-
28
- s.files = `git ls-files`.split("\n")
29
- s.test_files = `git ls-files -- {spec,features}/*`.split("\n")
30
- s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
3
+ s.name = 'gherkin'
4
+ s.version = '3.2.0'
5
+ s.authors = ["Gáspár Nagy", "Aslak Hellesøy", "Steve Tooke"]
6
+ s.description = 'Gherkin parser'
7
+ s.summary = "gherkin-#{s.version}"
8
+ s.email = 'cukes@googlegroups.com'
9
+ s.homepage = "https://github.com/cucumber/gherkin"
10
+ s.platform = Gem::Platform::RUBY
11
+ s.license = "MIT"
12
+ s.required_ruby_version = ">= 1.9.3"
13
+
14
+ s.add_development_dependency 'bundler', '~> 1.7'
15
+ s.add_development_dependency 'rake', '~> 10.4'
16
+ s.add_development_dependency 'rspec', '~> 3.3'
17
+
18
+ # For coverage reports
19
+ s.add_development_dependency 'coveralls', '~> 0.8', '< 0.8.8'
20
+
21
+ s.rubygems_version = ">= 1.6.1"
22
+ s.files = `git ls-files`.split("\n").reject {|path| path =~ /\.gitignore$/ }
23
+ s.test_files = `git ls-files -- spec/*`.split("\n")
31
24
  s.rdoc_options = ["--charset=UTF-8"]
32
25
  s.require_path = "lib"
33
-
34
- s.files -= Dir['ikvm/**/*']
35
- s.files -= Dir['java/**/*']
36
- s.files -= Dir['js/**/*']
37
- s.files -= Dir['ext/**/*']
38
- s.files -= Dir['lib/gherkin.jar']
39
- s.files -= Dir['lib/**/*.dll']
40
- s.files -= Dir['lib/**/*.bundle']
41
- s.files -= Dir['lib/**/*.so']
42
-
43
- if ENV['GEM_PLATFORM']
44
- puts "GEM_PLATFORM:#{ENV['GEM_PLATFORM']}"
45
- end
46
- s.platform = ENV['GEM_PLATFORM'] if ENV['GEM_PLATFORM']
47
- case s.platform.to_s
48
- when /java/
49
- s.files += Dir['lib/*.jar']
50
- when /mswin|mingw32/
51
- s.files += Dir['lib/*/*.so']
52
- when /dotnet/
53
- s.files += Dir['lib/*.dll']
54
- else # MRI or Rubinius
55
- s.files += Dir['lib/gherkin/lexer/*.rb']
56
- s.files += Dir['ext/**/*.c']
57
- s.extensions = Dir['ext/**/extconf.rb']
58
- s.add_development_dependency('rake-compiler', '= 0.8.2')
59
- end
60
- s.files -= Dir['**/.gitignore']
61
-
62
- s.add_runtime_dependency('multi_json', '~> 1.3')
63
-
64
- s.add_development_dependency('cucumber', '>= 1.3.8')
65
- s.add_development_dependency('rake', '>= 10.1.0')
66
- s.add_development_dependency('bundler', '>= 1.3.5') # Make sure it's in sync with /.travis.yml
67
- s.add_development_dependency('rspec', '~> 2.14.1')
68
- s.add_development_dependency('rubyzip', '>= 1.0.0')
69
- s.add_development_dependency('ruby-beautify', '= 0.92.2')
70
-
71
- unless ENV['RUBY_CC_VERSION'] || Gherkin::JRUBY
72
- s.add_development_dependency('therubyracer', '>= 0.12.0') if ENV['GHERKIN_JS']
73
- # For Documentation:
74
- s.add_development_dependency('yard', '>= 0.8.7.2')
75
- s.add_development_dependency('rdiscount', '>= 2.1.6')
76
- end
77
-
78
- # Only needed by Cucumber. Remove when Cucumber no longer needs those.
79
- s.add_development_dependency('builder', '>= 3.2.2')
80
26
  end
@@ -0,0 +1,243 @@
1
+ require_relative 'ast_node'
2
+
3
+ module Gherkin
4
+ class AstBuilder
5
+ def initialize
6
+ reset
7
+ end
8
+
9
+ def reset
10
+ @stack = [AstNode.new(:None)]
11
+ @comments = []
12
+ end
13
+
14
+ def start_rule(rule_type)
15
+ @stack.push AstNode.new(rule_type)
16
+ end
17
+
18
+ def end_rule(rule_type)
19
+ node = @stack.pop
20
+ current_node.add(node.rule_type, transform_node(node))
21
+ end
22
+
23
+ def build(token)
24
+ if token.matched_type == :Comment
25
+ @comments.push({
26
+ type: 'Comment',
27
+ location: get_location(token),
28
+ text: token.matched_text
29
+ })
30
+ else
31
+ current_node.add(token.matched_type, token)
32
+ end
33
+ end
34
+
35
+ def get_result
36
+ current_node.get_single(:Feature)
37
+ end
38
+
39
+ def current_node
40
+ @stack.last
41
+ end
42
+
43
+ def get_location(token, column=nil)
44
+ # TODO: translated from JS... is it right?
45
+ (column.nil? || column.zero?) ? token.location : {line: token.location[:line], column: column}
46
+ end
47
+
48
+ def get_tags(node)
49
+ tags = []
50
+ tags_node = node.get_single(:Tags)
51
+ return tags unless tags_node
52
+
53
+ tags_node.get_tokens(:TagLine).each do |token|
54
+ token.matched_items.each do |tag_item|
55
+ tags.push({
56
+ type: 'Tag',
57
+ location: get_location(token, tag_item.column),
58
+ name: tag_item.text
59
+ })
60
+ end
61
+ end
62
+
63
+ tags
64
+ end
65
+
66
+ def get_table_rows(node)
67
+ rows = node.get_tokens(:TableRow).map do |token|
68
+ {
69
+ type: 'TableRow',
70
+ location: get_location(token),
71
+ cells: get_cells(token)
72
+ }
73
+ end
74
+ ensure_cell_count(rows);
75
+ rows
76
+ end
77
+
78
+ def ensure_cell_count(rows)
79
+ return if rows.empty?
80
+ cell_count = rows[0][:cells].length
81
+ rows.each do |row|
82
+ if (row[:cells].length != cell_count)
83
+ raise AstBuilderException.new("inconsistent cell count within the table", row[:location]);
84
+ end
85
+ end
86
+ end
87
+
88
+ def get_cells(table_row_token)
89
+ table_row_token.matched_items.map do |cell_item|
90
+ {
91
+ type: 'TableCell',
92
+ location: get_location(table_row_token, cell_item.column),
93
+ value: cell_item.text
94
+ }
95
+ end
96
+ end
97
+
98
+ def get_description(node)
99
+ node.get_single(:Description)
100
+ end
101
+
102
+ def get_steps(node)
103
+ node.get_items(:Step)
104
+ end
105
+
106
+ def transform_node(node)
107
+ case node.rule_type
108
+ when :Step
109
+ step_line = node.get_token(:StepLine)
110
+ step_argument = node.get_single(:DataTable) || node.get_single(:DocString) || nil
111
+
112
+ reject_nils(
113
+ type: node.rule_type,
114
+ location: get_location(step_line),
115
+ keyword: step_line.matched_keyword,
116
+ text: step_line.matched_text,
117
+ argument: step_argument
118
+ )
119
+ when :DocString
120
+ separator_token = node.get_tokens(:DocStringSeparator)[0]
121
+ content_type = separator_token.matched_text == '' ? nil : separator_token.matched_text
122
+ line_tokens = node.get_tokens(:Other)
123
+ content = line_tokens.map { |t| t.matched_text }.join("\n")
124
+
125
+ reject_nils(
126
+ type: node.rule_type,
127
+ location: get_location(separator_token),
128
+ contentType: content_type,
129
+ content: content
130
+ )
131
+ when :DataTable
132
+ rows = get_table_rows(node)
133
+ reject_nils(
134
+ type: node.rule_type,
135
+ location: rows[0][:location],
136
+ rows: rows,
137
+ )
138
+ when :Background
139
+ background_line = node.get_token(:BackgroundLine)
140
+ description = get_description(node)
141
+ steps = get_steps(node)
142
+
143
+ reject_nils(
144
+ type: node.rule_type,
145
+ location: get_location(background_line),
146
+ keyword: background_line.matched_keyword,
147
+ name: background_line.matched_text,
148
+ description: description,
149
+ steps: steps
150
+ )
151
+ when :Scenario_Definition
152
+ tags = get_tags(node)
153
+ scenario_node = node.get_single(:Scenario)
154
+ if(scenario_node)
155
+ scenario_line = scenario_node.get_token(:ScenarioLine)
156
+ description = get_description(scenario_node)
157
+ steps = get_steps(scenario_node)
158
+
159
+ reject_nils(
160
+ type: scenario_node.rule_type,
161
+ tags: tags,
162
+ location: get_location(scenario_line),
163
+ keyword: scenario_line.matched_keyword,
164
+ name: scenario_line.matched_text,
165
+ description: description,
166
+ steps: steps
167
+ )
168
+ else
169
+ scenario_outline_node = node.get_single(:ScenarioOutline)
170
+ raise 'Internal grammar error' unless scenario_outline_node
171
+
172
+ scenario_outline_line = scenario_outline_node.get_token(:ScenarioOutlineLine)
173
+ description = get_description(scenario_outline_node)
174
+ steps = get_steps(scenario_outline_node)
175
+ examples = scenario_outline_node.get_items(:Examples_Definition)
176
+
177
+ reject_nils(
178
+ type: scenario_outline_node.rule_type,
179
+ tags: tags,
180
+ location: get_location(scenario_outline_line),
181
+ keyword: scenario_outline_line.matched_keyword,
182
+ name: scenario_outline_line.matched_text,
183
+ description: description,
184
+ steps: steps,
185
+ examples: examples
186
+ )
187
+ end
188
+ when :Examples_Definition
189
+ tags = get_tags(node)
190
+ examples_node = node.get_single(:Examples)
191
+ examples_line = examples_node.get_token(:ExamplesLine)
192
+ description = get_description(examples_node)
193
+ rows = get_table_rows(examples_node)
194
+
195
+ reject_nils(
196
+ type: examples_node.rule_type,
197
+ tags: tags,
198
+ location: get_location(examples_line),
199
+ keyword: examples_line.matched_keyword,
200
+ name: examples_line.matched_text,
201
+ description: description,
202
+ tableHeader: rows.first,
203
+ tableBody: rows[1..-1]
204
+ )
205
+ when :Description
206
+ line_tokens = node.get_tokens(:Other)
207
+ # Trim trailing empty lines
208
+ last_non_empty = line_tokens.rindex { |token| !token.line.trimmed_line_text.empty? }
209
+ description = line_tokens[0..last_non_empty].map { |token| token.matched_text }.join("\n")
210
+ return description
211
+ when :Feature
212
+ header = node.get_single(:Feature_Header)
213
+ return unless header
214
+ tags = get_tags(header)
215
+ feature_line = header.get_token(:FeatureLine)
216
+ return unless feature_line
217
+ background = node.get_single(:Background)
218
+ scenario_definitions = node.get_items(:Scenario_Definition)
219
+ description = get_description(header)
220
+ language = feature_line.matched_gherkin_dialect
221
+
222
+ reject_nils(
223
+ type: node.rule_type,
224
+ tags: tags,
225
+ location: get_location(feature_line),
226
+ language: language,
227
+ keyword: feature_line.matched_keyword,
228
+ name: feature_line.matched_text,
229
+ description: description,
230
+ background: background,
231
+ scenarioDefinitions: scenario_definitions,
232
+ comments: @comments
233
+ )
234
+ else
235
+ return node
236
+ end
237
+ end
238
+
239
+ def reject_nils(values)
240
+ values.reject { |k,v| v.nil? }
241
+ end
242
+ end
243
+ end