rubygems-update 3.1.6 → 3.3.22

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 (703) hide show
  1. checksums.yaml +4 -4
  2. data/{History.txt → CHANGELOG.md} +1452 -514
  3. data/CODE_OF_CONDUCT.md +55 -19
  4. data/CONTRIBUTING.md +74 -27
  5. data/Manifest.txt +154 -117
  6. data/POLICIES.md +27 -19
  7. data/README.md +10 -8
  8. data/UPGRADING.md +5 -81
  9. data/bin/gem +3 -8
  10. data/bin/update_rubygems +5 -5
  11. data/bundler/CHANGELOG.md +2358 -1430
  12. data/bundler/README.md +7 -9
  13. data/bundler/UPGRADING.md +27 -34
  14. data/bundler/bundler.gemspec +8 -11
  15. data/bundler/exe/bundle +10 -8
  16. data/bundler/exe/bundler +1 -1
  17. data/bundler/lib/bundler/.document +1 -0
  18. data/bundler/lib/bundler/build_metadata.rb +4 -12
  19. data/bundler/lib/bundler/cli/add.rb +1 -1
  20. data/bundler/lib/bundler/cli/binstubs.rb +6 -2
  21. data/bundler/lib/bundler/cli/cache.rb +3 -8
  22. data/bundler/lib/bundler/cli/check.rb +4 -2
  23. data/bundler/lib/bundler/cli/clean.rb +1 -1
  24. data/bundler/lib/bundler/cli/common.rb +31 -3
  25. data/bundler/lib/bundler/cli/config.rb +10 -1
  26. data/bundler/lib/bundler/cli/console.rb +1 -1
  27. data/bundler/lib/bundler/cli/doctor.rb +25 -6
  28. data/bundler/lib/bundler/cli/exec.rb +5 -10
  29. data/bundler/lib/bundler/cli/fund.rb +36 -0
  30. data/bundler/lib/bundler/cli/gem.rb +219 -28
  31. data/bundler/lib/bundler/cli/info.rb +38 -6
  32. data/bundler/lib/bundler/cli/init.rb +3 -3
  33. data/bundler/lib/bundler/cli/inject.rb +1 -1
  34. data/bundler/lib/bundler/cli/install.rb +25 -54
  35. data/bundler/lib/bundler/cli/issue.rb +5 -4
  36. data/bundler/lib/bundler/cli/list.rb +19 -11
  37. data/bundler/lib/bundler/cli/lock.rb +5 -1
  38. data/bundler/lib/bundler/cli/open.rb +1 -2
  39. data/bundler/lib/bundler/cli/outdated.rb +105 -76
  40. data/bundler/lib/bundler/cli/platform.rb +2 -2
  41. data/bundler/lib/bundler/cli/plugin.rb +10 -0
  42. data/bundler/lib/bundler/cli/pristine.rb +5 -0
  43. data/bundler/lib/bundler/cli/remove.rb +1 -2
  44. data/bundler/lib/bundler/cli/show.rb +2 -2
  45. data/bundler/lib/bundler/cli/update.rb +20 -9
  46. data/bundler/lib/bundler/cli.rb +106 -83
  47. data/bundler/lib/bundler/compact_index_client/cache.rb +6 -23
  48. data/bundler/lib/bundler/compact_index_client/gem_parser.rb +28 -0
  49. data/bundler/lib/bundler/compact_index_client/updater.rb +29 -25
  50. data/bundler/lib/bundler/compact_index_client.rb +3 -9
  51. data/bundler/lib/bundler/current_ruby.rb +17 -6
  52. data/bundler/lib/bundler/definition.rb +297 -396
  53. data/bundler/lib/bundler/dep_proxy.rb +16 -9
  54. data/bundler/lib/bundler/dependency.rb +22 -77
  55. data/bundler/lib/bundler/digest.rb +71 -0
  56. data/bundler/lib/bundler/dsl.rb +76 -83
  57. data/bundler/lib/bundler/endpoint_specification.rb +16 -14
  58. data/bundler/lib/bundler/env.rb +2 -2
  59. data/bundler/lib/bundler/environment_preserver.rb +29 -2
  60. data/bundler/lib/bundler/errors.rb +30 -3
  61. data/bundler/lib/bundler/feature_flag.rb +0 -9
  62. data/bundler/lib/bundler/fetcher/base.rb +7 -9
  63. data/bundler/lib/bundler/fetcher/compact_index.rb +11 -16
  64. data/bundler/lib/bundler/fetcher/downloader.rb +10 -7
  65. data/bundler/lib/bundler/fetcher/index.rb +2 -30
  66. data/bundler/lib/bundler/fetcher.rb +24 -25
  67. data/bundler/lib/bundler/friendly_errors.rb +46 -47
  68. data/bundler/lib/bundler/gem_helper.rb +53 -31
  69. data/bundler/lib/bundler/gem_helpers.rb +43 -24
  70. data/bundler/lib/bundler/gem_version_promoter.rb +10 -20
  71. data/bundler/lib/bundler/graph.rb +1 -1
  72. data/bundler/lib/bundler/index.rb +19 -25
  73. data/bundler/lib/bundler/injector.rb +38 -6
  74. data/bundler/lib/bundler/inline.rb +4 -13
  75. data/bundler/lib/bundler/installer/gem_installer.rb +19 -24
  76. data/bundler/lib/bundler/installer/parallel_installer.rb +46 -25
  77. data/bundler/lib/bundler/installer/standalone.rb +58 -11
  78. data/bundler/lib/bundler/installer.rb +51 -81
  79. data/bundler/lib/bundler/lazy_specification.rb +79 -36
  80. data/bundler/lib/bundler/lockfile_generator.rb +3 -3
  81. data/bundler/lib/bundler/lockfile_parser.rb +19 -44
  82. data/bundler/lib/bundler/man/.document +1 -0
  83. data/bundler/{man → lib/bundler/man}/bundle-add.1 +21 -5
  84. data/bundler/{man/bundle-add.ronn → lib/bundler/man/bundle-add.1.ronn} +16 -4
  85. data/bundler/{man → lib/bundler/man}/bundle-binstubs.1 +5 -3
  86. data/bundler/{man/bundle-binstubs.ronn → lib/bundler/man/bundle-binstubs.1.ronn} +2 -4
  87. data/bundler/{man → lib/bundler/man}/bundle-cache.1 +7 -1
  88. data/bundler/{man/bundle-cache.ronn → lib/bundler/man/bundle-cache.1.ronn} +7 -0
  89. data/bundler/{man → lib/bundler/man}/bundle-check.1 +1 -1
  90. data/bundler/{man → lib/bundler/man}/bundle-clean.1 +1 -1
  91. data/bundler/{man → lib/bundler/man}/bundle-config.1 +65 -47
  92. data/bundler/{man/bundle-config.ronn → lib/bundler/man/bundle-config.1.ronn} +74 -65
  93. data/bundler/lib/bundler/man/bundle-console.1 +53 -0
  94. data/bundler/lib/bundler/man/bundle-console.1.ronn +44 -0
  95. data/bundler/{man → lib/bundler/man}/bundle-doctor.1 +1 -1
  96. data/bundler/{man → lib/bundler/man}/bundle-exec.1 +2 -2
  97. data/bundler/{man/bundle-exec.ronn → lib/bundler/man/bundle-exec.1.ronn} +1 -1
  98. data/bundler/{man → lib/bundler/man}/bundle-gem.1 +38 -3
  99. data/bundler/{man/bundle-gem.ronn → lib/bundler/man/bundle-gem.1.ronn} +46 -7
  100. data/bundler/lib/bundler/man/bundle-help.1 +13 -0
  101. data/bundler/lib/bundler/man/bundle-help.1.ronn +12 -0
  102. data/bundler/{man → lib/bundler/man}/bundle-info.1 +1 -1
  103. data/bundler/{man → lib/bundler/man}/bundle-init.1 +1 -1
  104. data/bundler/{man → lib/bundler/man}/bundle-inject.1 +5 -2
  105. data/bundler/{man/bundle-inject.ronn → lib/bundler/man/bundle-inject.1.ronn} +3 -1
  106. data/bundler/{man → lib/bundler/man}/bundle-install.1 +35 -4
  107. data/bundler/{man/bundle-install.ronn → lib/bundler/man/bundle-install.1.ronn} +33 -5
  108. data/bundler/{man → lib/bundler/man}/bundle-list.1 +7 -7
  109. data/bundler/{man/bundle-list.ronn → lib/bundler/man/bundle-list.1.ronn} +6 -6
  110. data/bundler/{man → lib/bundler/man}/bundle-lock.1 +1 -1
  111. data/bundler/{man → lib/bundler/man}/bundle-open.1 +1 -1
  112. data/bundler/{man → lib/bundler/man}/bundle-outdated.1 +3 -10
  113. data/bundler/{man/bundle-outdated.ronn → lib/bundler/man/bundle-outdated.1.ronn} +1 -10
  114. data/bundler/{man → lib/bundler/man}/bundle-platform.1 +16 -6
  115. data/bundler/{man/bundle-platform.ronn → lib/bundler/man/bundle-platform.1.ronn} +14 -7
  116. data/bundler/lib/bundler/man/bundle-plugin.1 +81 -0
  117. data/bundler/lib/bundler/man/bundle-plugin.1.ronn +59 -0
  118. data/bundler/{man → lib/bundler/man}/bundle-pristine.1 +1 -1
  119. data/bundler/{man → lib/bundler/man}/bundle-remove.1 +1 -1
  120. data/bundler/{man → lib/bundler/man}/bundle-show.1 +1 -1
  121. data/bundler/{man → lib/bundler/man}/bundle-update.1 +5 -5
  122. data/bundler/{man/bundle-update.ronn → lib/bundler/man/bundle-update.1.ronn} +5 -4
  123. data/bundler/lib/bundler/man/bundle-version.1 +35 -0
  124. data/bundler/lib/bundler/man/bundle-version.1.ronn +24 -0
  125. data/bundler/{man → lib/bundler/man}/bundle-viz.1 +4 -1
  126. data/bundler/{man/bundle-viz.ronn → lib/bundler/man/bundle-viz.1.ronn} +2 -0
  127. data/bundler/{man → lib/bundler/man}/bundle.1 +15 -10
  128. data/bundler/{man/bundle.ronn → lib/bundler/man/bundle.1.ronn} +12 -7
  129. data/bundler/{man → lib/bundler/man}/gemfile.5 +106 -80
  130. data/bundler/{man → lib/bundler/man}/gemfile.5.ronn +94 -83
  131. data/bundler/{man → lib/bundler/man}/index.txt +4 -0
  132. data/bundler/lib/bundler/match_metadata.rb +13 -0
  133. data/bundler/lib/bundler/match_platform.rb +0 -1
  134. data/bundler/lib/bundler/match_remote_metadata.rb +29 -0
  135. data/bundler/lib/bundler/mirror.rb +2 -2
  136. data/bundler/lib/bundler/plugin/api/source.rb +25 -9
  137. data/bundler/lib/bundler/plugin/dsl.rb +1 -1
  138. data/bundler/lib/bundler/plugin/index.rb +13 -1
  139. data/bundler/lib/bundler/plugin/installer/git.rb +0 -4
  140. data/bundler/lib/bundler/plugin/installer/rubygems.rb +1 -5
  141. data/bundler/lib/bundler/plugin/installer.rb +11 -11
  142. data/bundler/lib/bundler/plugin/source_list.rb +5 -1
  143. data/bundler/lib/bundler/plugin.rb +58 -11
  144. data/bundler/lib/bundler/process_lock.rb +1 -1
  145. data/bundler/lib/bundler/remote_specification.rb +11 -6
  146. data/bundler/lib/bundler/resolver/base.rb +50 -0
  147. data/bundler/lib/bundler/resolver/spec_group.rb +64 -57
  148. data/bundler/lib/bundler/resolver.rb +234 -230
  149. data/bundler/lib/bundler/retry.rb +2 -2
  150. data/bundler/lib/bundler/ruby_dsl.rb +1 -1
  151. data/bundler/lib/bundler/ruby_version.rb +6 -19
  152. data/bundler/lib/bundler/rubygems_ext.rb +194 -33
  153. data/bundler/lib/bundler/rubygems_gem_installer.rb +87 -16
  154. data/bundler/lib/bundler/rubygems_integration.rb +71 -153
  155. data/bundler/lib/bundler/runtime.rb +22 -26
  156. data/bundler/lib/bundler/self_manager.rb +168 -0
  157. data/bundler/lib/bundler/settings.rb +145 -66
  158. data/bundler/lib/bundler/setup.rb +2 -2
  159. data/bundler/lib/bundler/shared_helpers.rb +17 -32
  160. data/bundler/lib/bundler/similarity_detector.rb +1 -1
  161. data/bundler/lib/bundler/source/git/git_proxy.rb +88 -84
  162. data/bundler/lib/bundler/source/git.rb +50 -32
  163. data/bundler/lib/bundler/source/metadata.rb +3 -7
  164. data/bundler/lib/bundler/source/path/installer.rb +10 -10
  165. data/bundler/lib/bundler/source/path.rb +10 -4
  166. data/bundler/lib/bundler/source/rubygems/remote.rb +1 -1
  167. data/bundler/lib/bundler/source/rubygems.rb +201 -179
  168. data/bundler/lib/bundler/source/rubygems_aggregate.rb +68 -0
  169. data/bundler/lib/bundler/source.rb +25 -5
  170. data/bundler/lib/bundler/source_list.rb +105 -63
  171. data/bundler/lib/bundler/source_map.rb +71 -0
  172. data/bundler/lib/bundler/spec_set.rb +62 -57
  173. data/bundler/lib/bundler/stub_specification.rb +30 -10
  174. data/bundler/lib/bundler/templates/Executable +2 -4
  175. data/bundler/lib/bundler/templates/Executable.bundler +8 -8
  176. data/bundler/lib/bundler/templates/Executable.standalone +2 -4
  177. data/bundler/lib/bundler/templates/Gemfile +0 -2
  178. data/bundler/lib/bundler/templates/gems.rb +0 -3
  179. data/bundler/lib/bundler/templates/newgem/CHANGELOG.md.tt +5 -0
  180. data/bundler/lib/bundler/templates/newgem/CODE_OF_CONDUCT.md.tt +57 -47
  181. data/bundler/lib/bundler/templates/newgem/Gemfile.tt +12 -1
  182. data/bundler/lib/bundler/templates/newgem/README.md.tt +9 -14
  183. data/bundler/lib/bundler/templates/newgem/Rakefile.tt +32 -5
  184. data/bundler/lib/bundler/templates/newgem/bin/console.tt +1 -0
  185. data/bundler/lib/bundler/templates/newgem/circleci/config.yml.tt +13 -0
  186. data/bundler/lib/bundler/templates/newgem/ext/newgem/extconf.rb.tt +2 -0
  187. data/bundler/lib/bundler/templates/newgem/github/workflows/main.yml.tt +27 -0
  188. data/bundler/lib/bundler/templates/newgem/gitlab-ci.yml.tt +9 -0
  189. data/bundler/lib/bundler/templates/newgem/lib/newgem/version.rb.tt +2 -0
  190. data/bundler/lib/bundler/templates/newgem/lib/newgem.rb.tt +4 -2
  191. data/bundler/lib/bundler/templates/newgem/newgem.gemspec.tt +27 -17
  192. data/bundler/lib/bundler/templates/newgem/rubocop.yml.tt +13 -0
  193. data/bundler/lib/bundler/templates/newgem/sig/newgem.rbs.tt +8 -0
  194. data/bundler/lib/bundler/templates/newgem/spec/newgem_spec.rb.tt +2 -0
  195. data/bundler/lib/bundler/templates/newgem/spec/spec_helper.rb.tt +2 -1
  196. data/bundler/lib/bundler/templates/newgem/standard.yml.tt +3 -0
  197. data/bundler/lib/bundler/templates/newgem/test/{test_helper.rb.tt → minitest/test_helper.rb.tt} +2 -0
  198. data/bundler/lib/bundler/templates/newgem/test/{newgem_test.rb.tt → minitest/test_newgem.rb.tt} +3 -1
  199. data/bundler/lib/bundler/templates/newgem/test/test-unit/newgem_test.rb.tt +15 -0
  200. data/bundler/lib/bundler/templates/newgem/test/test-unit/test_helper.rb.tt +6 -0
  201. data/bundler/lib/bundler/ui/shell.rb +6 -6
  202. data/bundler/lib/bundler/uri_credentials_filter.rb +3 -1
  203. data/bundler/lib/bundler/vendor/.document +1 -0
  204. data/bundler/lib/bundler/vendor/connection_pool/LICENSE +20 -0
  205. data/bundler/lib/bundler/vendor/connection_pool/lib/connection_pool/timed_stack.rb +19 -21
  206. data/bundler/lib/bundler/vendor/connection_pool/lib/connection_pool/version.rb +1 -1
  207. data/bundler/lib/bundler/vendor/connection_pool/lib/connection_pool/wrapper.rb +57 -0
  208. data/bundler/lib/bundler/vendor/connection_pool/lib/connection_pool.rb +39 -74
  209. data/bundler/lib/bundler/vendor/fileutils/LICENSE.txt +22 -0
  210. data/bundler/lib/bundler/vendor/molinillo/LICENSE +9 -0
  211. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/delegates/specification_provider.rb +7 -0
  212. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/dependency_graph/tag.rb +2 -2
  213. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/dependency_graph/vertex.rb +11 -5
  214. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/dependency_graph.rb +37 -5
  215. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/errors.rb +34 -28
  216. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/gem_metadata.rb +1 -1
  217. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/modules/specification_provider.rb +12 -1
  218. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/resolution.rb +49 -47
  219. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo.rb +0 -1
  220. data/bundler/lib/bundler/vendor/net-http-persistent/README.rdoc +82 -0
  221. data/bundler/lib/bundler/vendor/net-http-persistent/lib/net/http/persistent.rb +82 -189
  222. data/bundler/lib/bundler/vendor/thor/LICENSE.md +20 -0
  223. data/bundler/lib/bundler/vendor/thor/lib/thor/actions/create_link.rb +2 -1
  224. data/bundler/lib/bundler/vendor/thor/lib/thor/actions/file_manipulation.rb +9 -7
  225. data/bundler/lib/bundler/vendor/thor/lib/thor/actions/inject_into_file.rb +1 -2
  226. data/bundler/lib/bundler/vendor/thor/lib/thor/actions.rb +7 -3
  227. data/bundler/lib/bundler/vendor/thor/lib/thor/base.rb +9 -0
  228. data/bundler/lib/bundler/vendor/thor/lib/thor/core_ext/hash_with_indifferent_access.rb +6 -0
  229. data/bundler/lib/bundler/vendor/thor/lib/thor/error.rb +10 -5
  230. data/bundler/lib/bundler/vendor/thor/lib/thor/parser/arguments.rb +5 -1
  231. data/bundler/lib/bundler/vendor/thor/lib/thor/parser/options.rb +28 -9
  232. data/bundler/lib/bundler/vendor/thor/lib/thor/shell/basic.rb +27 -6
  233. data/bundler/lib/bundler/vendor/thor/lib/thor/shell/color.rb +5 -1
  234. data/bundler/lib/bundler/vendor/thor/lib/thor/shell.rb +1 -1
  235. data/bundler/lib/bundler/vendor/thor/lib/thor/util.rb +1 -1
  236. data/bundler/lib/bundler/vendor/thor/lib/thor/version.rb +1 -1
  237. data/bundler/lib/bundler/vendor/thor/lib/thor.rb +5 -13
  238. data/bundler/lib/bundler/vendor/tmpdir/lib/tmpdir.rb +154 -0
  239. data/bundler/lib/bundler/vendor/tsort/LICENSE.txt +22 -0
  240. data/bundler/lib/bundler/vendor/tsort/lib/tsort.rb +452 -0
  241. data/bundler/lib/bundler/vendor/uri/LICENSE.txt +22 -0
  242. data/bundler/lib/bundler/vendor/uri/lib/uri/common.rb +17 -80
  243. data/bundler/lib/bundler/vendor/uri/lib/uri/ftp.rb +0 -1
  244. data/bundler/lib/bundler/vendor/uri/lib/uri/generic.rb +5 -6
  245. data/bundler/lib/bundler/vendor/uri/lib/uri/http.rb +0 -1
  246. data/bundler/lib/bundler/vendor/uri/lib/uri/https.rb +0 -1
  247. data/bundler/lib/bundler/vendor/uri/lib/uri/ldap.rb +1 -1
  248. data/bundler/lib/bundler/vendor/uri/lib/uri/mailto.rb +0 -1
  249. data/bundler/lib/bundler/vendor/uri/lib/uri/rfc2396_parser.rb +1 -14
  250. data/bundler/lib/bundler/vendor/uri/lib/uri/rfc3986_parser.rb +1 -12
  251. data/bundler/lib/bundler/vendor/uri/lib/uri/version.rb +1 -1
  252. data/bundler/lib/bundler/vendor/uri/lib/uri/ws.rb +84 -0
  253. data/bundler/lib/bundler/vendor/uri/lib/uri/wss.rb +22 -0
  254. data/bundler/lib/bundler/vendor/uri/lib/uri.rb +0 -1
  255. data/bundler/lib/bundler/vendored_persistent.rb +0 -7
  256. data/bundler/lib/bundler/vendored_tmpdir.rb +4 -0
  257. data/bundler/lib/bundler/vendored_tsort.rb +4 -0
  258. data/bundler/lib/bundler/version.rb +1 -1
  259. data/bundler/lib/bundler/worker.rb +20 -5
  260. data/bundler/lib/bundler/yaml_serializer.rb +1 -1
  261. data/bundler/lib/bundler.rb +79 -47
  262. data/hide_lib_for_update/note.txt +0 -4
  263. data/lib/rubygems/available_set.rb +7 -9
  264. data/lib/rubygems/basic_specification.rb +23 -21
  265. data/lib/rubygems/bundler_version_finder.rb +26 -53
  266. data/lib/rubygems/command.rb +48 -45
  267. data/lib/rubygems/command_manager.rb +20 -14
  268. data/lib/rubygems/commands/build_command.rb +47 -27
  269. data/lib/rubygems/commands/cert_command.rb +78 -75
  270. data/lib/rubygems/commands/check_command.rb +20 -22
  271. data/lib/rubygems/commands/cleanup_command.rb +30 -26
  272. data/lib/rubygems/commands/contents_command.rb +16 -18
  273. data/lib/rubygems/commands/dependency_command.rb +39 -51
  274. data/lib/rubygems/commands/environment_command.rb +11 -10
  275. data/lib/rubygems/commands/fetch_command.rb +33 -16
  276. data/lib/rubygems/commands/generate_index_command.rb +17 -19
  277. data/lib/rubygems/commands/help_command.rb +6 -6
  278. data/lib/rubygems/commands/info_command.rb +10 -7
  279. data/lib/rubygems/commands/install_command.rb +29 -27
  280. data/lib/rubygems/commands/list_command.rb +9 -8
  281. data/lib/rubygems/commands/lock_command.rb +5 -7
  282. data/lib/rubygems/commands/mirror_command.rb +3 -5
  283. data/lib/rubygems/commands/open_command.rb +10 -14
  284. data/lib/rubygems/commands/outdated_command.rb +5 -7
  285. data/lib/rubygems/commands/owner_command.rb +26 -16
  286. data/lib/rubygems/commands/pristine_command.rb +61 -49
  287. data/lib/rubygems/commands/push_command.rb +22 -59
  288. data/lib/rubygems/commands/query_command.rb +21 -351
  289. data/lib/rubygems/commands/rdoc_command.rb +23 -25
  290. data/lib/rubygems/commands/search_command.rb +8 -8
  291. data/lib/rubygems/commands/server_command.rb +16 -77
  292. data/lib/rubygems/commands/setup_command.rb +255 -229
  293. data/lib/rubygems/commands/signin_command.rb +9 -11
  294. data/lib/rubygems/commands/signout_command.rb +7 -9
  295. data/lib/rubygems/commands/sources_command.rb +27 -25
  296. data/lib/rubygems/commands/specification_command.rb +25 -21
  297. data/lib/rubygems/commands/stale_command.rb +3 -5
  298. data/lib/rubygems/commands/uninstall_command.rb +41 -43
  299. data/lib/rubygems/commands/unpack_command.rb +13 -15
  300. data/lib/rubygems/commands/update_command.rb +132 -74
  301. data/lib/rubygems/commands/which_command.rb +7 -9
  302. data/lib/rubygems/commands/yank_command.rb +16 -19
  303. data/lib/rubygems/compatibility.rb +4 -2
  304. data/lib/rubygems/config_file.rb +42 -26
  305. data/lib/rubygems/core_ext/kernel_gem.rb +1 -1
  306. data/lib/rubygems/core_ext/kernel_require.rb +6 -3
  307. data/lib/rubygems/core_ext/kernel_warn.rb +12 -13
  308. data/lib/rubygems/core_ext/tcpsocket_init.rb +52 -0
  309. data/lib/rubygems/defaults.rb +108 -32
  310. data/lib/rubygems/dependency.rb +21 -22
  311. data/lib/rubygems/dependency_installer.rb +19 -91
  312. data/lib/rubygems/dependency_list.rb +13 -15
  313. data/lib/rubygems/deprecate.rb +102 -8
  314. data/lib/rubygems/doctor.rb +22 -22
  315. data/lib/rubygems/errors.rb +6 -21
  316. data/lib/rubygems/exceptions.rb +33 -34
  317. data/lib/rubygems/ext/build_error.rb +2 -0
  318. data/lib/rubygems/ext/builder.rb +48 -54
  319. data/lib/rubygems/ext/cargo_builder/link_flag_converter.rb +23 -0
  320. data/lib/rubygems/ext/cargo_builder.rb +321 -0
  321. data/lib/rubygems/ext/cmake_builder.rb +6 -9
  322. data/lib/rubygems/ext/configure_builder.rb +5 -8
  323. data/lib/rubygems/ext/ext_conf_builder.rb +42 -66
  324. data/lib/rubygems/ext/rake_builder.rb +7 -10
  325. data/lib/rubygems/ext.rb +7 -6
  326. data/lib/rubygems/gem_runner.rb +15 -26
  327. data/lib/rubygems/gemcutter_utilities.rb +153 -32
  328. data/lib/rubygems/indexer.rb +30 -49
  329. data/lib/rubygems/install_default_message.rb +2 -2
  330. data/lib/rubygems/install_message.rb +2 -2
  331. data/lib/rubygems/install_update_options.rb +73 -64
  332. data/lib/rubygems/installer.rb +194 -151
  333. data/lib/rubygems/installer_uninstaller_utils.rb +29 -0
  334. data/lib/rubygems/local_remote_options.rb +22 -24
  335. data/lib/rubygems/mock_gem_ui.rb +2 -8
  336. data/lib/rubygems/name_tuple.rb +9 -14
  337. data/lib/rubygems/openssl.rb +7 -0
  338. data/lib/rubygems/optparse/.document +1 -0
  339. data/lib/rubygems/optparse/COPYING +56 -0
  340. data/lib/rubygems/optparse/lib/optionparser.rb +2 -0
  341. data/lib/rubygems/optparse/lib/optparse/ac.rb +54 -0
  342. data/lib/rubygems/optparse/lib/optparse/date.rb +18 -0
  343. data/lib/rubygems/optparse/lib/optparse/kwargs.rb +22 -0
  344. data/lib/rubygems/optparse/lib/optparse/shellwords.rb +7 -0
  345. data/lib/rubygems/optparse/lib/optparse/time.rb +11 -0
  346. data/lib/rubygems/optparse/lib/optparse/uri.rb +7 -0
  347. data/lib/rubygems/optparse/lib/optparse/version.rb +71 -0
  348. data/lib/rubygems/optparse/lib/optparse.rb +2303 -0
  349. data/lib/rubygems/optparse.rb +3 -0
  350. data/lib/rubygems/package/digest_io.rb +0 -2
  351. data/lib/rubygems/package/file_source.rb +2 -4
  352. data/lib/rubygems/package/io_source.rb +4 -2
  353. data/lib/rubygems/package/old.rb +9 -11
  354. data/lib/rubygems/package/tar_header.rb +54 -56
  355. data/lib/rubygems/package/tar_reader/entry.rb +2 -5
  356. data/lib/rubygems/package/tar_reader.rb +1 -4
  357. data/lib/rubygems/package/tar_writer.rb +8 -18
  358. data/lib/rubygems/package.rb +101 -117
  359. data/lib/rubygems/package_task.rb +5 -11
  360. data/lib/rubygems/path_support.rb +2 -9
  361. data/lib/rubygems/platform.rb +101 -76
  362. data/lib/rubygems/psych_tree.rb +1 -3
  363. data/lib/rubygems/query_utils.rb +351 -0
  364. data/lib/rubygems/rdoc.rb +2 -14
  365. data/lib/rubygems/remote_fetcher.rb +48 -74
  366. data/lib/rubygems/request/connection_pools.rb +6 -10
  367. data/lib/rubygems/request/http_pool.rb +2 -4
  368. data/lib/rubygems/request/https_pool.rb +0 -2
  369. data/lib/rubygems/request.rb +26 -31
  370. data/lib/rubygems/request_set/gem_dependency_api.rb +23 -25
  371. data/lib/rubygems/request_set/lockfile/parser.rb +26 -28
  372. data/lib/rubygems/request_set/lockfile/tokenizer.rb +5 -7
  373. data/lib/rubygems/request_set/lockfile.rb +17 -21
  374. data/lib/rubygems/request_set.rb +27 -40
  375. data/lib/rubygems/requirement.rb +28 -48
  376. data/lib/rubygems/resolver/activation_request.rb +12 -6
  377. data/lib/rubygems/resolver/api_set/gem_parser.rb +20 -0
  378. data/lib/rubygems/resolver/api_set.rb +32 -25
  379. data/lib/rubygems/resolver/api_specification.rb +29 -15
  380. data/lib/rubygems/resolver/best_set.rb +7 -9
  381. data/lib/rubygems/resolver/composed_set.rb +3 -5
  382. data/lib/rubygems/resolver/conflict.rb +12 -14
  383. data/lib/rubygems/resolver/current_set.rb +0 -2
  384. data/lib/rubygems/resolver/dependency_request.rb +3 -5
  385. data/lib/rubygems/resolver/git_set.rb +2 -4
  386. data/lib/rubygems/resolver/git_specification.rb +6 -8
  387. data/lib/rubygems/resolver/index_set.rb +4 -6
  388. data/lib/rubygems/resolver/index_specification.rb +38 -7
  389. data/lib/rubygems/resolver/installed_specification.rb +4 -6
  390. data/lib/rubygems/resolver/installer_set.rb +66 -24
  391. data/lib/rubygems/resolver/local_specification.rb +2 -4
  392. data/lib/rubygems/resolver/lock_set.rb +6 -8
  393. data/lib/rubygems/resolver/lock_specification.rb +4 -6
  394. data/lib/rubygems/resolver/molinillo/LICENSE +9 -0
  395. data/lib/rubygems/resolver/molinillo/lib/molinillo/delegates/resolution_state.rb +7 -0
  396. data/lib/rubygems/resolver/molinillo/lib/molinillo/delegates/specification_provider.rb +8 -0
  397. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/action.rb +1 -0
  398. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/add_edge_no_circular.rb +2 -1
  399. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/add_vertex.rb +2 -1
  400. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/delete_edge.rb +2 -1
  401. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/detach_vertex_named.rb +2 -1
  402. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/log.rb +7 -6
  403. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/set_payload.rb +2 -1
  404. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/tag.rb +4 -3
  405. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph/vertex.rb +51 -12
  406. data/lib/rubygems/resolver/molinillo/lib/molinillo/dependency_graph.rb +41 -8
  407. data/lib/rubygems/resolver/molinillo/lib/molinillo/errors.rb +76 -8
  408. data/lib/rubygems/resolver/molinillo/lib/molinillo/gem_metadata.rb +2 -1
  409. data/lib/rubygems/resolver/molinillo/lib/molinillo/modules/specification_provider.rb +13 -1
  410. data/lib/rubygems/resolver/molinillo/lib/molinillo/modules/ui.rb +3 -1
  411. data/lib/rubygems/resolver/molinillo/lib/molinillo/resolution.rb +510 -165
  412. data/lib/rubygems/resolver/molinillo/lib/molinillo/resolver.rb +3 -2
  413. data/lib/rubygems/resolver/molinillo/lib/molinillo/state.rb +8 -4
  414. data/lib/rubygems/resolver/molinillo/lib/molinillo.rb +6 -5
  415. data/lib/rubygems/resolver/molinillo.rb +1 -1
  416. data/lib/rubygems/resolver/requirement_list.rb +0 -2
  417. data/lib/rubygems/resolver/set.rb +0 -3
  418. data/lib/rubygems/resolver/source_set.rb +0 -2
  419. data/lib/rubygems/resolver/spec_specification.rb +14 -2
  420. data/lib/rubygems/resolver/specification.rb +14 -4
  421. data/lib/rubygems/resolver/stats.rb +0 -2
  422. data/lib/rubygems/resolver/vendor_set.rb +1 -3
  423. data/lib/rubygems/resolver/vendor_specification.rb +3 -5
  424. data/lib/rubygems/resolver.rb +53 -51
  425. data/lib/rubygems/s3_uri_signer.rb +7 -15
  426. data/lib/rubygems/safe_yaml.rb +14 -16
  427. data/lib/rubygems/security/policies.rb +11 -11
  428. data/lib/rubygems/security/policy.rb +25 -29
  429. data/lib/rubygems/security/signer.rb +13 -16
  430. data/lib/rubygems/security/trust_dir.rb +4 -5
  431. data/lib/rubygems/security.rb +103 -69
  432. data/lib/rubygems/security_option.rb +7 -7
  433. data/lib/rubygems/source/git.rb +30 -30
  434. data/lib/rubygems/source/installed.rb +1 -3
  435. data/lib/rubygems/source/local.rb +4 -6
  436. data/lib/rubygems/source/lock.rb +0 -2
  437. data/lib/rubygems/source/specific_file.rb +1 -3
  438. data/lib/rubygems/source/vendor.rb +0 -2
  439. data/lib/rubygems/source.rb +44 -38
  440. data/lib/rubygems/source_list.rb +9 -16
  441. data/lib/rubygems/spec_fetcher.rb +49 -48
  442. data/lib/rubygems/specification.rb +338 -295
  443. data/lib/rubygems/specification_policy.rb +112 -61
  444. data/lib/rubygems/stub_specification.rb +21 -28
  445. data/lib/rubygems/text.rb +21 -20
  446. data/lib/rubygems/tsort/.document +1 -0
  447. data/lib/rubygems/tsort/LICENSE.txt +22 -0
  448. data/lib/rubygems/tsort/lib/tsort.rb +454 -0
  449. data/lib/rubygems/tsort.rb +3 -0
  450. data/lib/rubygems/uninstaller.rb +98 -45
  451. data/lib/rubygems/unknown_command_spell_checker.rb +21 -0
  452. data/lib/rubygems/uri.rb +126 -0
  453. data/lib/rubygems/uri_formatter.rb +2 -3
  454. data/lib/rubygems/user_interaction.rb +43 -54
  455. data/lib/rubygems/util/licenses.rb +115 -10
  456. data/lib/rubygems/util/list.rb +0 -2
  457. data/lib/rubygems/util.rb +18 -12
  458. data/lib/rubygems/validator.rb +6 -8
  459. data/lib/rubygems/version.rb +29 -20
  460. data/lib/rubygems/version_option.rb +11 -5
  461. data/lib/rubygems.rb +192 -289
  462. data/rubygems-update.gemspec +4 -4
  463. data/setup.rb +11 -22
  464. data/test/rubygems/data/excon-0.7.7.gemspec.rz +0 -0
  465. data/test/rubygems/data/null-required-ruby-version.gemspec.rz +0 -0
  466. data/test/rubygems/data/null-required-rubygems-version.gemspec.rz +0 -0
  467. data/test/rubygems/data/pry-0.4.7.gemspec.rz +0 -0
  468. data/test/rubygems/encrypted_private_key.pem +26 -26
  469. data/{lib/rubygems/test_case.rb → test/rubygems/helper.rb} +380 -314
  470. data/{lib → test}/rubygems/installer_test_case.rb +38 -24
  471. data/{lib → test}/rubygems/package/tar_test_case.rb +4 -6
  472. data/test/rubygems/packages/ascii_binder-0.1.10.1.gem +0 -0
  473. data/test/rubygems/packages/ill-formatted-platform-1.0.0.10.gem +0 -0
  474. data/test/rubygems/plugin/exception/rubygems_plugin.rb +1 -1
  475. data/test/rubygems/plugin/load/rubygems_plugin.rb +0 -2
  476. data/test/rubygems/plugin/standarderror/rubygems_plugin.rb +1 -1
  477. data/test/rubygems/private_ec_key.pem +9 -0
  478. data/test/rubygems/rubygems/commands/crash_command.rb +0 -2
  479. data/test/rubygems/rubygems_plugin.rb +2 -4
  480. data/test/rubygems/specifications/bar-0.0.2.gemspec +0 -2
  481. data/test/rubygems/specifications/rubyforge-0.0.1.gemspec +4 -6
  482. data/test/rubygems/test_bundled_ca.rb +43 -50
  483. data/test/rubygems/test_config.rb +5 -7
  484. data/test/rubygems/test_deprecate.rb +87 -10
  485. data/test/rubygems/test_exit.rb +11 -0
  486. data/test/rubygems/test_gem.rb +719 -639
  487. data/test/rubygems/test_gem_available_set.rb +24 -26
  488. data/test/rubygems/test_gem_bundler_version_finder.rb +37 -44
  489. data/test/rubygems/test_gem_command.rb +78 -49
  490. data/test/rubygems/test_gem_command_manager.rb +98 -35
  491. data/test/rubygems/test_gem_commands_build_command.rb +275 -57
  492. data/test/rubygems/test_gem_commands_cert_command.rb +180 -125
  493. data/test/rubygems/test_gem_commands_check_command.rb +9 -11
  494. data/test/rubygems/test_gem_commands_cleanup_command.rb +78 -69
  495. data/test/rubygems/test_gem_commands_contents_command.rb +73 -42
  496. data/test/rubygems/test_gem_commands_dependency_command.rb +38 -40
  497. data/test/rubygems/test_gem_commands_environment_command.rb +60 -38
  498. data/test/rubygems/test_gem_commands_fetch_command.rb +163 -32
  499. data/test/rubygems/test_gem_commands_generate_index_command.rb +8 -14
  500. data/test/rubygems/test_gem_commands_help_command.rb +34 -16
  501. data/test/rubygems/test_gem_commands_info_command.rb +33 -9
  502. data/test/rubygems/test_gem_commands_install_command.rb +401 -198
  503. data/test/rubygems/test_gem_commands_list_command.rb +5 -7
  504. data/test/rubygems/test_gem_commands_lock_command.rb +11 -13
  505. data/test/rubygems/test_gem_commands_mirror.rb +3 -5
  506. data/test/rubygems/test_gem_commands_open_command.rb +16 -19
  507. data/test/rubygems/test_gem_commands_outdated_command.rb +23 -7
  508. data/test/rubygems/test_gem_commands_owner_command.rb +100 -46
  509. data/test/rubygems/test_gem_commands_pristine_command.rb +131 -99
  510. data/test/rubygems/test_gem_commands_push_command.rb +134 -66
  511. data/test/rubygems/test_gem_commands_query_command.rb +91 -91
  512. data/test/rubygems/test_gem_commands_search_command.rb +2 -4
  513. data/test/rubygems/test_gem_commands_server_command.rb +6 -50
  514. data/test/rubygems/test_gem_commands_setup_command.rb +288 -179
  515. data/test/rubygems/test_gem_commands_signin_command.rb +162 -28
  516. data/test/rubygems/test_gem_commands_signout_command.rb +3 -10
  517. data/test/rubygems/test_gem_commands_sources_command.rb +139 -32
  518. data/test/rubygems/test_gem_commands_specification_command.rb +81 -55
  519. data/test/rubygems/test_gem_commands_stale_command.rb +4 -6
  520. data/test/rubygems/test_gem_commands_uninstall_command.rb +80 -82
  521. data/test/rubygems/test_gem_commands_unpack_command.rb +32 -34
  522. data/test/rubygems/test_gem_commands_update_command.rb +335 -88
  523. data/test/rubygems/test_gem_commands_which_command.rb +12 -14
  524. data/test/rubygems/test_gem_commands_yank_command.rb +80 -44
  525. data/test/rubygems/test_gem_config_file.rb +114 -97
  526. data/test/rubygems/test_gem_dependency.rb +87 -87
  527. data/test/rubygems/test_gem_dependency_installer.rb +277 -391
  528. data/test/rubygems/test_gem_dependency_list.rb +57 -59
  529. data/test/rubygems/test_gem_dependency_resolution_error.rb +5 -7
  530. data/test/rubygems/test_gem_doctor.rb +73 -47
  531. data/test/rubygems/test_gem_ext_builder.rb +115 -109
  532. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/.gitignore +1 -0
  533. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/Cargo.lock +243 -0
  534. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/Cargo.toml +10 -0
  535. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/build.rb +21 -0
  536. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/custom_name.gemspec +10 -0
  537. data/test/rubygems/test_gem_ext_cargo_builder/custom_name/src/lib.rs +27 -0
  538. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/.gitignore +1 -0
  539. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/Cargo.lock +243 -0
  540. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/Cargo.toml +10 -0
  541. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/build.rb +21 -0
  542. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/rust_ruby_example.gemspec +8 -0
  543. data/test/rubygems/test_gem_ext_cargo_builder/rust_ruby_example/src/lib.rs +39 -0
  544. data/test/rubygems/test_gem_ext_cargo_builder.rb +178 -0
  545. data/test/rubygems/test_gem_ext_cargo_builder_link_flag_converter.rb +33 -0
  546. data/test/rubygems/test_gem_ext_cargo_builder_unit.rb +75 -0
  547. data/test/rubygems/test_gem_ext_cmake_builder.rb +31 -38
  548. data/test/rubygems/test_gem_ext_configure_builder.rb +23 -39
  549. data/test/rubygems/test_gem_ext_ext_conf_builder.rb +66 -82
  550. data/test/rubygems/test_gem_ext_rake_builder.rb +49 -32
  551. data/test/rubygems/test_gem_gem_runner.rb +51 -8
  552. data/test/rubygems/test_gem_gemcutter_utilities.rb +87 -76
  553. data/test/rubygems/test_gem_impossible_dependencies_error.rb +4 -6
  554. data/test/rubygems/test_gem_indexer.rb +62 -64
  555. data/test/rubygems/test_gem_install_update_options.rb +31 -21
  556. data/test/rubygems/test_gem_installer.rb +798 -491
  557. data/test/rubygems/test_gem_local_remote_options.rb +11 -13
  558. data/test/rubygems/test_gem_name_tuple.rb +4 -6
  559. data/test/rubygems/test_gem_package.rb +318 -283
  560. data/test/rubygems/test_gem_package_old.rb +18 -20
  561. data/test/rubygems/test_gem_package_tar_header.rb +64 -47
  562. data/test/rubygems/test_gem_package_tar_reader.rb +8 -10
  563. data/test/rubygems/test_gem_package_tar_reader_entry.rb +18 -20
  564. data/test/rubygems/test_gem_package_tar_writer.rb +96 -100
  565. data/test/rubygems/test_gem_package_task.rb +58 -25
  566. data/test/rubygems/test_gem_path_support.rb +15 -21
  567. data/test/rubygems/test_gem_platform.rb +354 -201
  568. data/test/rubygems/test_gem_rdoc.rb +19 -21
  569. data/test/rubygems/test_gem_remote_fetcher.rb +376 -337
  570. data/test/rubygems/test_gem_request.rb +114 -86
  571. data/test/rubygems/test_gem_request_connection_pools.rb +30 -34
  572. data/test/rubygems/test_gem_request_set.rb +123 -124
  573. data/test/rubygems/test_gem_request_set_gem_dependency_api.rb +222 -226
  574. data/test/rubygems/test_gem_request_set_lockfile.rb +93 -95
  575. data/test/rubygems/test_gem_request_set_lockfile_parser.rb +68 -70
  576. data/test/rubygems/test_gem_request_set_lockfile_tokenizer.rb +136 -138
  577. data/test/rubygems/test_gem_requirement.rb +133 -54
  578. data/test/rubygems/test_gem_resolver.rb +149 -117
  579. data/test/rubygems/test_gem_resolver_activation_request.rb +6 -8
  580. data/test/rubygems/test_gem_resolver_api_set.rb +79 -78
  581. data/test/rubygems/test_gem_resolver_api_specification.rb +47 -49
  582. data/test/rubygems/test_gem_resolver_best_set.rb +43 -22
  583. data/test/rubygems/test_gem_resolver_composed_set.rb +1 -3
  584. data/test/rubygems/test_gem_resolver_conflict.rb +12 -14
  585. data/test/rubygems/test_gem_resolver_dependency_request.rb +15 -17
  586. data/test/rubygems/test_gem_resolver_git_set.rb +22 -24
  587. data/test/rubygems/test_gem_resolver_git_specification.rb +22 -23
  588. data/test/rubygems/test_gem_resolver_index_set.rb +14 -16
  589. data/test/rubygems/test_gem_resolver_index_specification.rb +16 -18
  590. data/test/rubygems/test_gem_resolver_installed_specification.rb +5 -7
  591. data/test/rubygems/test_gem_resolver_installer_set.rb +104 -44
  592. data/test/rubygems/test_gem_resolver_local_specification.rb +7 -9
  593. data/test/rubygems/test_gem_resolver_lock_set.rb +15 -17
  594. data/test/rubygems/test_gem_resolver_lock_specification.rb +17 -19
  595. data/test/rubygems/test_gem_resolver_requirement_list.rb +1 -3
  596. data/test/rubygems/test_gem_resolver_specification.rb +8 -12
  597. data/test/rubygems/test_gem_resolver_vendor_set.rb +9 -11
  598. data/test/rubygems/test_gem_resolver_vendor_specification.rb +10 -12
  599. data/test/rubygems/test_gem_security.rb +104 -78
  600. data/test/rubygems/test_gem_security_policy.rb +92 -97
  601. data/test/rubygems/test_gem_security_signer.rb +45 -47
  602. data/test/rubygems/test_gem_security_trust_dir.rb +12 -14
  603. data/test/rubygems/test_gem_silent_ui.rb +46 -42
  604. data/test/rubygems/test_gem_source.rb +53 -52
  605. data/test/rubygems/test_gem_source_fetch_problem.rb +16 -8
  606. data/test/rubygems/test_gem_source_git.rb +69 -69
  607. data/test/rubygems/test_gem_source_installed.rb +16 -18
  608. data/test/rubygems/test_gem_source_list.rb +5 -6
  609. data/test/rubygems/test_gem_source_local.rb +15 -17
  610. data/test/rubygems/test_gem_source_lock.rb +31 -33
  611. data/test/rubygems/test_gem_source_specific_file.rb +18 -20
  612. data/test/rubygems/test_gem_source_subpath_problem.rb +49 -0
  613. data/test/rubygems/test_gem_source_vendor.rb +13 -15
  614. data/test/rubygems/test_gem_spec_fetcher.rb +74 -67
  615. data/test/rubygems/test_gem_specification.rb +888 -1024
  616. data/test/rubygems/test_gem_stream_ui.rb +23 -23
  617. data/test/rubygems/test_gem_stub_specification.rb +39 -57
  618. data/test/rubygems/test_gem_text.rb +8 -4
  619. data/test/rubygems/test_gem_uninstaller.rb +233 -104
  620. data/test/rubygems/test_gem_unsatisfiable_dependency_error.rb +3 -5
  621. data/test/rubygems/test_gem_uri.rb +39 -0
  622. data/test/rubygems/test_gem_uri_formatter.rb +14 -16
  623. data/test/rubygems/test_gem_util.rb +37 -35
  624. data/test/rubygems/test_gem_validator.rb +10 -12
  625. data/test/rubygems/test_gem_version.rb +31 -31
  626. data/test/rubygems/test_gem_version_option.rb +16 -18
  627. data/test/rubygems/test_kernel.rb +46 -60
  628. data/test/rubygems/test_project_sanity.rb +8 -3
  629. data/test/rubygems/test_remote_fetch_error.rb +7 -9
  630. data/test/rubygems/test_require.rb +240 -167
  631. data/test/rubygems/test_rubygems.rb +74 -0
  632. data/{lib/rubygems/test_utilities.rb → test/rubygems/utilities.rb} +30 -39
  633. data/test/test_changelog_generator.rb +17 -0
  634. metadata +158 -146
  635. data/.bundle/config +0 -2
  636. data/.rubocop.yml +0 -91
  637. data/Gemfile +0 -8
  638. data/Gemfile.lock +0 -43
  639. data/Rakefile +0 -428
  640. data/bundler/CODE_OF_CONDUCT.md +0 -136
  641. data/bundler/lib/bundler/gemdeps.rb +0 -29
  642. data/bundler/lib/bundler/psyched_yaml.rb +0 -37
  643. data/bundler/lib/bundler/vendor/connection_pool/lib/connection_pool/monotonic_time.rb +0 -66
  644. data/bundler/lib/bundler/vendor/molinillo/lib/molinillo/compatibility.rb +0 -26
  645. data/bundler/man/bundle-add.1.txt +0 -58
  646. data/bundler/man/bundle-binstubs.1.txt +0 -48
  647. data/bundler/man/bundle-cache.1.txt +0 -78
  648. data/bundler/man/bundle-check.1.txt +0 -33
  649. data/bundler/man/bundle-clean.1.txt +0 -26
  650. data/bundler/man/bundle-config.1.txt +0 -528
  651. data/bundler/man/bundle-doctor.1.txt +0 -44
  652. data/bundler/man/bundle-exec.1.txt +0 -178
  653. data/bundler/man/bundle-gem.1.txt +0 -91
  654. data/bundler/man/bundle-info.1.txt +0 -21
  655. data/bundler/man/bundle-init.1.txt +0 -34
  656. data/bundler/man/bundle-inject.1.txt +0 -32
  657. data/bundler/man/bundle-install.1.txt +0 -401
  658. data/bundler/man/bundle-list.1.txt +0 -43
  659. data/bundler/man/bundle-lock.1.txt +0 -93
  660. data/bundler/man/bundle-open.1.txt +0 -29
  661. data/bundler/man/bundle-outdated.1.txt +0 -131
  662. data/bundler/man/bundle-platform.1.txt +0 -57
  663. data/bundler/man/bundle-pristine.1.txt +0 -44
  664. data/bundler/man/bundle-remove.1.txt +0 -34
  665. data/bundler/man/bundle-show.1.txt +0 -27
  666. data/bundler/man/bundle-update.1.txt +0 -390
  667. data/bundler/man/bundle-viz.1.txt +0 -39
  668. data/bundler/man/bundle.1.txt +0 -116
  669. data/bundler/man/gemfile.5.txt +0 -649
  670. data/lib/rubygems/psych_additions.rb +0 -10
  671. data/lib/rubygems/server.rb +0 -879
  672. data/lib/rubygems/source_local.rb +0 -7
  673. data/lib/rubygems/source_specific_file.rb +0 -6
  674. data/lib/rubygems/syck_hack.rb +0 -79
  675. data/lib/rubygems/uri_parser.rb +0 -36
  676. data/lib/rubygems/uri_parsing.rb +0 -23
  677. data/lib/ubygems.rb +0 -14
  678. data/test/rubygems/bogussources.rb +0 -9
  679. data/test/rubygems/data/null-type.gemspec.rz +0 -0
  680. data/test/rubygems/test_gem_server.rb +0 -612
  681. data/tmp/.keep +0 -0
  682. data/util/CL2notes +0 -55
  683. data/util/bisect +0 -10
  684. data/util/ci.sh +0 -62
  685. data/util/cops/deprecations.rb +0 -52
  686. data/util/create_certs.rb +0 -171
  687. data/util/create_certs.sh +0 -27
  688. data/util/create_encrypted_key.rb +0 -16
  689. data/util/generate_spdx_license_list.rb +0 -63
  690. data/util/patch_with_prs.rb +0 -77
  691. data/util/rubocop +0 -8
  692. data/util/update_bundled_ca_certificates.rb +0 -137
  693. data/util/update_changelog.rb +0 -64
  694. /data/bundler/{man/bundle-check.ronn → lib/bundler/man/bundle-check.1.ronn} +0 -0
  695. /data/bundler/{man/bundle-clean.ronn → lib/bundler/man/bundle-clean.1.ronn} +0 -0
  696. /data/bundler/{man/bundle-doctor.ronn → lib/bundler/man/bundle-doctor.1.ronn} +0 -0
  697. /data/bundler/{man/bundle-info.ronn → lib/bundler/man/bundle-info.1.ronn} +0 -0
  698. /data/bundler/{man/bundle-init.ronn → lib/bundler/man/bundle-init.1.ronn} +0 -0
  699. /data/bundler/{man/bundle-lock.ronn → lib/bundler/man/bundle-lock.1.ronn} +0 -0
  700. /data/bundler/{man/bundle-open.ronn → lib/bundler/man/bundle-open.1.ronn} +0 -0
  701. /data/bundler/{man/bundle-pristine.ronn → lib/bundler/man/bundle-pristine.1.ronn} +0 -0
  702. /data/bundler/{man/bundle-remove.ronn → lib/bundler/man/bundle-remove.1.ronn} +0 -0
  703. /data/bundler/{man/bundle-show.ronn → lib/bundler/man/bundle-show.1.ronn} +0 -0
@@ -0,0 +1,2303 @@
1
+ # frozen_string_literal: true
2
+ #
3
+ # optparse.rb - command-line option analysis with the Gem::OptionParser class.
4
+ #
5
+ # Author:: Nobu Nakada
6
+ # Documentation:: Nobu Nakada and Gavin Sinclair.
7
+ #
8
+ # See Gem::OptionParser for documentation.
9
+ #
10
+
11
+
12
+ #--
13
+ # == Developer Documentation (not for RDoc output)
14
+ #
15
+ # === Class tree
16
+ #
17
+ # - Gem::OptionParser:: front end
18
+ # - Gem::OptionParser::Switch:: each switches
19
+ # - Gem::OptionParser::List:: options list
20
+ # - Gem::OptionParser::ParseError:: errors on parsing
21
+ # - Gem::OptionParser::AmbiguousOption
22
+ # - Gem::OptionParser::NeedlessArgument
23
+ # - Gem::OptionParser::MissingArgument
24
+ # - Gem::OptionParser::InvalidOption
25
+ # - Gem::OptionParser::InvalidArgument
26
+ # - Gem::OptionParser::AmbiguousArgument
27
+ #
28
+ # === Object relationship diagram
29
+ #
30
+ # +--------------+
31
+ # | Gem::OptionParser |<>-----+
32
+ # +--------------+ | +--------+
33
+ # | ,-| Switch |
34
+ # on_head -------->+---------------+ / +--------+
35
+ # accept/reject -->| List |<|>-
36
+ # | |<|>- +----------+
37
+ # on ------------->+---------------+ `-| argument |
38
+ # : : | class |
39
+ # +---------------+ |==========|
40
+ # on_tail -------->| | |pattern |
41
+ # +---------------+ |----------|
42
+ # Gem::OptionParser.accept ->| DefaultList | |converter |
43
+ # reject |(shared between| +----------+
44
+ # | all instances)|
45
+ # +---------------+
46
+ #
47
+ #++
48
+ #
49
+ # == Gem::OptionParser
50
+ #
51
+ # === New to \Gem::OptionParser?
52
+ #
53
+ # See the {Tutorial}[optparse/tutorial.rdoc].
54
+ #
55
+ # === Introduction
56
+ #
57
+ # Gem::OptionParser is a class for command-line option analysis. It is much more
58
+ # advanced, yet also easier to use, than GetoptLong, and is a more Ruby-oriented
59
+ # solution.
60
+ #
61
+ # === Features
62
+ #
63
+ # 1. The argument specification and the code to handle it are written in the
64
+ # same place.
65
+ # 2. It can output an option summary; you don't need to maintain this string
66
+ # separately.
67
+ # 3. Optional and mandatory arguments are specified very gracefully.
68
+ # 4. Arguments can be automatically converted to a specified class.
69
+ # 5. Arguments can be restricted to a certain set.
70
+ #
71
+ # All of these features are demonstrated in the examples below. See
72
+ # #make_switch for full documentation.
73
+ #
74
+ # === Minimal example
75
+ #
76
+ # require 'rubygems/optparse/lib/optparse'
77
+ #
78
+ # options = {}
79
+ # Gem::OptionParser.new do |parser|
80
+ # parser.banner = "Usage: example.rb [options]"
81
+ #
82
+ # parser.on("-v", "--[no-]verbose", "Run verbosely") do |v|
83
+ # options[:verbose] = v
84
+ # end
85
+ # end.parse!
86
+ #
87
+ # p options
88
+ # p ARGV
89
+ #
90
+ # === Generating Help
91
+ #
92
+ # Gem::OptionParser can be used to automatically generate help for the commands you
93
+ # write:
94
+ #
95
+ # require 'rubygems/optparse/lib/optparse'
96
+ #
97
+ # Options = Struct.new(:name)
98
+ #
99
+ # class Parser
100
+ # def self.parse(options)
101
+ # args = Options.new("world")
102
+ #
103
+ # opt_parser = Gem::OptionParser.new do |parser|
104
+ # parser.banner = "Usage: example.rb [options]"
105
+ #
106
+ # parser.on("-nNAME", "--name=NAME", "Name to say hello to") do |n|
107
+ # args.name = n
108
+ # end
109
+ #
110
+ # parser.on("-h", "--help", "Prints this help") do
111
+ # puts parser
112
+ # exit
113
+ # end
114
+ # end
115
+ #
116
+ # opt_parser.parse!(options)
117
+ # return args
118
+ # end
119
+ # end
120
+ # options = Parser.parse %w[--help]
121
+ #
122
+ # #=>
123
+ # # Usage: example.rb [options]
124
+ # # -n, --name=NAME Name to say hello to
125
+ # # -h, --help Prints this help
126
+ #
127
+ # === Required Arguments
128
+ #
129
+ # For options that require an argument, option specification strings may include an
130
+ # option name in all caps. If an option is used without the required argument,
131
+ # an exception will be raised.
132
+ #
133
+ # require 'rubygems/optparse/lib/optparse'
134
+ #
135
+ # options = {}
136
+ # Gem::OptionParser.new do |parser|
137
+ # parser.on("-r", "--require LIBRARY",
138
+ # "Require the LIBRARY before executing your script") do |lib|
139
+ # puts "You required #{lib}!"
140
+ # end
141
+ # end.parse!
142
+ #
143
+ # Used:
144
+ #
145
+ # $ ruby optparse-test.rb -r
146
+ # optparse-test.rb:9:in `<main>': missing argument: -r (Gem::OptionParser::MissingArgument)
147
+ # $ ruby optparse-test.rb -r my-library
148
+ # You required my-library!
149
+ #
150
+ # === Type Coercion
151
+ #
152
+ # Gem::OptionParser supports the ability to coerce command line arguments
153
+ # into objects for us.
154
+ #
155
+ # Gem::OptionParser comes with a few ready-to-use kinds of type
156
+ # coercion. They are:
157
+ #
158
+ # - Date -- Anything accepted by +Date.parse+
159
+ # - DateTime -- Anything accepted by +DateTime.parse+
160
+ # - Time -- Anything accepted by +Time.httpdate+ or +Time.parse+
161
+ # - URI -- Anything accepted by +URI.parse+
162
+ # - Shellwords -- Anything accepted by +Shellwords.shellwords+
163
+ # - String -- Any non-empty string
164
+ # - Integer -- Any integer. Will convert octal. (e.g. 124, -3, 040)
165
+ # - Float -- Any float. (e.g. 10, 3.14, -100E+13)
166
+ # - Numeric -- Any integer, float, or rational (1, 3.4, 1/3)
167
+ # - DecimalInteger -- Like +Integer+, but no octal format.
168
+ # - OctalInteger -- Like +Integer+, but no decimal format.
169
+ # - DecimalNumeric -- Decimal integer or float.
170
+ # - TrueClass -- Accepts '+, yes, true, -, no, false' and
171
+ # defaults as +true+
172
+ # - FalseClass -- Same as +TrueClass+, but defaults to +false+
173
+ # - Array -- Strings separated by ',' (e.g. 1,2,3)
174
+ # - Regexp -- Regular expressions. Also includes options.
175
+ #
176
+ # We can also add our own coercions, which we will cover below.
177
+ #
178
+ # ==== Using Built-in Conversions
179
+ #
180
+ # As an example, the built-in +Time+ conversion is used. The other built-in
181
+ # conversions behave in the same way.
182
+ # Gem::OptionParser will attempt to parse the argument
183
+ # as a +Time+. If it succeeds, that time will be passed to the
184
+ # handler block. Otherwise, an exception will be raised.
185
+ #
186
+ # require 'rubygems/optparse/lib/optparse'
187
+ # require 'rubygems/optparse/lib/optparse/time'
188
+ # Gem::OptionParser.new do |parser|
189
+ # parser.on("-t", "--time [TIME]", Time, "Begin execution at given time") do |time|
190
+ # p time
191
+ # end
192
+ # end.parse!
193
+ #
194
+ # Used:
195
+ #
196
+ # $ ruby optparse-test.rb -t nonsense
197
+ # ... invalid argument: -t nonsense (Gem::OptionParser::InvalidArgument)
198
+ # $ ruby optparse-test.rb -t 10-11-12
199
+ # 2010-11-12 00:00:00 -0500
200
+ # $ ruby optparse-test.rb -t 9:30
201
+ # 2014-08-13 09:30:00 -0400
202
+ #
203
+ # ==== Creating Custom Conversions
204
+ #
205
+ # The +accept+ method on Gem::OptionParser may be used to create converters.
206
+ # It specifies which conversion block to call whenever a class is specified.
207
+ # The example below uses it to fetch a +User+ object before the +on+ handler receives it.
208
+ #
209
+ # require 'rubygems/optparse/lib/optparse'
210
+ #
211
+ # User = Struct.new(:id, :name)
212
+ #
213
+ # def find_user id
214
+ # not_found = ->{ raise "No User Found for id #{id}" }
215
+ # [ User.new(1, "Sam"),
216
+ # User.new(2, "Gandalf") ].find(not_found) do |u|
217
+ # u.id == id
218
+ # end
219
+ # end
220
+ #
221
+ # op = Gem::OptionParser.new
222
+ # op.accept(User) do |user_id|
223
+ # find_user user_id.to_i
224
+ # end
225
+ #
226
+ # op.on("--user ID", User) do |user|
227
+ # puts user
228
+ # end
229
+ #
230
+ # op.parse!
231
+ #
232
+ # Used:
233
+ #
234
+ # $ ruby optparse-test.rb --user 1
235
+ # #<struct User id=1, name="Sam">
236
+ # $ ruby optparse-test.rb --user 2
237
+ # #<struct User id=2, name="Gandalf">
238
+ # $ ruby optparse-test.rb --user 3
239
+ # optparse-test.rb:15:in `block in find_user': No User Found for id 3 (RuntimeError)
240
+ #
241
+ # === Store options to a Hash
242
+ #
243
+ # The +into+ option of +order+, +parse+ and so on methods stores command line options into a Hash.
244
+ #
245
+ # require 'rubygems/optparse/lib/optparse'
246
+ #
247
+ # options = {}
248
+ # Gem::OptionParser.new do |parser|
249
+ # parser.on('-a')
250
+ # parser.on('-b NUM', Integer)
251
+ # parser.on('-v', '--verbose')
252
+ # end.parse!(into: options)
253
+ #
254
+ # p options
255
+ #
256
+ # Used:
257
+ #
258
+ # $ ruby optparse-test.rb -a
259
+ # {:a=>true}
260
+ # $ ruby optparse-test.rb -a -v
261
+ # {:a=>true, :verbose=>true}
262
+ # $ ruby optparse-test.rb -a -b 100
263
+ # {:a=>true, :b=>100}
264
+ #
265
+ # === Complete example
266
+ #
267
+ # The following example is a complete Ruby program. You can run it and see the
268
+ # effect of specifying various options. This is probably the best way to learn
269
+ # the features of +optparse+.
270
+ #
271
+ # require 'rubygems/optparse/lib/optparse'
272
+ # require 'rubygems/optparse/lib/optparse/time'
273
+ # require 'ostruct'
274
+ # require 'pp'
275
+ #
276
+ # class OptparseExample
277
+ # Version = '1.0.0'
278
+ #
279
+ # CODES = %w[iso-2022-jp shift_jis euc-jp utf8 binary]
280
+ # CODE_ALIASES = { "jis" => "iso-2022-jp", "sjis" => "shift_jis" }
281
+ #
282
+ # class ScriptOptions
283
+ # attr_accessor :library, :inplace, :encoding, :transfer_type,
284
+ # :verbose, :extension, :delay, :time, :record_separator,
285
+ # :list
286
+ #
287
+ # def initialize
288
+ # self.library = []
289
+ # self.inplace = false
290
+ # self.encoding = "utf8"
291
+ # self.transfer_type = :auto
292
+ # self.verbose = false
293
+ # end
294
+ #
295
+ # def define_options(parser)
296
+ # parser.banner = "Usage: example.rb [options]"
297
+ # parser.separator ""
298
+ # parser.separator "Specific options:"
299
+ #
300
+ # # add additional options
301
+ # perform_inplace_option(parser)
302
+ # delay_execution_option(parser)
303
+ # execute_at_time_option(parser)
304
+ # specify_record_separator_option(parser)
305
+ # list_example_option(parser)
306
+ # specify_encoding_option(parser)
307
+ # optional_option_argument_with_keyword_completion_option(parser)
308
+ # boolean_verbose_option(parser)
309
+ #
310
+ # parser.separator ""
311
+ # parser.separator "Common options:"
312
+ # # No argument, shows at tail. This will print an options summary.
313
+ # # Try it and see!
314
+ # parser.on_tail("-h", "--help", "Show this message") do
315
+ # puts parser
316
+ # exit
317
+ # end
318
+ # # Another typical switch to print the version.
319
+ # parser.on_tail("--version", "Show version") do
320
+ # puts Version
321
+ # exit
322
+ # end
323
+ # end
324
+ #
325
+ # def perform_inplace_option(parser)
326
+ # # Specifies an optional option argument
327
+ # parser.on("-i", "--inplace [EXTENSION]",
328
+ # "Edit ARGV files in place",
329
+ # "(make backup if EXTENSION supplied)") do |ext|
330
+ # self.inplace = true
331
+ # self.extension = ext || ''
332
+ # self.extension.sub!(/\A\.?(?=.)/, ".") # Ensure extension begins with dot.
333
+ # end
334
+ # end
335
+ #
336
+ # def delay_execution_option(parser)
337
+ # # Cast 'delay' argument to a Float.
338
+ # parser.on("--delay N", Float, "Delay N seconds before executing") do |n|
339
+ # self.delay = n
340
+ # end
341
+ # end
342
+ #
343
+ # def execute_at_time_option(parser)
344
+ # # Cast 'time' argument to a Time object.
345
+ # parser.on("-t", "--time [TIME]", Time, "Begin execution at given time") do |time|
346
+ # self.time = time
347
+ # end
348
+ # end
349
+ #
350
+ # def specify_record_separator_option(parser)
351
+ # # Cast to octal integer.
352
+ # parser.on("-F", "--irs [OCTAL]", Gem::OptionParser::OctalInteger,
353
+ # "Specify record separator (default \\0)") do |rs|
354
+ # self.record_separator = rs
355
+ # end
356
+ # end
357
+ #
358
+ # def list_example_option(parser)
359
+ # # List of arguments.
360
+ # parser.on("--list x,y,z", Array, "Example 'list' of arguments") do |list|
361
+ # self.list = list
362
+ # end
363
+ # end
364
+ #
365
+ # def specify_encoding_option(parser)
366
+ # # Keyword completion. We are specifying a specific set of arguments (CODES
367
+ # # and CODE_ALIASES - notice the latter is a Hash), and the user may provide
368
+ # # the shortest unambiguous text.
369
+ # code_list = (CODE_ALIASES.keys + CODES).join(', ')
370
+ # parser.on("--code CODE", CODES, CODE_ALIASES, "Select encoding",
371
+ # "(#{code_list})") do |encoding|
372
+ # self.encoding = encoding
373
+ # end
374
+ # end
375
+ #
376
+ # def optional_option_argument_with_keyword_completion_option(parser)
377
+ # # Optional '--type' option argument with keyword completion.
378
+ # parser.on("--type [TYPE]", [:text, :binary, :auto],
379
+ # "Select transfer type (text, binary, auto)") do |t|
380
+ # self.transfer_type = t
381
+ # end
382
+ # end
383
+ #
384
+ # def boolean_verbose_option(parser)
385
+ # # Boolean switch.
386
+ # parser.on("-v", "--[no-]verbose", "Run verbosely") do |v|
387
+ # self.verbose = v
388
+ # end
389
+ # end
390
+ # end
391
+ #
392
+ # #
393
+ # # Return a structure describing the options.
394
+ # #
395
+ # def parse(args)
396
+ # # The options specified on the command line will be collected in
397
+ # # *options*.
398
+ #
399
+ # @options = ScriptOptions.new
400
+ # @args = Gem::OptionParser.new do |parser|
401
+ # @options.define_options(parser)
402
+ # parser.parse!(args)
403
+ # end
404
+ # @options
405
+ # end
406
+ #
407
+ # attr_reader :parser, :options
408
+ # end # class OptparseExample
409
+ #
410
+ # example = OptparseExample.new
411
+ # options = example.parse(ARGV)
412
+ # pp options # example.options
413
+ # pp ARGV
414
+ #
415
+ # === Shell Completion
416
+ #
417
+ # For modern shells (e.g. bash, zsh, etc.), you can use shell
418
+ # completion for command line options.
419
+ #
420
+ # === Further documentation
421
+ #
422
+ # The above examples, along with the accompanying
423
+ # {Tutorial}[optparse/tutorial.rdoc],
424
+ # should be enough to learn how to use this class.
425
+ # If you have any questions, file a ticket at http://bugs.ruby-lang.org.
426
+ #
427
+ class Gem::OptionParser
428
+ Gem::OptionParser::Version = "0.2.0"
429
+
430
+ # :stopdoc:
431
+ NoArgument = [NO_ARGUMENT = :NONE, nil].freeze
432
+ RequiredArgument = [REQUIRED_ARGUMENT = :REQUIRED, true].freeze
433
+ OptionalArgument = [OPTIONAL_ARGUMENT = :OPTIONAL, false].freeze
434
+ # :startdoc:
435
+
436
+ #
437
+ # Keyword completion module. This allows partial arguments to be specified
438
+ # and resolved against a list of acceptable values.
439
+ #
440
+ module Completion
441
+ def self.regexp(key, icase)
442
+ Regexp.new('\A' + Regexp.quote(key).gsub(/\w+\b/, '\&\w*'), icase)
443
+ end
444
+
445
+ def self.candidate(key, icase = false, pat = nil, &block)
446
+ pat ||= Completion.regexp(key, icase)
447
+ candidates = []
448
+ block.call do |k, *v|
449
+ (if Regexp === k
450
+ kn = ""
451
+ k === key
452
+ else
453
+ kn = defined?(k.id2name) ? k.id2name : k
454
+ pat === kn
455
+ end) or next
456
+ v << k if v.empty?
457
+ candidates << [k, v, kn]
458
+ end
459
+ candidates
460
+ end
461
+
462
+ def candidate(key, icase = false, pat = nil)
463
+ Completion.candidate(key, icase, pat, &method(:each))
464
+ end
465
+
466
+ public
467
+ def complete(key, icase = false, pat = nil)
468
+ candidates = candidate(key, icase, pat, &method(:each)).sort_by {|k, v, kn| kn.size}
469
+ if candidates.size == 1
470
+ canon, sw, * = candidates[0]
471
+ elsif candidates.size > 1
472
+ canon, sw, cn = candidates.shift
473
+ candidates.each do |k, v, kn|
474
+ next if sw == v
475
+ if String === cn and String === kn
476
+ if cn.rindex(kn, 0)
477
+ canon, sw, cn = k, v, kn
478
+ next
479
+ elsif kn.rindex(cn, 0)
480
+ next
481
+ end
482
+ end
483
+ throw :ambiguous, key
484
+ end
485
+ end
486
+ if canon
487
+ block_given? or return key, *sw
488
+ yield(key, *sw)
489
+ end
490
+ end
491
+
492
+ def convert(opt = nil, val = nil, *)
493
+ val
494
+ end
495
+ end
496
+
497
+
498
+ #
499
+ # Map from option/keyword string to object with completion.
500
+ #
501
+ class OptionMap < Hash
502
+ include Completion
503
+ end
504
+
505
+
506
+ #
507
+ # Individual switch class. Not important to the user.
508
+ #
509
+ # Defined within Switch are several Switch-derived classes: NoArgument,
510
+ # RequiredArgument, etc.
511
+ #
512
+ class Switch
513
+ attr_reader :pattern, :conv, :short, :long, :arg, :desc, :block
514
+
515
+ #
516
+ # Guesses argument style from +arg+. Returns corresponding
517
+ # Gem::OptionParser::Switch class (OptionalArgument, etc.).
518
+ #
519
+ def self.guess(arg)
520
+ case arg
521
+ when ""
522
+ t = self
523
+ when /\A=?\[/
524
+ t = Switch::OptionalArgument
525
+ when /\A\s+\[/
526
+ t = Switch::PlacedArgument
527
+ else
528
+ t = Switch::RequiredArgument
529
+ end
530
+ self >= t or incompatible_argument_styles(arg, t)
531
+ t
532
+ end
533
+
534
+ def self.incompatible_argument_styles(arg, t)
535
+ raise(ArgumentError, "#{arg}: incompatible argument styles\n #{self}, #{t}",
536
+ ParseError.filter_backtrace(caller(2)))
537
+ end
538
+
539
+ def self.pattern
540
+ NilClass
541
+ end
542
+
543
+ def initialize(pattern = nil, conv = nil,
544
+ short = nil, long = nil, arg = nil,
545
+ desc = ([] if short or long), block = nil, &_block)
546
+ raise if Array === pattern
547
+ block ||= _block
548
+ @pattern, @conv, @short, @long, @arg, @desc, @block =
549
+ pattern, conv, short, long, arg, desc, block
550
+ end
551
+
552
+ #
553
+ # Parses +arg+ and returns rest of +arg+ and matched portion to the
554
+ # argument pattern. Yields when the pattern doesn't match substring.
555
+ #
556
+ def parse_arg(arg) # :nodoc:
557
+ pattern or return nil, [arg]
558
+ unless m = pattern.match(arg)
559
+ yield(InvalidArgument, arg)
560
+ return arg, []
561
+ end
562
+ if String === m
563
+ m = [s = m]
564
+ else
565
+ m = m.to_a
566
+ s = m[0]
567
+ return nil, m unless String === s
568
+ end
569
+ raise InvalidArgument, arg unless arg.rindex(s, 0)
570
+ return nil, m if s.length == arg.length
571
+ yield(InvalidArgument, arg) # didn't match whole arg
572
+ return arg[s.length..-1], m
573
+ end
574
+ private :parse_arg
575
+
576
+ #
577
+ # Parses argument, converts and returns +arg+, +block+ and result of
578
+ # conversion. Yields at semi-error condition instead of raising an
579
+ # exception.
580
+ #
581
+ def conv_arg(arg, val = []) # :nodoc:
582
+ if conv
583
+ val = conv.call(*val)
584
+ else
585
+ val = proc {|v| v}.call(*val)
586
+ end
587
+ return arg, block, val
588
+ end
589
+ private :conv_arg
590
+
591
+ #
592
+ # Produces the summary text. Each line of the summary is yielded to the
593
+ # block (without newline).
594
+ #
595
+ # +sdone+:: Already summarized short style options keyed hash.
596
+ # +ldone+:: Already summarized long style options keyed hash.
597
+ # +width+:: Width of left side (option part). In other words, the right
598
+ # side (description part) starts after +width+ columns.
599
+ # +max+:: Maximum width of left side -> the options are filled within
600
+ # +max+ columns.
601
+ # +indent+:: Prefix string indents all summarized lines.
602
+ #
603
+ def summarize(sdone = {}, ldone = {}, width = 1, max = width - 1, indent = "")
604
+ sopts, lopts = [], [], nil
605
+ @short.each {|s| sdone.fetch(s) {sopts << s}; sdone[s] = true} if @short
606
+ @long.each {|s| ldone.fetch(s) {lopts << s}; ldone[s] = true} if @long
607
+ return if sopts.empty? and lopts.empty? # completely hidden
608
+
609
+ left = [sopts.join(', ')]
610
+ right = desc.dup
611
+
612
+ while s = lopts.shift
613
+ l = left[-1].length + s.length
614
+ l += arg.length if left.size == 1 && arg
615
+ l < max or sopts.empty? or left << +''
616
+ left[-1] << (left[-1].empty? ? ' ' * 4 : ', ') << s
617
+ end
618
+
619
+ if arg
620
+ left[0] << (left[1] ? arg.sub(/\A(\[?)=/, '\1') + ',' : arg)
621
+ end
622
+ mlen = left.collect {|ss| ss.length}.max.to_i
623
+ while mlen > width and l = left.shift
624
+ mlen = left.collect {|ss| ss.length}.max.to_i if l.length == mlen
625
+ if l.length < width and (r = right[0]) and !r.empty?
626
+ l = l.to_s.ljust(width) + ' ' + r
627
+ right.shift
628
+ end
629
+ yield(indent + l)
630
+ end
631
+
632
+ while begin l = left.shift; r = right.shift; l or r end
633
+ l = l.to_s.ljust(width) + ' ' + r if r and !r.empty?
634
+ yield(indent + l)
635
+ end
636
+
637
+ self
638
+ end
639
+
640
+ def add_banner(to) # :nodoc:
641
+ unless @short or @long
642
+ s = desc.join
643
+ to << " [" + s + "]..." unless s.empty?
644
+ end
645
+ to
646
+ end
647
+
648
+ def match_nonswitch?(str) # :nodoc:
649
+ @pattern =~ str unless @short or @long
650
+ end
651
+
652
+ #
653
+ # Main name of the switch.
654
+ #
655
+ def switch_name
656
+ (long.first || short.first).sub(/\A-+(?:\[no-\])?/, '')
657
+ end
658
+
659
+ def compsys(sdone, ldone) # :nodoc:
660
+ sopts, lopts = [], []
661
+ @short.each {|s| sdone.fetch(s) {sopts << s}; sdone[s] = true} if @short
662
+ @long.each {|s| ldone.fetch(s) {lopts << s}; ldone[s] = true} if @long
663
+ return if sopts.empty? and lopts.empty? # completely hidden
664
+
665
+ (sopts+lopts).each do |opt|
666
+ # "(-x -c -r)-l[left justify]"
667
+ if /^--\[no-\](.+)$/ =~ opt
668
+ o = $1
669
+ yield("--#{o}", desc.join(""))
670
+ yield("--no-#{o}", desc.join(""))
671
+ else
672
+ yield("#{opt}", desc.join(""))
673
+ end
674
+ end
675
+ end
676
+
677
+ def pretty_print_contents(q) # :nodoc:
678
+ if @block
679
+ q.text ":" + @block.source_location.join(":") + ":"
680
+ first = false
681
+ else
682
+ first = true
683
+ end
684
+ [@short, @long].each do |list|
685
+ list.each do |opt|
686
+ if first
687
+ q.text ":"
688
+ first = false
689
+ end
690
+ q.breakable
691
+ q.text opt
692
+ end
693
+ end
694
+ end
695
+
696
+ def pretty_print(q) # :nodoc:
697
+ q.object_group(self) {pretty_print_contents(q)}
698
+ end
699
+
700
+ #
701
+ # Switch that takes no arguments.
702
+ #
703
+ class NoArgument < self
704
+
705
+ #
706
+ # Raises an exception if any arguments given.
707
+ #
708
+ def parse(arg, argv)
709
+ yield(NeedlessArgument, arg) if arg
710
+ conv_arg(arg)
711
+ end
712
+
713
+ def self.incompatible_argument_styles(*)
714
+ end
715
+
716
+ def self.pattern
717
+ Object
718
+ end
719
+
720
+ def pretty_head # :nodoc:
721
+ "NoArgument"
722
+ end
723
+ end
724
+
725
+ #
726
+ # Switch that takes an argument.
727
+ #
728
+ class RequiredArgument < self
729
+
730
+ #
731
+ # Raises an exception if argument is not present.
732
+ #
733
+ def parse(arg, argv)
734
+ unless arg
735
+ raise MissingArgument if argv.empty?
736
+ arg = argv.shift
737
+ end
738
+ conv_arg(*parse_arg(arg, &method(:raise)))
739
+ end
740
+
741
+ def pretty_head # :nodoc:
742
+ "Required"
743
+ end
744
+ end
745
+
746
+ #
747
+ # Switch that can omit argument.
748
+ #
749
+ class OptionalArgument < self
750
+
751
+ #
752
+ # Parses argument if given, or uses default value.
753
+ #
754
+ def parse(arg, argv, &error)
755
+ if arg
756
+ conv_arg(*parse_arg(arg, &error))
757
+ else
758
+ conv_arg(arg)
759
+ end
760
+ end
761
+
762
+ def pretty_head # :nodoc:
763
+ "Optional"
764
+ end
765
+ end
766
+
767
+ #
768
+ # Switch that takes an argument, which does not begin with '-'.
769
+ #
770
+ class PlacedArgument < self
771
+
772
+ #
773
+ # Returns nil if argument is not present or begins with '-'.
774
+ #
775
+ def parse(arg, argv, &error)
776
+ if !(val = arg) and (argv.empty? or /\A-/ =~ (val = argv[0]))
777
+ return nil, block, nil
778
+ end
779
+ opt = (val = parse_arg(val, &error))[1]
780
+ val = conv_arg(*val)
781
+ if opt and !arg
782
+ argv.shift
783
+ else
784
+ val[0] = nil
785
+ end
786
+ val
787
+ end
788
+
789
+ def pretty_head # :nodoc:
790
+ "Placed"
791
+ end
792
+ end
793
+ end
794
+
795
+ #
796
+ # Simple option list providing mapping from short and/or long option
797
+ # string to Gem::OptionParser::Switch and mapping from acceptable argument to
798
+ # matching pattern and converter pair. Also provides summary feature.
799
+ #
800
+ class List
801
+ # Map from acceptable argument types to pattern and converter pairs.
802
+ attr_reader :atype
803
+
804
+ # Map from short style option switches to actual switch objects.
805
+ attr_reader :short
806
+
807
+ # Map from long style option switches to actual switch objects.
808
+ attr_reader :long
809
+
810
+ # List of all switches and summary string.
811
+ attr_reader :list
812
+
813
+ #
814
+ # Just initializes all instance variables.
815
+ #
816
+ def initialize
817
+ @atype = {}
818
+ @short = OptionMap.new
819
+ @long = OptionMap.new
820
+ @list = []
821
+ end
822
+
823
+ def pretty_print(q) # :nodoc:
824
+ q.group(1, "(", ")") do
825
+ @list.each do |sw|
826
+ next unless Switch === sw
827
+ q.group(1, "(" + sw.pretty_head, ")") do
828
+ sw.pretty_print_contents(q)
829
+ end
830
+ end
831
+ end
832
+ end
833
+
834
+ #
835
+ # See Gem::OptionParser.accept.
836
+ #
837
+ def accept(t, pat = /.*/m, &block)
838
+ if pat
839
+ pat.respond_to?(:match) or
840
+ raise TypeError, "has no `match'", ParseError.filter_backtrace(caller(2))
841
+ else
842
+ pat = t if t.respond_to?(:match)
843
+ end
844
+ unless block
845
+ block = pat.method(:convert).to_proc if pat.respond_to?(:convert)
846
+ end
847
+ @atype[t] = [pat, block]
848
+ end
849
+
850
+ #
851
+ # See Gem::OptionParser.reject.
852
+ #
853
+ def reject(t)
854
+ @atype.delete(t)
855
+ end
856
+
857
+ #
858
+ # Adds +sw+ according to +sopts+, +lopts+ and +nlopts+.
859
+ #
860
+ # +sw+:: Gem::OptionParser::Switch instance to be added.
861
+ # +sopts+:: Short style option list.
862
+ # +lopts+:: Long style option list.
863
+ # +nlopts+:: Negated long style options list.
864
+ #
865
+ def update(sw, sopts, lopts, nsw = nil, nlopts = nil) # :nodoc:
866
+ sopts.each {|o| @short[o] = sw} if sopts
867
+ lopts.each {|o| @long[o] = sw} if lopts
868
+ nlopts.each {|o| @long[o] = nsw} if nsw and nlopts
869
+ used = @short.invert.update(@long.invert)
870
+ @list.delete_if {|o| Switch === o and !used[o]}
871
+ end
872
+ private :update
873
+
874
+ #
875
+ # Inserts +switch+ at the head of the list, and associates short, long
876
+ # and negated long options. Arguments are:
877
+ #
878
+ # +switch+:: Gem::OptionParser::Switch instance to be inserted.
879
+ # +short_opts+:: List of short style options.
880
+ # +long_opts+:: List of long style options.
881
+ # +nolong_opts+:: List of long style options with "no-" prefix.
882
+ #
883
+ # prepend(switch, short_opts, long_opts, nolong_opts)
884
+ #
885
+ def prepend(*args)
886
+ update(*args)
887
+ @list.unshift(args[0])
888
+ end
889
+
890
+ #
891
+ # Appends +switch+ at the tail of the list, and associates short, long
892
+ # and negated long options. Arguments are:
893
+ #
894
+ # +switch+:: Gem::OptionParser::Switch instance to be inserted.
895
+ # +short_opts+:: List of short style options.
896
+ # +long_opts+:: List of long style options.
897
+ # +nolong_opts+:: List of long style options with "no-" prefix.
898
+ #
899
+ # append(switch, short_opts, long_opts, nolong_opts)
900
+ #
901
+ def append(*args)
902
+ update(*args)
903
+ @list.push(args[0])
904
+ end
905
+
906
+ #
907
+ # Searches +key+ in +id+ list. The result is returned or yielded if a
908
+ # block is given. If it isn't found, nil is returned.
909
+ #
910
+ def search(id, key)
911
+ if list = __send__(id)
912
+ val = list.fetch(key) {return nil}
913
+ block_given? ? yield(val) : val
914
+ end
915
+ end
916
+
917
+ #
918
+ # Searches list +id+ for +opt+ and the optional patterns for completion
919
+ # +pat+. If +icase+ is true, the search is case insensitive. The result
920
+ # is returned or yielded if a block is given. If it isn't found, nil is
921
+ # returned.
922
+ #
923
+ def complete(id, opt, icase = false, *pat, &block)
924
+ __send__(id).complete(opt, icase, *pat, &block)
925
+ end
926
+
927
+ def get_candidates(id)
928
+ yield __send__(id).keys
929
+ end
930
+
931
+ #
932
+ # Iterates over each option, passing the option to the +block+.
933
+ #
934
+ def each_option(&block)
935
+ list.each(&block)
936
+ end
937
+
938
+ #
939
+ # Creates the summary table, passing each line to the +block+ (without
940
+ # newline). The arguments +args+ are passed along to the summarize
941
+ # method which is called on every option.
942
+ #
943
+ def summarize(*args, &block)
944
+ sum = []
945
+ list.reverse_each do |opt|
946
+ if opt.respond_to?(:summarize) # perhaps Gem::OptionParser::Switch
947
+ s = []
948
+ opt.summarize(*args) {|l| s << l}
949
+ sum.concat(s.reverse)
950
+ elsif !opt or opt.empty?
951
+ sum << ""
952
+ elsif opt.respond_to?(:each_line)
953
+ sum.concat([*opt.each_line].reverse)
954
+ else
955
+ sum.concat([*opt.each].reverse)
956
+ end
957
+ end
958
+ sum.reverse_each(&block)
959
+ end
960
+
961
+ def add_banner(to) # :nodoc:
962
+ list.each do |opt|
963
+ if opt.respond_to?(:add_banner)
964
+ opt.add_banner(to)
965
+ end
966
+ end
967
+ to
968
+ end
969
+
970
+ def compsys(*args, &block) # :nodoc:
971
+ list.each do |opt|
972
+ if opt.respond_to?(:compsys)
973
+ opt.compsys(*args, &block)
974
+ end
975
+ end
976
+ end
977
+ end
978
+
979
+ #
980
+ # Hash with completion search feature. See Gem::OptionParser::Completion.
981
+ #
982
+ class CompletingHash < Hash
983
+ include Completion
984
+
985
+ #
986
+ # Completion for hash key.
987
+ #
988
+ def match(key)
989
+ *values = fetch(key) {
990
+ raise AmbiguousArgument, catch(:ambiguous) {return complete(key)}
991
+ }
992
+ return key, *values
993
+ end
994
+ end
995
+
996
+ # :stopdoc:
997
+
998
+ #
999
+ # Enumeration of acceptable argument styles. Possible values are:
1000
+ #
1001
+ # NO_ARGUMENT:: The switch takes no arguments. (:NONE)
1002
+ # REQUIRED_ARGUMENT:: The switch requires an argument. (:REQUIRED)
1003
+ # OPTIONAL_ARGUMENT:: The switch requires an optional argument. (:OPTIONAL)
1004
+ #
1005
+ # Use like --switch=argument (long style) or -Xargument (short style). For
1006
+ # short style, only portion matched to argument pattern is treated as
1007
+ # argument.
1008
+ #
1009
+ ArgumentStyle = {}
1010
+ NoArgument.each {|el| ArgumentStyle[el] = Switch::NoArgument}
1011
+ RequiredArgument.each {|el| ArgumentStyle[el] = Switch::RequiredArgument}
1012
+ OptionalArgument.each {|el| ArgumentStyle[el] = Switch::OptionalArgument}
1013
+ ArgumentStyle.freeze
1014
+
1015
+ #
1016
+ # Switches common used such as '--', and also provides default
1017
+ # argument classes
1018
+ #
1019
+ DefaultList = List.new
1020
+ DefaultList.short['-'] = Switch::NoArgument.new {}
1021
+ DefaultList.long[''] = Switch::NoArgument.new {throw :terminate}
1022
+
1023
+
1024
+ COMPSYS_HEADER = <<'XXX' # :nodoc:
1025
+
1026
+ typeset -A opt_args
1027
+ local context state line
1028
+
1029
+ _arguments -s -S \
1030
+ XXX
1031
+
1032
+ def compsys(to, name = File.basename($0)) # :nodoc:
1033
+ to << "#compdef #{name}\n"
1034
+ to << COMPSYS_HEADER
1035
+ visit(:compsys, {}, {}) {|o, d|
1036
+ to << %Q[ "#{o}[#{d.gsub(/[\"\[\]]/, '\\\\\&')}]" \\\n]
1037
+ }
1038
+ to << " '*:file:_files' && return 0\n"
1039
+ end
1040
+
1041
+ #
1042
+ # Default options for ARGV, which never appear in option summary.
1043
+ #
1044
+ Officious = {}
1045
+
1046
+ #
1047
+ # --help
1048
+ # Shows option summary.
1049
+ #
1050
+ Officious['help'] = proc do |parser|
1051
+ Switch::NoArgument.new do |arg|
1052
+ puts parser.help
1053
+ exit
1054
+ end
1055
+ end
1056
+
1057
+ #
1058
+ # --*-completion-bash=WORD
1059
+ # Shows candidates for command line completion.
1060
+ #
1061
+ Officious['*-completion-bash'] = proc do |parser|
1062
+ Switch::RequiredArgument.new do |arg|
1063
+ puts parser.candidate(arg)
1064
+ exit
1065
+ end
1066
+ end
1067
+
1068
+ #
1069
+ # --*-completion-zsh[=NAME:FILE]
1070
+ # Creates zsh completion file.
1071
+ #
1072
+ Officious['*-completion-zsh'] = proc do |parser|
1073
+ Switch::OptionalArgument.new do |arg|
1074
+ parser.compsys(STDOUT, arg)
1075
+ exit
1076
+ end
1077
+ end
1078
+
1079
+ #
1080
+ # --version
1081
+ # Shows version string if Version is defined.
1082
+ #
1083
+ Officious['version'] = proc do |parser|
1084
+ Switch::OptionalArgument.new do |pkg|
1085
+ if pkg
1086
+ begin
1087
+ require 'rubygems/optparse/lib/optparse/version'
1088
+ rescue LoadError
1089
+ else
1090
+ show_version(*pkg.split(/,/)) or
1091
+ abort("#{parser.program_name}: no version found in package #{pkg}")
1092
+ exit
1093
+ end
1094
+ end
1095
+ v = parser.ver or abort("#{parser.program_name}: version unknown")
1096
+ puts v
1097
+ exit
1098
+ end
1099
+ end
1100
+
1101
+ # :startdoc:
1102
+
1103
+ #
1104
+ # Class methods
1105
+ #
1106
+
1107
+ #
1108
+ # Initializes a new instance and evaluates the optional block in context
1109
+ # of the instance. Arguments +args+ are passed to #new, see there for
1110
+ # description of parameters.
1111
+ #
1112
+ # This method is *deprecated*, its behavior corresponds to the older #new
1113
+ # method.
1114
+ #
1115
+ def self.with(*args, &block)
1116
+ opts = new(*args)
1117
+ opts.instance_eval(&block)
1118
+ opts
1119
+ end
1120
+
1121
+ #
1122
+ # Returns an incremented value of +default+ according to +arg+.
1123
+ #
1124
+ def self.inc(arg, default = nil)
1125
+ case arg
1126
+ when Integer
1127
+ arg.nonzero?
1128
+ when nil
1129
+ default.to_i + 1
1130
+ end
1131
+ end
1132
+ def inc(*args)
1133
+ self.class.inc(*args)
1134
+ end
1135
+
1136
+ #
1137
+ # Initializes the instance and yields itself if called with a block.
1138
+ #
1139
+ # +banner+:: Banner message.
1140
+ # +width+:: Summary width.
1141
+ # +indent+:: Summary indent.
1142
+ #
1143
+ def initialize(banner = nil, width = 32, indent = ' ' * 4)
1144
+ @stack = [DefaultList, List.new, List.new]
1145
+ @program_name = nil
1146
+ @banner = banner
1147
+ @summary_width = width
1148
+ @summary_indent = indent
1149
+ @default_argv = ARGV
1150
+ @require_exact = false
1151
+ add_officious
1152
+ yield self if block_given?
1153
+ end
1154
+
1155
+ def add_officious # :nodoc:
1156
+ list = base()
1157
+ Officious.each do |opt, block|
1158
+ list.long[opt] ||= block.call(self)
1159
+ end
1160
+ end
1161
+
1162
+ #
1163
+ # Terminates option parsing. Optional parameter +arg+ is a string pushed
1164
+ # back to be the first non-option argument.
1165
+ #
1166
+ def terminate(arg = nil)
1167
+ self.class.terminate(arg)
1168
+ end
1169
+ def self.terminate(arg = nil)
1170
+ throw :terminate, arg
1171
+ end
1172
+
1173
+ @stack = [DefaultList]
1174
+ def self.top() DefaultList end
1175
+
1176
+ #
1177
+ # Directs to accept specified class +t+. The argument string is passed to
1178
+ # the block in which it should be converted to the desired class.
1179
+ #
1180
+ # +t+:: Argument class specifier, any object including Class.
1181
+ # +pat+:: Pattern for argument, defaults to +t+ if it responds to match.
1182
+ #
1183
+ # accept(t, pat, &block)
1184
+ #
1185
+ def accept(*args, &blk) top.accept(*args, &blk) end
1186
+ #
1187
+ # See #accept.
1188
+ #
1189
+ def self.accept(*args, &blk) top.accept(*args, &blk) end
1190
+
1191
+ #
1192
+ # Directs to reject specified class argument.
1193
+ #
1194
+ # +t+:: Argument class specifier, any object including Class.
1195
+ #
1196
+ # reject(t)
1197
+ #
1198
+ def reject(*args, &blk) top.reject(*args, &blk) end
1199
+ #
1200
+ # See #reject.
1201
+ #
1202
+ def self.reject(*args, &blk) top.reject(*args, &blk) end
1203
+
1204
+ #
1205
+ # Instance methods
1206
+ #
1207
+
1208
+ # Heading banner preceding summary.
1209
+ attr_writer :banner
1210
+
1211
+ # Program name to be emitted in error message and default banner,
1212
+ # defaults to $0.
1213
+ attr_writer :program_name
1214
+
1215
+ # Width for option list portion of summary. Must be Numeric.
1216
+ attr_accessor :summary_width
1217
+
1218
+ # Indentation for summary. Must be String (or have + String method).
1219
+ attr_accessor :summary_indent
1220
+
1221
+ # Strings to be parsed in default.
1222
+ attr_accessor :default_argv
1223
+
1224
+ # Whether to require that options match exactly (disallows providing
1225
+ # abbreviated long option as short option).
1226
+ attr_accessor :require_exact
1227
+
1228
+ #
1229
+ # Heading banner preceding summary.
1230
+ #
1231
+ def banner
1232
+ unless @banner
1233
+ @banner = +"Usage: #{program_name} [options]"
1234
+ visit(:add_banner, @banner)
1235
+ end
1236
+ @banner
1237
+ end
1238
+
1239
+ #
1240
+ # Program name to be emitted in error message and default banner, defaults
1241
+ # to $0.
1242
+ #
1243
+ def program_name
1244
+ @program_name || File.basename($0, '.*')
1245
+ end
1246
+
1247
+ # for experimental cascading :-)
1248
+ alias set_banner banner=
1249
+ alias set_program_name program_name=
1250
+ alias set_summary_width summary_width=
1251
+ alias set_summary_indent summary_indent=
1252
+
1253
+ # Version
1254
+ attr_writer :version
1255
+ # Release code
1256
+ attr_writer :release
1257
+
1258
+ #
1259
+ # Version
1260
+ #
1261
+ def version
1262
+ (defined?(@version) && @version) || (defined?(::Version) && ::Version)
1263
+ end
1264
+
1265
+ #
1266
+ # Release code
1267
+ #
1268
+ def release
1269
+ (defined?(@release) && @release) || (defined?(::Release) && ::Release) || (defined?(::RELEASE) && ::RELEASE)
1270
+ end
1271
+
1272
+ #
1273
+ # Returns version string from program_name, version and release.
1274
+ #
1275
+ def ver
1276
+ if v = version
1277
+ str = +"#{program_name} #{[v].join('.')}"
1278
+ str << " (#{v})" if v = release
1279
+ str
1280
+ end
1281
+ end
1282
+
1283
+ def warn(mesg = $!)
1284
+ super("#{program_name}: #{mesg}")
1285
+ end
1286
+
1287
+ def abort(mesg = $!)
1288
+ super("#{program_name}: #{mesg}")
1289
+ end
1290
+
1291
+ #
1292
+ # Subject of #on / #on_head, #accept / #reject
1293
+ #
1294
+ def top
1295
+ @stack[-1]
1296
+ end
1297
+
1298
+ #
1299
+ # Subject of #on_tail.
1300
+ #
1301
+ def base
1302
+ @stack[1]
1303
+ end
1304
+
1305
+ #
1306
+ # Pushes a new List.
1307
+ #
1308
+ def new
1309
+ @stack.push(List.new)
1310
+ if block_given?
1311
+ yield self
1312
+ else
1313
+ self
1314
+ end
1315
+ end
1316
+
1317
+ #
1318
+ # Removes the last List.
1319
+ #
1320
+ def remove
1321
+ @stack.pop
1322
+ end
1323
+
1324
+ #
1325
+ # Puts option summary into +to+ and returns +to+. Yields each line if
1326
+ # a block is given.
1327
+ #
1328
+ # +to+:: Output destination, which must have method <<. Defaults to [].
1329
+ # +width+:: Width of left side, defaults to @summary_width.
1330
+ # +max+:: Maximum length allowed for left side, defaults to +width+ - 1.
1331
+ # +indent+:: Indentation, defaults to @summary_indent.
1332
+ #
1333
+ def summarize(to = [], width = @summary_width, max = width - 1, indent = @summary_indent, &blk)
1334
+ nl = "\n"
1335
+ blk ||= proc {|l| to << (l.index(nl, -1) ? l : l + nl)}
1336
+ visit(:summarize, {}, {}, width, max, indent, &blk)
1337
+ to
1338
+ end
1339
+
1340
+ #
1341
+ # Returns option summary string.
1342
+ #
1343
+ def help; summarize("#{banner}".sub(/\n?\z/, "\n")) end
1344
+ alias to_s help
1345
+
1346
+ def pretty_print(q) # :nodoc:
1347
+ q.object_group(self) do
1348
+ first = true
1349
+ if @stack.size > 2
1350
+ @stack.each_with_index do |s, i|
1351
+ next if i < 2
1352
+ next if s.list.empty?
1353
+ if first
1354
+ first = false
1355
+ q.text ":"
1356
+ end
1357
+ q.breakable
1358
+ s.pretty_print(q)
1359
+ end
1360
+ end
1361
+ end
1362
+ end
1363
+
1364
+ def inspect # :nodoc:
1365
+ require 'pp'
1366
+ pretty_print_inspect
1367
+ end
1368
+
1369
+ #
1370
+ # Returns option summary list.
1371
+ #
1372
+ def to_a; summarize("#{banner}".split(/^/)) end
1373
+
1374
+ #
1375
+ # Checks if an argument is given twice, in which case an ArgumentError is
1376
+ # raised. Called from Gem::OptionParser#switch only.
1377
+ #
1378
+ # +obj+:: New argument.
1379
+ # +prv+:: Previously specified argument.
1380
+ # +msg+:: Exception message.
1381
+ #
1382
+ def notwice(obj, prv, msg) # :nodoc:
1383
+ unless !prv or prv == obj
1384
+ raise(ArgumentError, "argument #{msg} given twice: #{obj}",
1385
+ ParseError.filter_backtrace(caller(2)))
1386
+ end
1387
+ obj
1388
+ end
1389
+ private :notwice
1390
+
1391
+ SPLAT_PROC = proc {|*a| a.length <= 1 ? a.first : a} # :nodoc:
1392
+
1393
+ # :call-seq:
1394
+ # make_switch(params, block = nil)
1395
+ #
1396
+ # :include: ../doc/optparse/creates_option.rdoc
1397
+ #
1398
+ def make_switch(opts, block = nil)
1399
+ short, long, nolong, style, pattern, conv, not_pattern, not_conv, not_style = [], [], []
1400
+ ldesc, sdesc, desc, arg = [], [], []
1401
+ default_style = Switch::NoArgument
1402
+ default_pattern = nil
1403
+ klass = nil
1404
+ q, a = nil
1405
+ has_arg = false
1406
+
1407
+ opts.each do |o|
1408
+ # argument class
1409
+ next if search(:atype, o) do |pat, c|
1410
+ klass = notwice(o, klass, 'type')
1411
+ if not_style and not_style != Switch::NoArgument
1412
+ not_pattern, not_conv = pat, c
1413
+ else
1414
+ default_pattern, conv = pat, c
1415
+ end
1416
+ end
1417
+
1418
+ # directly specified pattern(any object possible to match)
1419
+ if (!(String === o || Symbol === o)) and o.respond_to?(:match)
1420
+ pattern = notwice(o, pattern, 'pattern')
1421
+ if pattern.respond_to?(:convert)
1422
+ conv = pattern.method(:convert).to_proc
1423
+ else
1424
+ conv = SPLAT_PROC
1425
+ end
1426
+ next
1427
+ end
1428
+
1429
+ # anything others
1430
+ case o
1431
+ when Proc, Method
1432
+ block = notwice(o, block, 'block')
1433
+ when Array, Hash
1434
+ case pattern
1435
+ when CompletingHash
1436
+ when nil
1437
+ pattern = CompletingHash.new
1438
+ conv = pattern.method(:convert).to_proc if pattern.respond_to?(:convert)
1439
+ else
1440
+ raise ArgumentError, "argument pattern given twice"
1441
+ end
1442
+ o.each {|pat, *v| pattern[pat] = v.fetch(0) {pat}}
1443
+ when Module
1444
+ raise ArgumentError, "unsupported argument type: #{o}", ParseError.filter_backtrace(caller(4))
1445
+ when *ArgumentStyle.keys
1446
+ style = notwice(ArgumentStyle[o], style, 'style')
1447
+ when /^--no-([^\[\]=\s]*)(.+)?/
1448
+ q, a = $1, $2
1449
+ o = notwice(a ? Object : TrueClass, klass, 'type')
1450
+ not_pattern, not_conv = search(:atype, o) unless not_style
1451
+ not_style = (not_style || default_style).guess(arg = a) if a
1452
+ default_style = Switch::NoArgument
1453
+ default_pattern, conv = search(:atype, FalseClass) unless default_pattern
1454
+ ldesc << "--no-#{q}"
1455
+ (q = q.downcase).tr!('_', '-')
1456
+ long << "no-#{q}"
1457
+ nolong << q
1458
+ when /^--\[no-\]([^\[\]=\s]*)(.+)?/
1459
+ q, a = $1, $2
1460
+ o = notwice(a ? Object : TrueClass, klass, 'type')
1461
+ if a
1462
+ default_style = default_style.guess(arg = a)
1463
+ default_pattern, conv = search(:atype, o) unless default_pattern
1464
+ end
1465
+ ldesc << "--[no-]#{q}"
1466
+ (o = q.downcase).tr!('_', '-')
1467
+ long << o
1468
+ not_pattern, not_conv = search(:atype, FalseClass) unless not_style
1469
+ not_style = Switch::NoArgument
1470
+ nolong << "no-#{o}"
1471
+ when /^--([^\[\]=\s]*)(.+)?/
1472
+ q, a = $1, $2
1473
+ if a
1474
+ o = notwice(NilClass, klass, 'type')
1475
+ default_style = default_style.guess(arg = a)
1476
+ default_pattern, conv = search(:atype, o) unless default_pattern
1477
+ end
1478
+ ldesc << "--#{q}"
1479
+ (o = q.downcase).tr!('_', '-')
1480
+ long << o
1481
+ when /^-(\[\^?\]?(?:[^\\\]]|\\.)*\])(.+)?/
1482
+ q, a = $1, $2
1483
+ o = notwice(Object, klass, 'type')
1484
+ if a
1485
+ default_style = default_style.guess(arg = a)
1486
+ default_pattern, conv = search(:atype, o) unless default_pattern
1487
+ else
1488
+ has_arg = true
1489
+ end
1490
+ sdesc << "-#{q}"
1491
+ short << Regexp.new(q)
1492
+ when /^-(.)(.+)?/
1493
+ q, a = $1, $2
1494
+ if a
1495
+ o = notwice(NilClass, klass, 'type')
1496
+ default_style = default_style.guess(arg = a)
1497
+ default_pattern, conv = search(:atype, o) unless default_pattern
1498
+ end
1499
+ sdesc << "-#{q}"
1500
+ short << q
1501
+ when /^=/
1502
+ style = notwice(default_style.guess(arg = o), style, 'style')
1503
+ default_pattern, conv = search(:atype, Object) unless default_pattern
1504
+ else
1505
+ desc.push(o)
1506
+ end
1507
+ end
1508
+
1509
+ default_pattern, conv = search(:atype, default_style.pattern) unless default_pattern
1510
+ if !(short.empty? and long.empty?)
1511
+ if has_arg and default_style == Switch::NoArgument
1512
+ default_style = Switch::RequiredArgument
1513
+ end
1514
+ s = (style || default_style).new(pattern || default_pattern,
1515
+ conv, sdesc, ldesc, arg, desc, block)
1516
+ elsif !block
1517
+ if style or pattern
1518
+ raise ArgumentError, "no switch given", ParseError.filter_backtrace(caller)
1519
+ end
1520
+ s = desc
1521
+ else
1522
+ short << pattern
1523
+ s = (style || default_style).new(pattern,
1524
+ conv, nil, nil, arg, desc, block)
1525
+ end
1526
+ return s, short, long,
1527
+ (not_style.new(not_pattern, not_conv, sdesc, ldesc, nil, desc, block) if not_style),
1528
+ nolong
1529
+ end
1530
+
1531
+ # :call-seq:
1532
+ # define(*params, &block)
1533
+ #
1534
+ # :include: ../doc/optparse/creates_option.rdoc
1535
+ #
1536
+ def define(*opts, &block)
1537
+ top.append(*(sw = make_switch(opts, block)))
1538
+ sw[0]
1539
+ end
1540
+
1541
+ # :call-seq:
1542
+ # on(*params, &block)
1543
+ #
1544
+ # :include: ../doc/optparse/creates_option.rdoc
1545
+ #
1546
+ def on(*opts, &block)
1547
+ define(*opts, &block)
1548
+ self
1549
+ end
1550
+ alias def_option define
1551
+
1552
+ # :call-seq:
1553
+ # define_head(*params, &block)
1554
+ #
1555
+ # :include: ../doc/optparse/creates_option.rdoc
1556
+ #
1557
+ def define_head(*opts, &block)
1558
+ top.prepend(*(sw = make_switch(opts, block)))
1559
+ sw[0]
1560
+ end
1561
+
1562
+ # :call-seq:
1563
+ # on_head(*params, &block)
1564
+ #
1565
+ # :include: ../doc/optparse/creates_option.rdoc
1566
+ #
1567
+ # The new option is added at the head of the summary.
1568
+ #
1569
+ def on_head(*opts, &block)
1570
+ define_head(*opts, &block)
1571
+ self
1572
+ end
1573
+ alias def_head_option define_head
1574
+
1575
+ # :call-seq:
1576
+ # define_tail(*params, &block)
1577
+ #
1578
+ # :include: ../doc/optparse/creates_option.rdoc
1579
+ #
1580
+ def define_tail(*opts, &block)
1581
+ base.append(*(sw = make_switch(opts, block)))
1582
+ sw[0]
1583
+ end
1584
+
1585
+ #
1586
+ # :call-seq:
1587
+ # on_tail(*params, &block)
1588
+ #
1589
+ # :include: ../doc/optparse/creates_option.rdoc
1590
+ #
1591
+ # The new option is added at the tail of the summary.
1592
+ #
1593
+ def on_tail(*opts, &block)
1594
+ define_tail(*opts, &block)
1595
+ self
1596
+ end
1597
+ alias def_tail_option define_tail
1598
+
1599
+ #
1600
+ # Add separator in summary.
1601
+ #
1602
+ def separator(string)
1603
+ top.append(string, nil, nil)
1604
+ end
1605
+
1606
+ #
1607
+ # Parses command line arguments +argv+ in order. When a block is given,
1608
+ # each non-option argument is yielded. When optional +into+ keyword
1609
+ # argument is provided, the parsed option values are stored there via
1610
+ # <code>[]=</code> method (so it can be Hash, or OpenStruct, or other
1611
+ # similar object).
1612
+ #
1613
+ # Returns the rest of +argv+ left unparsed.
1614
+ #
1615
+ def order(*argv, into: nil, &nonopt)
1616
+ argv = argv[0].dup if argv.size == 1 and Array === argv[0]
1617
+ order!(argv, into: into, &nonopt)
1618
+ end
1619
+
1620
+ #
1621
+ # Same as #order, but removes switches destructively.
1622
+ # Non-option arguments remain in +argv+.
1623
+ #
1624
+ def order!(argv = default_argv, into: nil, &nonopt)
1625
+ setter = ->(name, val) {into[name.to_sym] = val} if into
1626
+ parse_in_order(argv, setter, &nonopt)
1627
+ end
1628
+
1629
+ def parse_in_order(argv = default_argv, setter = nil, &nonopt) # :nodoc:
1630
+ opt, arg, val, rest = nil
1631
+ nonopt ||= proc {|a| throw :terminate, a}
1632
+ argv.unshift(arg) if arg = catch(:terminate) {
1633
+ while arg = argv.shift
1634
+ case arg
1635
+ # long option
1636
+ when /\A--([^=]*)(?:=(.*))?/m
1637
+ opt, rest = $1, $2
1638
+ opt.tr!('_', '-')
1639
+ begin
1640
+ sw, = complete(:long, opt, true)
1641
+ if require_exact && !sw.long.include?(arg)
1642
+ raise InvalidOption, arg
1643
+ end
1644
+ rescue ParseError
1645
+ raise $!.set_option(arg, true)
1646
+ end
1647
+ begin
1648
+ opt, cb, val = sw.parse(rest, argv) {|*exc| raise(*exc)}
1649
+ val = cb.call(val) if cb
1650
+ setter.call(sw.switch_name, val) if setter
1651
+ rescue ParseError
1652
+ raise $!.set_option(arg, rest)
1653
+ end
1654
+
1655
+ # short option
1656
+ when /\A-(.)((=).*|.+)?/m
1657
+ eq, rest, opt = $3, $2, $1
1658
+ has_arg, val = eq, rest
1659
+ begin
1660
+ sw, = search(:short, opt)
1661
+ unless sw
1662
+ begin
1663
+ sw, = complete(:short, opt)
1664
+ # short option matched.
1665
+ val = arg.delete_prefix('-')
1666
+ has_arg = true
1667
+ rescue InvalidOption
1668
+ raise if require_exact
1669
+ # if no short options match, try completion with long
1670
+ # options.
1671
+ sw, = complete(:long, opt)
1672
+ eq ||= !rest
1673
+ end
1674
+ end
1675
+ rescue ParseError
1676
+ raise $!.set_option(arg, true)
1677
+ end
1678
+ begin
1679
+ opt, cb, val = sw.parse(val, argv) {|*exc| raise(*exc) if eq}
1680
+ rescue ParseError
1681
+ raise $!.set_option(arg, arg.length > 2)
1682
+ else
1683
+ raise InvalidOption, arg if has_arg and !eq and arg == "-#{opt}"
1684
+ end
1685
+ begin
1686
+ argv.unshift(opt) if opt and (!rest or (opt = opt.sub(/\A-*/, '-')) != '-')
1687
+ val = cb.call(val) if cb
1688
+ setter.call(sw.switch_name, val) if setter
1689
+ rescue ParseError
1690
+ raise $!.set_option(arg, arg.length > 2)
1691
+ end
1692
+
1693
+ # non-option argument
1694
+ else
1695
+ catch(:prune) do
1696
+ visit(:each_option) do |sw0|
1697
+ sw = sw0
1698
+ sw.block.call(arg) if Switch === sw and sw.match_nonswitch?(arg)
1699
+ end
1700
+ nonopt.call(arg)
1701
+ end
1702
+ end
1703
+ end
1704
+
1705
+ nil
1706
+ }
1707
+
1708
+ visit(:search, :short, nil) {|sw| sw.block.call(*argv) if !sw.pattern}
1709
+
1710
+ argv
1711
+ end
1712
+ private :parse_in_order
1713
+
1714
+ #
1715
+ # Parses command line arguments +argv+ in permutation mode and returns
1716
+ # list of non-option arguments. When optional +into+ keyword
1717
+ # argument is provided, the parsed option values are stored there via
1718
+ # <code>[]=</code> method (so it can be Hash, or OpenStruct, or other
1719
+ # similar object).
1720
+ #
1721
+ def permute(*argv, into: nil)
1722
+ argv = argv[0].dup if argv.size == 1 and Array === argv[0]
1723
+ permute!(argv, into: into)
1724
+ end
1725
+
1726
+ #
1727
+ # Same as #permute, but removes switches destructively.
1728
+ # Non-option arguments remain in +argv+.
1729
+ #
1730
+ def permute!(argv = default_argv, into: nil)
1731
+ nonopts = []
1732
+ order!(argv, into: into, &nonopts.method(:<<))
1733
+ argv[0, 0] = nonopts
1734
+ argv
1735
+ end
1736
+
1737
+ #
1738
+ # Parses command line arguments +argv+ in order when environment variable
1739
+ # POSIXLY_CORRECT is set, and in permutation mode otherwise.
1740
+ # When optional +into+ keyword argument is provided, the parsed option
1741
+ # values are stored there via <code>[]=</code> method (so it can be Hash,
1742
+ # or OpenStruct, or other similar object).
1743
+ #
1744
+ def parse(*argv, into: nil)
1745
+ argv = argv[0].dup if argv.size == 1 and Array === argv[0]
1746
+ parse!(argv, into: into)
1747
+ end
1748
+
1749
+ #
1750
+ # Same as #parse, but removes switches destructively.
1751
+ # Non-option arguments remain in +argv+.
1752
+ #
1753
+ def parse!(argv = default_argv, into: nil)
1754
+ if ENV.include?('POSIXLY_CORRECT')
1755
+ order!(argv, into: into)
1756
+ else
1757
+ permute!(argv, into: into)
1758
+ end
1759
+ end
1760
+
1761
+ #
1762
+ # Wrapper method for getopts.rb.
1763
+ #
1764
+ # params = ARGV.getopts("ab:", "foo", "bar:", "zot:Z;zot option")
1765
+ # # params["a"] = true # -a
1766
+ # # params["b"] = "1" # -b1
1767
+ # # params["foo"] = "1" # --foo
1768
+ # # params["bar"] = "x" # --bar x
1769
+ # # params["zot"] = "z" # --zot Z
1770
+ #
1771
+ def getopts(*args)
1772
+ argv = Array === args.first ? args.shift : default_argv
1773
+ single_options, *long_options = *args
1774
+
1775
+ result = {}
1776
+
1777
+ single_options.scan(/(.)(:)?/) do |opt, val|
1778
+ if val
1779
+ result[opt] = nil
1780
+ define("-#{opt} VAL")
1781
+ else
1782
+ result[opt] = false
1783
+ define("-#{opt}")
1784
+ end
1785
+ end if single_options
1786
+
1787
+ long_options.each do |arg|
1788
+ arg, desc = arg.split(';', 2)
1789
+ opt, val = arg.split(':', 2)
1790
+ if val
1791
+ result[opt] = val.empty? ? nil : val
1792
+ define("--#{opt}=#{result[opt] || "VAL"}", *[desc].compact)
1793
+ else
1794
+ result[opt] = false
1795
+ define("--#{opt}", *[desc].compact)
1796
+ end
1797
+ end
1798
+
1799
+ parse_in_order(argv, result.method(:[]=))
1800
+ result
1801
+ end
1802
+
1803
+ #
1804
+ # See #getopts.
1805
+ #
1806
+ def self.getopts(*args)
1807
+ new.getopts(*args)
1808
+ end
1809
+
1810
+ #
1811
+ # Traverses @stack, sending each element method +id+ with +args+ and
1812
+ # +block+.
1813
+ #
1814
+ def visit(id, *args, &block) # :nodoc:
1815
+ @stack.reverse_each do |el|
1816
+ el.__send__(id, *args, &block)
1817
+ end
1818
+ nil
1819
+ end
1820
+ private :visit
1821
+
1822
+ #
1823
+ # Searches +key+ in @stack for +id+ hash and returns or yields the result.
1824
+ #
1825
+ def search(id, key) # :nodoc:
1826
+ block_given = block_given?
1827
+ visit(:search, id, key) do |k|
1828
+ return block_given ? yield(k) : k
1829
+ end
1830
+ end
1831
+ private :search
1832
+
1833
+ #
1834
+ # Completes shortened long style option switch and returns pair of
1835
+ # canonical switch and switch descriptor Gem::OptionParser::Switch.
1836
+ #
1837
+ # +typ+:: Searching table.
1838
+ # +opt+:: Searching key.
1839
+ # +icase+:: Search case insensitive if true.
1840
+ # +pat+:: Optional pattern for completion.
1841
+ #
1842
+ def complete(typ, opt, icase = false, *pat) # :nodoc:
1843
+ if pat.empty?
1844
+ search(typ, opt) {|sw| return [sw, opt]} # exact match or...
1845
+ end
1846
+ ambiguous = catch(:ambiguous) {
1847
+ visit(:complete, typ, opt, icase, *pat) {|o, *sw| return sw}
1848
+ }
1849
+ exc = ambiguous ? AmbiguousOption : InvalidOption
1850
+ raise exc.new(opt, additional: self.method(:additional_message).curry[typ])
1851
+ end
1852
+ private :complete
1853
+
1854
+ #
1855
+ # Returns additional info.
1856
+ #
1857
+ def additional_message(typ, opt)
1858
+ return unless typ and opt and defined?(DidYouMean::SpellChecker)
1859
+ all_candidates = []
1860
+ visit(:get_candidates, typ) do |candidates|
1861
+ all_candidates.concat(candidates)
1862
+ end
1863
+ all_candidates.select! {|cand| cand.is_a?(String) }
1864
+ checker = DidYouMean::SpellChecker.new(dictionary: all_candidates)
1865
+ suggestions = all_candidates & checker.correct(opt)
1866
+ if DidYouMean.respond_to?(:formatter)
1867
+ DidYouMean.formatter.message_for(suggestions)
1868
+ else
1869
+ "\nDid you mean? #{suggestions.join("\n ")}"
1870
+ end
1871
+ end
1872
+
1873
+ def candidate(word)
1874
+ list = []
1875
+ case word
1876
+ when '-'
1877
+ long = short = true
1878
+ when /\A--/
1879
+ word, arg = word.split(/=/, 2)
1880
+ argpat = Completion.regexp(arg, false) if arg and !arg.empty?
1881
+ long = true
1882
+ when /\A-/
1883
+ short = true
1884
+ end
1885
+ pat = Completion.regexp(word, long)
1886
+ visit(:each_option) do |opt|
1887
+ next unless Switch === opt
1888
+ opts = (long ? opt.long : []) + (short ? opt.short : [])
1889
+ opts = Completion.candidate(word, true, pat, &opts.method(:each)).map(&:first) if pat
1890
+ if /\A=/ =~ opt.arg
1891
+ opts.map! {|sw| sw + "="}
1892
+ if arg and CompletingHash === opt.pattern
1893
+ if opts = opt.pattern.candidate(arg, false, argpat)
1894
+ opts.map!(&:last)
1895
+ end
1896
+ end
1897
+ end
1898
+ list.concat(opts)
1899
+ end
1900
+ list
1901
+ end
1902
+
1903
+ #
1904
+ # Loads options from file names as +filename+. Does nothing when the file
1905
+ # is not present. Returns whether successfully loaded.
1906
+ #
1907
+ # +filename+ defaults to basename of the program without suffix in a
1908
+ # directory ~/.options, then the basename with '.options' suffix
1909
+ # under XDG and Haiku standard places.
1910
+ #
1911
+ def load(filename = nil)
1912
+ unless filename
1913
+ basename = File.basename($0, '.*')
1914
+ return true if load(File.expand_path(basename, '~/.options')) rescue nil
1915
+ basename << ".options"
1916
+ return [
1917
+ # XDG
1918
+ ENV['XDG_CONFIG_HOME'],
1919
+ '~/.config',
1920
+ *ENV['XDG_CONFIG_DIRS']&.split(File::PATH_SEPARATOR),
1921
+
1922
+ # Haiku
1923
+ '~/config/settings',
1924
+ ].any? {|dir|
1925
+ next if !dir or dir.empty?
1926
+ load(File.expand_path(basename, dir)) rescue nil
1927
+ }
1928
+ end
1929
+ begin
1930
+ parse(*IO.readlines(filename).each {|s| s.chomp!})
1931
+ true
1932
+ rescue Errno::ENOENT, Errno::ENOTDIR
1933
+ false
1934
+ end
1935
+ end
1936
+
1937
+ #
1938
+ # Parses environment variable +env+ or its uppercase with splitting like a
1939
+ # shell.
1940
+ #
1941
+ # +env+ defaults to the basename of the program.
1942
+ #
1943
+ def environment(env = File.basename($0, '.*'))
1944
+ env = ENV[env] || ENV[env.upcase] or return
1945
+ require 'shellwords'
1946
+ parse(*Shellwords.shellwords(env))
1947
+ end
1948
+
1949
+ #
1950
+ # Acceptable argument classes
1951
+ #
1952
+
1953
+ #
1954
+ # Any string and no conversion. This is fall-back.
1955
+ #
1956
+ accept(Object) {|s,|s or s.nil?}
1957
+
1958
+ accept(NilClass) {|s,|s}
1959
+
1960
+ #
1961
+ # Any non-empty string, and no conversion.
1962
+ #
1963
+ accept(String, /.+/m) {|s,*|s}
1964
+
1965
+ #
1966
+ # Ruby/C-like integer, octal for 0-7 sequence, binary for 0b, hexadecimal
1967
+ # for 0x, and decimal for others; with optional sign prefix. Converts to
1968
+ # Integer.
1969
+ #
1970
+ decimal = '\d+(?:_\d+)*'
1971
+ binary = 'b[01]+(?:_[01]+)*'
1972
+ hex = 'x[\da-f]+(?:_[\da-f]+)*'
1973
+ octal = "0(?:[0-7]+(?:_[0-7]+)*|#{binary}|#{hex})?"
1974
+ integer = "#{octal}|#{decimal}"
1975
+
1976
+ accept(Integer, %r"\A[-+]?(?:#{integer})\z"io) {|s,|
1977
+ begin
1978
+ Integer(s)
1979
+ rescue ArgumentError
1980
+ raise Gem::OptionParser::InvalidArgument, s
1981
+ end if s
1982
+ }
1983
+
1984
+ #
1985
+ # Float number format, and converts to Float.
1986
+ #
1987
+ float = "(?:#{decimal}(?=(.)?)(?:\\.(?:#{decimal})?)?|\\.#{decimal})(?:E[-+]?#{decimal})?"
1988
+ floatpat = %r"\A[-+]?#{float}\z"io
1989
+ accept(Float, floatpat) {|s,| s.to_f if s}
1990
+
1991
+ #
1992
+ # Generic numeric format, converts to Integer for integer format, Float
1993
+ # for float format, and Rational for rational format.
1994
+ #
1995
+ real = "[-+]?(?:#{octal}|#{float})"
1996
+ accept(Numeric, /\A(#{real})(?:\/(#{real}))?\z/io) {|s, d, f, n,|
1997
+ if n
1998
+ Rational(d, n)
1999
+ elsif f
2000
+ Float(s)
2001
+ else
2002
+ Integer(s)
2003
+ end
2004
+ }
2005
+
2006
+ #
2007
+ # Decimal integer format, to be converted to Integer.
2008
+ #
2009
+ DecimalInteger = /\A[-+]?#{decimal}\z/io
2010
+ accept(DecimalInteger, DecimalInteger) {|s,|
2011
+ begin
2012
+ Integer(s, 10)
2013
+ rescue ArgumentError
2014
+ raise Gem::OptionParser::InvalidArgument, s
2015
+ end if s
2016
+ }
2017
+
2018
+ #
2019
+ # Ruby/C like octal/hexadecimal/binary integer format, to be converted to
2020
+ # Integer.
2021
+ #
2022
+ OctalInteger = /\A[-+]?(?:[0-7]+(?:_[0-7]+)*|0(?:#{binary}|#{hex}))\z/io
2023
+ accept(OctalInteger, OctalInteger) {|s,|
2024
+ begin
2025
+ Integer(s, 8)
2026
+ rescue ArgumentError
2027
+ raise Gem::OptionParser::InvalidArgument, s
2028
+ end if s
2029
+ }
2030
+
2031
+ #
2032
+ # Decimal integer/float number format, to be converted to Integer for
2033
+ # integer format, Float for float format.
2034
+ #
2035
+ DecimalNumeric = floatpat # decimal integer is allowed as float also.
2036
+ accept(DecimalNumeric, floatpat) {|s, f|
2037
+ begin
2038
+ if f
2039
+ Float(s)
2040
+ else
2041
+ Integer(s)
2042
+ end
2043
+ rescue ArgumentError
2044
+ raise Gem::OptionParser::InvalidArgument, s
2045
+ end if s
2046
+ }
2047
+
2048
+ #
2049
+ # Boolean switch, which means whether it is present or not, whether it is
2050
+ # absent or not with prefix no-, or it takes an argument
2051
+ # yes/no/true/false/+/-.
2052
+ #
2053
+ yesno = CompletingHash.new
2054
+ %w[- no false].each {|el| yesno[el] = false}
2055
+ %w[+ yes true].each {|el| yesno[el] = true}
2056
+ yesno['nil'] = false # should be nil?
2057
+ accept(TrueClass, yesno) {|arg, val| val == nil or val}
2058
+ #
2059
+ # Similar to TrueClass, but defaults to false.
2060
+ #
2061
+ accept(FalseClass, yesno) {|arg, val| val != nil and val}
2062
+
2063
+ #
2064
+ # List of strings separated by ",".
2065
+ #
2066
+ accept(Array) do |s, |
2067
+ if s
2068
+ s = s.split(',').collect {|ss| ss unless ss.empty?}
2069
+ end
2070
+ s
2071
+ end
2072
+
2073
+ #
2074
+ # Regular expression with options.
2075
+ #
2076
+ accept(Regexp, %r"\A/((?:\\.|[^\\])*)/([[:alpha:]]+)?\z|.*") do |all, s, o|
2077
+ f = 0
2078
+ if o
2079
+ f |= Regexp::IGNORECASE if /i/ =~ o
2080
+ f |= Regexp::MULTILINE if /m/ =~ o
2081
+ f |= Regexp::EXTENDED if /x/ =~ o
2082
+ k = o.delete("imx")
2083
+ k = nil if k.empty?
2084
+ end
2085
+ Regexp.new(s || all, f, k)
2086
+ end
2087
+
2088
+ #
2089
+ # Exceptions
2090
+ #
2091
+
2092
+ #
2093
+ # Base class of exceptions from Gem::OptionParser.
2094
+ #
2095
+ class ParseError < RuntimeError
2096
+ # Reason which caused the error.
2097
+ Reason = 'parse error'
2098
+
2099
+ def initialize(*args, additional: nil)
2100
+ @additional = additional
2101
+ @arg0, = args
2102
+ @args = args
2103
+ @reason = nil
2104
+ end
2105
+
2106
+ attr_reader :args
2107
+ attr_writer :reason
2108
+ attr_accessor :additional
2109
+
2110
+ #
2111
+ # Pushes back erred argument(s) to +argv+.
2112
+ #
2113
+ def recover(argv)
2114
+ argv[0, 0] = @args
2115
+ argv
2116
+ end
2117
+
2118
+ def self.filter_backtrace(array)
2119
+ unless $DEBUG
2120
+ array.delete_if(&%r"\A#{Regexp.quote(__FILE__)}:"o.method(:=~))
2121
+ end
2122
+ array
2123
+ end
2124
+
2125
+ def set_backtrace(array)
2126
+ super(self.class.filter_backtrace(array))
2127
+ end
2128
+
2129
+ def set_option(opt, eq)
2130
+ if eq
2131
+ @args[0] = opt
2132
+ else
2133
+ @args.unshift(opt)
2134
+ end
2135
+ self
2136
+ end
2137
+
2138
+ #
2139
+ # Returns error reason. Override this for I18N.
2140
+ #
2141
+ def reason
2142
+ @reason || self.class::Reason
2143
+ end
2144
+
2145
+ def inspect
2146
+ "#<#{self.class}: #{args.join(' ')}>"
2147
+ end
2148
+
2149
+ #
2150
+ # Default stringizing method to emit standard error message.
2151
+ #
2152
+ def message
2153
+ "#{reason}: #{args.join(' ')}#{additional[@arg0] if additional}"
2154
+ end
2155
+
2156
+ alias to_s message
2157
+ end
2158
+
2159
+ #
2160
+ # Raises when ambiguously completable string is encountered.
2161
+ #
2162
+ class AmbiguousOption < ParseError
2163
+ const_set(:Reason, 'ambiguous option')
2164
+ end
2165
+
2166
+ #
2167
+ # Raises when there is an argument for a switch which takes no argument.
2168
+ #
2169
+ class NeedlessArgument < ParseError
2170
+ const_set(:Reason, 'needless argument')
2171
+ end
2172
+
2173
+ #
2174
+ # Raises when a switch with mandatory argument has no argument.
2175
+ #
2176
+ class MissingArgument < ParseError
2177
+ const_set(:Reason, 'missing argument')
2178
+ end
2179
+
2180
+ #
2181
+ # Raises when switch is undefined.
2182
+ #
2183
+ class InvalidOption < ParseError
2184
+ const_set(:Reason, 'invalid option')
2185
+ end
2186
+
2187
+ #
2188
+ # Raises when the given argument does not match required format.
2189
+ #
2190
+ class InvalidArgument < ParseError
2191
+ const_set(:Reason, 'invalid argument')
2192
+ end
2193
+
2194
+ #
2195
+ # Raises when the given argument word can't be completed uniquely.
2196
+ #
2197
+ class AmbiguousArgument < InvalidArgument
2198
+ const_set(:Reason, 'ambiguous argument')
2199
+ end
2200
+
2201
+ #
2202
+ # Miscellaneous
2203
+ #
2204
+
2205
+ #
2206
+ # Extends command line arguments array (ARGV) to parse itself.
2207
+ #
2208
+ module Arguable
2209
+
2210
+ #
2211
+ # Sets Gem::OptionParser object, when +opt+ is +false+ or +nil+, methods
2212
+ # Gem::OptionParser::Arguable#options and Gem::OptionParser::Arguable#options= are
2213
+ # undefined. Thus, there is no ways to access the Gem::OptionParser object
2214
+ # via the receiver object.
2215
+ #
2216
+ def options=(opt)
2217
+ unless @optparse = opt
2218
+ class << self
2219
+ undef_method(:options)
2220
+ undef_method(:options=)
2221
+ end
2222
+ end
2223
+ end
2224
+
2225
+ #
2226
+ # Actual Gem::OptionParser object, automatically created if nonexistent.
2227
+ #
2228
+ # If called with a block, yields the Gem::OptionParser object and returns the
2229
+ # result of the block. If an Gem::OptionParser::ParseError exception occurs
2230
+ # in the block, it is rescued, a error message printed to STDERR and
2231
+ # +nil+ returned.
2232
+ #
2233
+ def options
2234
+ @optparse ||= Gem::OptionParser.new
2235
+ @optparse.default_argv = self
2236
+ block_given? or return @optparse
2237
+ begin
2238
+ yield @optparse
2239
+ rescue ParseError
2240
+ @optparse.warn $!
2241
+ nil
2242
+ end
2243
+ end
2244
+
2245
+ #
2246
+ # Parses +self+ destructively in order and returns +self+ containing the
2247
+ # rest arguments left unparsed.
2248
+ #
2249
+ def order!(&blk) options.order!(self, &blk) end
2250
+
2251
+ #
2252
+ # Parses +self+ destructively in permutation mode and returns +self+
2253
+ # containing the rest arguments left unparsed.
2254
+ #
2255
+ def permute!() options.permute!(self) end
2256
+
2257
+ #
2258
+ # Parses +self+ destructively and returns +self+ containing the
2259
+ # rest arguments left unparsed.
2260
+ #
2261
+ def parse!() options.parse!(self) end
2262
+
2263
+ #
2264
+ # Substitution of getopts is possible as follows. Also see
2265
+ # Gem::OptionParser#getopts.
2266
+ #
2267
+ # def getopts(*args)
2268
+ # ($OPT = ARGV.getopts(*args)).each do |opt, val|
2269
+ # eval "$OPT_#{opt.gsub(/[^A-Za-z0-9_]/, '_')} = val"
2270
+ # end
2271
+ # rescue Gem::OptionParser::ParseError
2272
+ # end
2273
+ #
2274
+ def getopts(*args)
2275
+ options.getopts(self, *args)
2276
+ end
2277
+
2278
+ #
2279
+ # Initializes instance variable.
2280
+ #
2281
+ def self.extend_object(obj)
2282
+ super
2283
+ obj.instance_eval {@optparse = nil}
2284
+ end
2285
+ def initialize(*args)
2286
+ super
2287
+ @optparse = nil
2288
+ end
2289
+ end
2290
+
2291
+ #
2292
+ # Acceptable argument classes. Now contains DecimalInteger, OctalInteger
2293
+ # and DecimalNumeric. See Acceptable argument classes (in source code).
2294
+ #
2295
+ module Acceptables
2296
+ const_set(:DecimalInteger, Gem::OptionParser::DecimalInteger)
2297
+ const_set(:OctalInteger, Gem::OptionParser::OctalInteger)
2298
+ const_set(:DecimalNumeric, Gem::OptionParser::DecimalNumeric)
2299
+ end
2300
+ end
2301
+
2302
+ # ARGV is arguable by Gem::OptionParser
2303
+ ARGV.extend(Gem::OptionParser::Arguable)