bundler-prehistoric 1.6.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (284) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +17 -0
  3. data/.rspec +3 -0
  4. data/.travis.yml +100 -0
  5. data/CHANGELOG.md +1640 -0
  6. data/CONTRIBUTING.md +32 -0
  7. data/DEVELOPMENT.md +117 -0
  8. data/Gemfile +3 -0
  9. data/ISSUES.md +96 -0
  10. data/LICENSE.md +23 -0
  11. data/README.md +46 -0
  12. data/Rakefile +256 -0
  13. data/UPGRADING.md +103 -0
  14. data/bin/bundle +21 -0
  15. data/bin/bundle_ruby +56 -0
  16. data/bin/bundler +21 -0
  17. data/bundler.gemspec +33 -0
  18. data/lib/bundler.rb +442 -0
  19. data/lib/bundler/backports/time.rb +7 -0
  20. data/lib/bundler/capistrano.rb +16 -0
  21. data/lib/bundler/cli.rb +395 -0
  22. data/lib/bundler/cli/binstubs.rb +38 -0
  23. data/lib/bundler/cli/cache.rb +34 -0
  24. data/lib/bundler/cli/check.rb +35 -0
  25. data/lib/bundler/cli/clean.rb +19 -0
  26. data/lib/bundler/cli/common.rb +54 -0
  27. data/lib/bundler/cli/config.rb +84 -0
  28. data/lib/bundler/cli/console.rb +39 -0
  29. data/lib/bundler/cli/exec.rb +37 -0
  30. data/lib/bundler/cli/gem.rb +101 -0
  31. data/lib/bundler/cli/init.rb +33 -0
  32. data/lib/bundler/cli/inject.rb +33 -0
  33. data/lib/bundler/cli/install.rb +133 -0
  34. data/lib/bundler/cli/open.rb +23 -0
  35. data/lib/bundler/cli/outdated.rb +80 -0
  36. data/lib/bundler/cli/package.rb +36 -0
  37. data/lib/bundler/cli/platform.rb +43 -0
  38. data/lib/bundler/cli/show.rb +48 -0
  39. data/lib/bundler/cli/update.rb +73 -0
  40. data/lib/bundler/cli/viz.rb +27 -0
  41. data/lib/bundler/constants.rb +5 -0
  42. data/lib/bundler/current_ruby.rb +155 -0
  43. data/lib/bundler/definition.rb +604 -0
  44. data/lib/bundler/dep_proxy.rb +43 -0
  45. data/lib/bundler/dependency.rb +106 -0
  46. data/lib/bundler/deployment.rb +59 -0
  47. data/lib/bundler/deprecate.rb +15 -0
  48. data/lib/bundler/dsl.rb +305 -0
  49. data/lib/bundler/endpoint_specification.rb +76 -0
  50. data/lib/bundler/env.rb +56 -0
  51. data/lib/bundler/environment.rb +42 -0
  52. data/lib/bundler/fetcher.rb +396 -0
  53. data/lib/bundler/friendly_errors.rb +42 -0
  54. data/lib/bundler/gem_helper.rb +169 -0
  55. data/lib/bundler/gem_helpers.rb +25 -0
  56. data/lib/bundler/gem_installer.rb +9 -0
  57. data/lib/bundler/gem_path_manipulation.rb +8 -0
  58. data/lib/bundler/gem_tasks.rb +2 -0
  59. data/lib/bundler/graph.rb +169 -0
  60. data/lib/bundler/index.rb +184 -0
  61. data/lib/bundler/injector.rb +64 -0
  62. data/lib/bundler/installer.rb +332 -0
  63. data/lib/bundler/lazy_specification.rb +83 -0
  64. data/lib/bundler/lockfile_parser.rb +146 -0
  65. data/lib/bundler/match_platform.rb +13 -0
  66. data/lib/bundler/parallel_workers.rb +18 -0
  67. data/lib/bundler/parallel_workers/thread_worker.rb +30 -0
  68. data/lib/bundler/parallel_workers/unix_worker.rb +101 -0
  69. data/lib/bundler/parallel_workers/worker.rb +69 -0
  70. data/lib/bundler/psyched_yaml.rb +26 -0
  71. data/lib/bundler/remote_specification.rb +57 -0
  72. data/lib/bundler/resolver.rb +534 -0
  73. data/lib/bundler/retry.rb +60 -0
  74. data/lib/bundler/ruby_dsl.rb +11 -0
  75. data/lib/bundler/ruby_version.rb +117 -0
  76. data/lib/bundler/rubygems_ext.rb +171 -0
  77. data/lib/bundler/rubygems_integration.rb +578 -0
  78. data/lib/bundler/runtime.rb +310 -0
  79. data/lib/bundler/settings.rb +176 -0
  80. data/lib/bundler/setup.rb +23 -0
  81. data/lib/bundler/shared_helpers.rb +110 -0
  82. data/lib/bundler/similarity_detector.rb +63 -0
  83. data/lib/bundler/source.rb +29 -0
  84. data/lib/bundler/source/git.rb +289 -0
  85. data/lib/bundler/source/git/git_proxy.rb +158 -0
  86. data/lib/bundler/source/path.rb +225 -0
  87. data/lib/bundler/source/path/installer.rb +38 -0
  88. data/lib/bundler/source/rubygems.rb +334 -0
  89. data/lib/bundler/source/svn.rb +260 -0
  90. data/lib/bundler/source/svn/svn_proxy.rb +110 -0
  91. data/lib/bundler/spec_set.rb +154 -0
  92. data/lib/bundler/ssl_certs/.document +1 -0
  93. data/lib/bundler/ssl_certs/Class3PublicPrimaryCertificationAuthority.pem +14 -0
  94. data/lib/bundler/ssl_certs/DigiCertHighAssuranceEVRootCA.pem +23 -0
  95. data/lib/bundler/ssl_certs/EntrustnetSecureServerCertificationAuthority.pem +28 -0
  96. data/lib/bundler/ssl_certs/GeoTrustGlobalCA.pem +20 -0
  97. data/lib/bundler/ssl_certs/certificate_manager.rb +41 -0
  98. data/lib/bundler/templates/Executable +16 -0
  99. data/lib/bundler/templates/Executable.standalone +12 -0
  100. data/lib/bundler/templates/Gemfile +4 -0
  101. data/lib/bundler/templates/newgem/.travis.yml.tt +3 -0
  102. data/lib/bundler/templates/newgem/Gemfile.tt +4 -0
  103. data/lib/bundler/templates/newgem/LICENSE.txt.tt +22 -0
  104. data/lib/bundler/templates/newgem/README.md.tt +31 -0
  105. data/lib/bundler/templates/newgem/Rakefile.tt +26 -0
  106. data/lib/bundler/templates/newgem/bin/newgem.tt +3 -0
  107. data/lib/bundler/templates/newgem/consolerc.tt +3 -0
  108. data/lib/bundler/templates/newgem/ext/newgem/extconf.rb.tt +3 -0
  109. data/lib/bundler/templates/newgem/ext/newgem/newgem.c.tt +9 -0
  110. data/lib/bundler/templates/newgem/ext/newgem/newgem.h.tt +6 -0
  111. data/lib/bundler/templates/newgem/gitignore.tt +16 -0
  112. data/lib/bundler/templates/newgem/lib/newgem.rb.tt +12 -0
  113. data/lib/bundler/templates/newgem/lib/newgem/version.rb.tt +7 -0
  114. data/lib/bundler/templates/newgem/newgem.gemspec.tt +38 -0
  115. data/lib/bundler/templates/newgem/rspec.tt +2 -0
  116. data/lib/bundler/templates/newgem/spec/newgem_spec.rb.tt +11 -0
  117. data/lib/bundler/templates/newgem/spec/spec_helper.rb.tt +2 -0
  118. data/lib/bundler/templates/newgem/test/minitest_helper.rb.tt +4 -0
  119. data/lib/bundler/templates/newgem/test/test_newgem.rb.tt +11 -0
  120. data/lib/bundler/ui.rb +7 -0
  121. data/lib/bundler/ui/rg_proxy.rb +21 -0
  122. data/lib/bundler/ui/shell.rb +98 -0
  123. data/lib/bundler/ui/silent.rb +44 -0
  124. data/lib/bundler/vendor/.document +0 -0
  125. data/lib/bundler/vendor/net/http/faster.rb +26 -0
  126. data/lib/bundler/vendor/net/http/persistent.rb +1230 -0
  127. data/lib/bundler/vendor/net/http/persistent/ssl_reuse.rb +128 -0
  128. data/lib/bundler/vendor/thor.rb +480 -0
  129. data/lib/bundler/vendor/thor/actions.rb +319 -0
  130. data/lib/bundler/vendor/thor/actions/create_file.rb +103 -0
  131. data/lib/bundler/vendor/thor/actions/create_link.rb +59 -0
  132. data/lib/bundler/vendor/thor/actions/directory.rb +118 -0
  133. data/lib/bundler/vendor/thor/actions/empty_directory.rb +135 -0
  134. data/lib/bundler/vendor/thor/actions/file_manipulation.rb +316 -0
  135. data/lib/bundler/vendor/thor/actions/inject_into_file.rb +107 -0
  136. data/lib/bundler/vendor/thor/base.rb +656 -0
  137. data/lib/bundler/vendor/thor/command.rb +133 -0
  138. data/lib/bundler/vendor/thor/core_ext/hash_with_indifferent_access.rb +77 -0
  139. data/lib/bundler/vendor/thor/core_ext/io_binary_read.rb +10 -0
  140. data/lib/bundler/vendor/thor/core_ext/ordered_hash.rb +98 -0
  141. data/lib/bundler/vendor/thor/error.rb +32 -0
  142. data/lib/bundler/vendor/thor/group.rb +281 -0
  143. data/lib/bundler/vendor/thor/invocation.rb +173 -0
  144. data/lib/bundler/vendor/thor/line_editor.rb +17 -0
  145. data/lib/bundler/vendor/thor/line_editor/basic.rb +35 -0
  146. data/lib/bundler/vendor/thor/line_editor/readline.rb +88 -0
  147. data/lib/bundler/vendor/thor/parser.rb +4 -0
  148. data/lib/bundler/vendor/thor/parser/argument.rb +73 -0
  149. data/lib/bundler/vendor/thor/parser/arguments.rb +175 -0
  150. data/lib/bundler/vendor/thor/parser/option.rb +121 -0
  151. data/lib/bundler/vendor/thor/parser/options.rb +218 -0
  152. data/lib/bundler/vendor/thor/rake_compat.rb +71 -0
  153. data/lib/bundler/vendor/thor/runner.rb +322 -0
  154. data/lib/bundler/vendor/thor/shell.rb +81 -0
  155. data/lib/bundler/vendor/thor/shell/basic.rb +420 -0
  156. data/lib/bundler/vendor/thor/shell/color.rb +149 -0
  157. data/lib/bundler/vendor/thor/shell/html.rb +126 -0
  158. data/lib/bundler/vendor/thor/util.rb +267 -0
  159. data/lib/bundler/vendor/thor/version.rb +3 -0
  160. data/lib/bundler/vendored_persistent.rb +3 -0
  161. data/lib/bundler/vendored_thor.rb +8 -0
  162. data/lib/bundler/version.rb +6 -0
  163. data/lib/bundler/vlad.rb +11 -0
  164. data/man/bundle-config.ronn +155 -0
  165. data/man/bundle-exec.ronn +136 -0
  166. data/man/bundle-install.ronn +372 -0
  167. data/man/bundle-package.ronn +59 -0
  168. data/man/bundle-platform.ronn +42 -0
  169. data/man/bundle-update.ronn +179 -0
  170. data/man/bundle.ronn +92 -0
  171. data/man/gemfile.5.ronn +369 -0
  172. data/man/index.txt +7 -0
  173. data/spec/bundler/bundler_spec.rb +74 -0
  174. data/spec/bundler/cli_spec.rb +16 -0
  175. data/spec/bundler/definition_spec.rb +21 -0
  176. data/spec/bundler/dsl_spec.rb +108 -0
  177. data/spec/bundler/friendly_errors_spec.rb +13 -0
  178. data/spec/bundler/gem_helper_spec.rb +225 -0
  179. data/spec/bundler/psyched_yaml_spec.rb +8 -0
  180. data/spec/bundler/retry_spec.rb +59 -0
  181. data/spec/bundler/settings_spec.rb +13 -0
  182. data/spec/bundler/source_spec.rb +25 -0
  183. data/spec/cache/gems_spec.rb +316 -0
  184. data/spec/cache/git_spec.rb +188 -0
  185. data/spec/cache/path_spec.rb +121 -0
  186. data/spec/cache/platform_spec.rb +53 -0
  187. data/spec/cache/svn_spec.rb +82 -0
  188. data/spec/commands/binstubs_spec.rb +246 -0
  189. data/spec/commands/check_spec.rb +278 -0
  190. data/spec/commands/clean_spec.rb +652 -0
  191. data/spec/commands/config_spec.rb +227 -0
  192. data/spec/commands/console_spec.rb +102 -0
  193. data/spec/commands/exec_spec.rb +367 -0
  194. data/spec/commands/help_spec.rb +39 -0
  195. data/spec/commands/init_spec.rb +39 -0
  196. data/spec/commands/inject_spec.rb +78 -0
  197. data/spec/commands/licenses_spec.rb +31 -0
  198. data/spec/commands/newgem_spec.rb +451 -0
  199. data/spec/commands/open_spec.rb +80 -0
  200. data/spec/commands/outdated_spec.rb +168 -0
  201. data/spec/commands/package_spec.rb +128 -0
  202. data/spec/commands/show_spec.rb +152 -0
  203. data/spec/install/binstubs_spec.rb +24 -0
  204. data/spec/install/bundler_spec.rb +146 -0
  205. data/spec/install/deploy_spec.rb +237 -0
  206. data/spec/install/gemfile/gemspec_spec.rb +170 -0
  207. data/spec/install/gemfile/git_spec.rb +967 -0
  208. data/spec/install/gemfile/path_spec.rb +480 -0
  209. data/spec/install/gemfile/svn_spec.rb +582 -0
  210. data/spec/install/gemfile_spec.rb +44 -0
  211. data/spec/install/gems/c_ext_spec.rb +48 -0
  212. data/spec/install/gems/dependency_api_spec.rb +580 -0
  213. data/spec/install/gems/env_spec.rb +107 -0
  214. data/spec/install/gems/flex_spec.rb +314 -0
  215. data/spec/install/gems/groups_spec.rb +308 -0
  216. data/spec/install/gems/mirror_spec.rb +39 -0
  217. data/spec/install/gems/platform_spec.rb +221 -0
  218. data/spec/install/gems/post_install_spec.rb +121 -0
  219. data/spec/install/gems/resolving_spec.rb +124 -0
  220. data/spec/install/gems/simple_case_spec.rb +386 -0
  221. data/spec/install/gems/standalone_spec.rb +260 -0
  222. data/spec/install/gems/sudo_spec.rb +136 -0
  223. data/spec/install/gems/win32_spec.rb +26 -0
  224. data/spec/install/gemspecs_spec.rb +50 -0
  225. data/spec/install/path_spec.rb +150 -0
  226. data/spec/install/post_bundle_message_spec.rb +142 -0
  227. data/spec/install/prereleases_spec.rb +43 -0
  228. data/spec/install/security_policy_spec.rb +77 -0
  229. data/spec/install/upgrade_spec.rb +26 -0
  230. data/spec/lock/git_spec.rb +34 -0
  231. data/spec/lock/lockfile_spec.rb +912 -0
  232. data/spec/lock/svn_spec.rb +35 -0
  233. data/spec/other/bundle_ruby_spec.rb +142 -0
  234. data/spec/other/cli_dispatch_spec.rb +21 -0
  235. data/spec/other/ext_spec.rb +60 -0
  236. data/spec/other/platform_spec.rb +1285 -0
  237. data/spec/other/ssl_cert_spec.rb +10 -0
  238. data/spec/quality_spec.rb +88 -0
  239. data/spec/realworld/dependency_api_spec.rb +60 -0
  240. data/spec/realworld/edgecases_spec.rb +177 -0
  241. data/spec/realworld/parallel_spec.rb +69 -0
  242. data/spec/resolver/basic_spec.rb +66 -0
  243. data/spec/resolver/platform_spec.rb +88 -0
  244. data/spec/runtime/executable_spec.rb +149 -0
  245. data/spec/runtime/load_spec.rb +107 -0
  246. data/spec/runtime/platform_spec.rb +90 -0
  247. data/spec/runtime/require_spec.rb +332 -0
  248. data/spec/runtime/setup_spec.rb +856 -0
  249. data/spec/runtime/with_clean_env_spec.rb +91 -0
  250. data/spec/spec_helper.rb +114 -0
  251. data/spec/support/artifice/endopint_marshal_fail_basic_authentication.rb +13 -0
  252. data/spec/support/artifice/endpoint.rb +71 -0
  253. data/spec/support/artifice/endpoint_500.rb +37 -0
  254. data/spec/support/artifice/endpoint_api_missing.rb +16 -0
  255. data/spec/support/artifice/endpoint_basic_authentication.rb +13 -0
  256. data/spec/support/artifice/endpoint_creds_diff_host.rb +38 -0
  257. data/spec/support/artifice/endpoint_extra.rb +31 -0
  258. data/spec/support/artifice/endpoint_extra_missing.rb +15 -0
  259. data/spec/support/artifice/endpoint_fallback.rb +17 -0
  260. data/spec/support/artifice/endpoint_host_redirect.rb +15 -0
  261. data/spec/support/artifice/endpoint_marshal_fail.rb +11 -0
  262. data/spec/support/artifice/endpoint_redirect.rb +15 -0
  263. data/spec/support/artifice/endpoint_strict_basic_authentication.rb +18 -0
  264. data/spec/support/artifice/endpoint_timeout.rb +13 -0
  265. data/spec/support/builders.rb +748 -0
  266. data/spec/support/fakeweb/rack-1.0.0.marshal +2 -0
  267. data/spec/support/fakeweb/windows.rb +23 -0
  268. data/spec/support/hax.rb +22 -0
  269. data/spec/support/helpers.rb +348 -0
  270. data/spec/support/indexes.rb +280 -0
  271. data/spec/support/less_than_proc.rb +14 -0
  272. data/spec/support/matchers.rb +77 -0
  273. data/spec/support/path.rb +81 -0
  274. data/spec/support/permissions.rb +10 -0
  275. data/spec/support/platforms.rb +94 -0
  276. data/spec/support/ruby_ext.rb +20 -0
  277. data/spec/support/rubygems_ext.rb +39 -0
  278. data/spec/support/streams.rb +13 -0
  279. data/spec/support/sudo.rb +16 -0
  280. data/spec/update/gems_spec.rb +201 -0
  281. data/spec/update/git_spec.rb +236 -0
  282. data/spec/update/source_spec.rb +63 -0
  283. data/spec/update/svn_spec.rb +100 -0
  284. metadata +486 -0
@@ -0,0 +1,76 @@
1
+ module Bundler
2
+ # used for Creating Specifications from the Gemcutter Endpoint
3
+ class EndpointSpecification < Gem::Specification
4
+ include MatchPlatform
5
+
6
+ attr_reader :name, :version, :platform, :dependencies
7
+ attr_accessor :source, :source_uri
8
+
9
+ def initialize(name, version, platform, dependencies)
10
+ @name = name
11
+ @version = version
12
+ @platform = platform
13
+ @dependencies = dependencies
14
+ end
15
+
16
+ def fetch_platform
17
+ @platform
18
+ end
19
+
20
+ # needed for standalone, load required_paths from local gemspec
21
+ # after the gem is installed
22
+ def require_paths
23
+ if @remote_specification
24
+ @remote_specification.require_paths
25
+ elsif _local_specification
26
+ _local_specification.require_paths
27
+ else
28
+ super
29
+ end
30
+ end
31
+
32
+ # needed for binstubs
33
+ def executables
34
+ if @remote_specification
35
+ @remote_specification.executables
36
+ elsif _local_specification
37
+ _local_specification.executables
38
+ else
39
+ super
40
+ end
41
+ end
42
+
43
+ # needed for bundle clean
44
+ def bindir
45
+ if @remote_specification
46
+ @remote_specification.bindir
47
+ elsif _local_specification
48
+ _local_specification.bindir
49
+ else
50
+ super
51
+ end
52
+ end
53
+
54
+ # needed for post_install_messages during install
55
+ def post_install_message
56
+ if @remote_specification
57
+ @remote_specification.post_install_message
58
+ elsif _local_specification
59
+ _local_specification.post_install_message
60
+ end
61
+ end
62
+
63
+ def _local_specification
64
+ eval(File.read(local_specification_path)) if @loaded_from && File.exist?(local_specification_path)
65
+ end
66
+
67
+ def __swap__(spec)
68
+ @remote_specification = spec
69
+ end
70
+
71
+ private
72
+ def local_specification_path
73
+ "#{base_dir}/specifications/#{full_name}.gemspec"
74
+ end
75
+ end
76
+ end
@@ -0,0 +1,56 @@
1
+ module Bundler
2
+ class Env
3
+
4
+ def write(io)
5
+ io.write(report)
6
+ end
7
+
8
+ def report
9
+ out = "Bundler #{Bundler::VERSION}\n"
10
+
11
+ out << "Ruby #{RUBY_VERSION} (#{RUBY_RELEASE_DATE}"
12
+ out << " patchlevel #{RUBY_PATCHLEVEL}" if defined? RUBY_PATCHLEVEL
13
+ out << ") [#{RUBY_PLATFORM}]\n"
14
+
15
+ out << "Rubygems #{Gem::VERSION}\n"
16
+
17
+ out << "rvm #{ENV['rvm_version']}\n" if ENV['rvm_version']
18
+
19
+ out << "GEM_HOME #{ENV['GEM_HOME']}\n"
20
+
21
+ out << "GEM_PATH #{ENV['GEM_PATH']}\n" unless ENV['GEM_PATH'] == ENV['GEM_HOME']
22
+
23
+ %w(rubygems-bundler open_gem).each do |name|
24
+ specs = Gem::Specification.find_all{|s| s.name == name }
25
+ out << "#{name} (#{specs.map(&:version).join(',')})\n" unless specs.empty?
26
+ end
27
+
28
+ out << "\nBundler settings\n" unless Bundler.settings.all.empty?
29
+ Bundler.settings.all.each do |setting|
30
+ out << " #{setting}\n"
31
+ Bundler.settings.pretty_values_for(setting).each do |line|
32
+ out << " " << line << "\n"
33
+ end
34
+ end
35
+
36
+ out << "\n\n" << "Gemfile\n"
37
+ out << read_file("Gemfile") << "\n"
38
+
39
+ out << "\n\n" << "Gemfile.lock\n"
40
+ out << read_file("Gemfile.lock") << "\n"
41
+
42
+ out
43
+ end
44
+
45
+ private
46
+
47
+ def read_file(filename)
48
+ File.read(filename).strip
49
+ rescue Errno::ENOENT
50
+ "<No #{filename} found>"
51
+ rescue => e
52
+ "#{e.class}: #{e.message}"
53
+ end
54
+
55
+ end
56
+ end
@@ -0,0 +1,42 @@
1
+ module Bundler
2
+ class Environment
3
+ attr_reader :root
4
+
5
+ def initialize(root, definition)
6
+ @root = root
7
+ @definition = definition
8
+
9
+ env_file = Bundler.app_config_path.join('environment.rb')
10
+ env_file.rmtree if env_file.exist?
11
+ end
12
+
13
+ def inspect
14
+ @definition.to_lock.inspect
15
+ end
16
+
17
+ def requested_specs
18
+ @definition.requested_specs
19
+ end
20
+
21
+ def specs
22
+ @definition.specs
23
+ end
24
+
25
+ def dependencies
26
+ @definition.dependencies
27
+ end
28
+
29
+ def current_dependencies
30
+ @definition.current_dependencies
31
+ end
32
+
33
+ def lock
34
+ @definition.lock(Bundler.default_lockfile)
35
+ end
36
+
37
+ def update(*gems)
38
+ # Nothing
39
+ end
40
+
41
+ end
42
+ end
@@ -0,0 +1,396 @@
1
+ require 'bundler/vendored_persistent'
2
+ require 'securerandom'
3
+ require 'cgi'
4
+
5
+ module Bundler
6
+
7
+ # Handles all the fetching with the rubygems server
8
+ class Fetcher
9
+ # This error is raised if the API returns a 413 (only printed in verbose)
10
+ class FallbackError < HTTPError; end
11
+ # This is the error raised if OpenSSL fails the cert verification
12
+ class CertificateFailureError < HTTPError
13
+ def initialize(remote_uri)
14
+ super "Could not verify the SSL certificate for #{remote_uri}.\nThere" \
15
+ " is a chance you are experiencing a man-in-the-middle attack, but" \
16
+ " most likely your system doesn't have the CA certificates needed" \
17
+ " for verification. For information about OpenSSL certificates, see" \
18
+ " bit.ly/ruby-ssl. To connect without using SSL, edit your Gemfile" \
19
+ " sources and change 'https' to 'http'."
20
+ end
21
+ end
22
+ # This is the error raised when a source is HTTPS and OpenSSL didn't load
23
+ class SSLError < HTTPError
24
+ def initialize(msg = nil)
25
+ super msg || "Could not load OpenSSL.\n" \
26
+ "You must recompile Ruby with OpenSSL support or change the sources in your " \
27
+ "Gemfile from 'https' to 'http'. Instructions for compiling with OpenSSL " \
28
+ "using RVM are available at rvm.io/packages/openssl."
29
+ end
30
+ end
31
+ # This error is raised if HTTP authentication is required, but not provided.
32
+ class AuthenticationRequiredError < HTTPError
33
+ def initialize(remote_uri)
34
+ super "Authentication is required for #{remote_uri}.\n" \
35
+ "Please supply credentials for this source. You can do this by running:\n" \
36
+ " bundle config #{remote_uri} username:password"
37
+ end
38
+ end
39
+ # This error is raised if HTTP authentication is provided, but incorrect.
40
+ class BadAuthenticationError < HTTPError
41
+ def initialize(remote_uri)
42
+ super "Bad username or password for #{remote_uri}.\n" \
43
+ "Please double-check your credentials and correct them."
44
+ end
45
+ end
46
+
47
+ # Exceptions classes that should bypass retry attempts. If your password didn't work the
48
+ # first time, it's not going to the third time.
49
+ AUTH_ERRORS = [AuthenticationRequiredError, BadAuthenticationError]
50
+
51
+ class << self
52
+ attr_accessor :disable_endpoint, :api_timeout, :redirect_limit, :max_retries
53
+
54
+ def download_gem_from_uri(spec, uri)
55
+ spec.fetch_platform
56
+
57
+ download_path = Bundler.requires_sudo? ? Bundler.tmp(spec.full_name) : Bundler.rubygems.gem_dir
58
+ gem_path = "#{Bundler.rubygems.gem_dir}/cache/#{spec.full_name}.gem"
59
+
60
+ FileUtils.mkdir_p("#{download_path}/cache")
61
+ Bundler.rubygems.download_gem(spec, uri, download_path)
62
+
63
+ if Bundler.requires_sudo?
64
+ Bundler.mkdir_p "#{Bundler.rubygems.gem_dir}/cache"
65
+ Bundler.sudo "mv #{Bundler.tmp(spec.full_name)}/cache/#{spec.full_name}.gem #{gem_path}"
66
+ end
67
+
68
+ gem_path
69
+ end
70
+
71
+ def user_agent
72
+ @user_agent ||= begin
73
+ ruby = Bundler.ruby_version
74
+
75
+ agent = "bundler/#{Bundler::VERSION}"
76
+ agent += " rubygems/#{Gem::VERSION}"
77
+ agent += " ruby/#{ruby.version}"
78
+ agent += " (#{ruby.host})"
79
+ agent += " command/#{ARGV.first}"
80
+
81
+ if ruby.engine != "ruby"
82
+ # engine_version raises on unknown engines
83
+ engine_version = ruby.engine_version rescue "???"
84
+ agent += " #{ruby.engine}/#{engine_version}"
85
+ end
86
+ # add a random ID so we can consolidate runs server-side
87
+ agent << " " << SecureRandom.hex(8)
88
+ end
89
+ end
90
+
91
+ end
92
+
93
+ def initialize(remote_uri)
94
+
95
+ # How many redirects to allew in one request
96
+ @redirect_limit = 5
97
+ # How long to wait for each gemcutter API call
98
+ @api_timeout = 10
99
+ # How many retries for the gemcutter API call
100
+ @max_retries = 3
101
+
102
+ @remote_uri = Bundler::Source.mirror_for(remote_uri)
103
+ @public_uri = @remote_uri.dup
104
+ @public_uri.user, @public_uri.password = nil, nil # don't print these
105
+
106
+ Socket.do_not_reverse_lookup = true
107
+ end
108
+
109
+ def connection
110
+ return @connection if @connection
111
+
112
+ needs_ssl = @remote_uri.scheme == "https" ||
113
+ Bundler.settings[:ssl_verify_mode] ||
114
+ Bundler.settings[:ssl_client_cert]
115
+ raise SSLError if needs_ssl && !defined?(OpenSSL)
116
+
117
+ @connection = Net::HTTP::Persistent.new 'bundler', :ENV
118
+
119
+ if @remote_uri.scheme == "https"
120
+ @connection.verify_mode = (Bundler.settings[:ssl_verify_mode] ||
121
+ OpenSSL::SSL::VERIFY_PEER)
122
+ @connection.cert_store = bundler_cert_store
123
+ end
124
+
125
+ if Bundler.settings[:ssl_client_cert]
126
+ pem = File.read(Bundler.settings[:ssl_client_cert])
127
+ @connection.cert = OpenSSL::X509::Certificate.new(pem)
128
+ @connection.key = OpenSSL::PKey::RSA.new(pem)
129
+ end
130
+
131
+ @connection.read_timeout = @api_timeout
132
+ @connection.override_headers["User-Agent"] = self.class.user_agent
133
+
134
+ @connection
135
+ end
136
+
137
+ def uri
138
+ @public_uri
139
+ end
140
+
141
+ # fetch a gem specification
142
+ def fetch_spec(spec)
143
+ spec = spec - [nil, 'ruby', '']
144
+ spec_file_name = "#{spec.join '-'}.gemspec"
145
+
146
+ uri = URI.parse("#{@remote_uri}#{Gem::MARSHAL_SPEC_DIR}#{spec_file_name}.rz")
147
+ if uri.scheme == 'file'
148
+ Bundler.load_marshal Gem.inflate(Gem.read_binary(uri.path))
149
+ elsif cached_spec_path = gemspec_cached_path(spec_file_name)
150
+ Bundler.load_gemspec(cached_spec_path)
151
+ else
152
+ Bundler.load_marshal Gem.inflate(fetch(uri))
153
+ end
154
+ rescue MarshalError
155
+ raise HTTPError, "Gemspec #{spec} contained invalid data.\n" \
156
+ "Your network or your gem server is probably having issues right now."
157
+ end
158
+
159
+ # cached gem specification path, if one exists
160
+ def gemspec_cached_path spec_file_name
161
+ paths = Bundler.rubygems.spec_cache_dirs.map { |dir| File.join(dir, spec_file_name) }
162
+ paths = paths.select {|path| File.file? path }
163
+ paths.first
164
+ end
165
+
166
+ # return the specs in the bundler format as an index
167
+ def specs(gem_names, source)
168
+ index = Index.new
169
+
170
+ if gem_names && use_api
171
+ specs = fetch_remote_specs(gem_names)
172
+ end
173
+
174
+ if specs.nil?
175
+ # API errors mean we should treat this as a non-API source
176
+ @use_api = false
177
+
178
+ specs = Bundler::Retry.new("source fetch", AUTH_ERRORS).attempts do
179
+ fetch_all_remote_specs
180
+ end
181
+ end
182
+
183
+ specs[@remote_uri].each do |name, version, platform, dependencies|
184
+ next if name == 'bundler'
185
+ spec = nil
186
+ if dependencies
187
+ spec = EndpointSpecification.new(name, version, platform, dependencies)
188
+ else
189
+ spec = RemoteSpecification.new(name, version, platform, self)
190
+ end
191
+ spec.source = source
192
+ spec.source_uri = @remote_uri
193
+ index << spec
194
+ end
195
+
196
+ index
197
+ rescue CertificateFailureError => e
198
+ Bundler.ui.info "" if gem_names && use_api # newline after dots
199
+ raise e
200
+ end
201
+
202
+ # fetch index
203
+ def fetch_remote_specs(gem_names, full_dependency_list = [], last_spec_list = [])
204
+ query_list = gem_names - full_dependency_list
205
+
206
+ # only display the message on the first run
207
+ if Bundler.ui.debug?
208
+ Bundler.ui.debug "Query List: #{query_list.inspect}"
209
+ else
210
+ Bundler.ui.info ".", false
211
+ end
212
+
213
+ return {@remote_uri => last_spec_list} if query_list.empty?
214
+
215
+ remote_specs = Bundler::Retry.new("dependency api", AUTH_ERRORS).attempts do
216
+ fetch_dependency_remote_specs(query_list)
217
+ end
218
+
219
+ spec_list, deps_list = remote_specs
220
+ returned_gems = spec_list.map {|spec| spec.first }.uniq
221
+ fetch_remote_specs(deps_list, full_dependency_list + returned_gems, spec_list + last_spec_list)
222
+ rescue HTTPError, MarshalError, GemspecError
223
+ Bundler.ui.info "" unless Bundler.ui.debug? # new line now that the dots are over
224
+ Bundler.ui.debug "could not fetch from the dependency API, trying the full index"
225
+ @use_api = false
226
+ return nil
227
+ end
228
+
229
+ def use_api
230
+ return @use_api if defined?(@use_api)
231
+
232
+ if @remote_uri.scheme == "file" || Bundler::Fetcher.disable_endpoint
233
+ @use_api = false
234
+ elsif fetch(dependency_api_uri)
235
+ @use_api = true
236
+ end
237
+ rescue HTTPError
238
+ @use_api = false
239
+ end
240
+
241
+ def inspect
242
+ "#<#{self.class}:0x#{object_id} uri=#{uri}>"
243
+ end
244
+
245
+ private
246
+
247
+ HTTP_ERRORS = [
248
+ Timeout::Error, EOFError, SocketError, Errno::ENETDOWN,
249
+ Errno::EINVAL, Errno::ECONNRESET, Errno::ETIMEDOUT, Errno::EAGAIN,
250
+ Net::HTTPBadResponse, Net::HTTPHeaderSyntaxError, Net::ProtocolError,
251
+ Net::HTTP::Persistent::Error
252
+ ]
253
+
254
+ def fetch(uri, counter = 0)
255
+ raise HTTPError, "Too many redirects" if counter >= @redirect_limit
256
+
257
+ response = request(uri)
258
+ case response
259
+ when Net::HTTPRedirection
260
+ Bundler.ui.debug("HTTP Redirection")
261
+ new_uri = URI.parse(response["location"])
262
+ if new_uri.host == uri.host
263
+ new_uri.user = uri.user
264
+ new_uri.password = uri.password
265
+ end
266
+ fetch(new_uri, counter + 1)
267
+ when Net::HTTPSuccess
268
+ Bundler.ui.debug("HTTP Success")
269
+ response.body
270
+ when Net::HTTPRequestEntityTooLarge
271
+ raise FallbackError, response.body
272
+ else
273
+ raise HTTPError, "#{response.class}: #{response.body}"
274
+ end
275
+ end
276
+
277
+ def request(uri)
278
+ Bundler.ui.debug "Fetching from: #{uri}"
279
+ req = Net::HTTP::Get.new uri.request_uri
280
+ if uri.user
281
+ user = CGI.unescape(uri.user)
282
+ password = uri.password ? CGI.unescape(uri.password) : nil
283
+ req.basic_auth(user, password)
284
+ end
285
+ connection.request(uri, req)
286
+ rescue Net::HTTPUnauthorized, Net::HTTPForbidden
287
+ retry_with_auth { request(uri) }
288
+ rescue OpenSSL::SSL::SSLError
289
+ raise CertificateFailureError.new(uri)
290
+ rescue *HTTP_ERRORS
291
+ raise HTTPError, "Network error while fetching #{uri}"
292
+ end
293
+
294
+ def dependency_api_uri(gem_names = [])
295
+ url = "#{redirected_uri}api/v1/dependencies"
296
+ url << "?gems=#{URI.encode(gem_names.join(","))}" if gem_names.any?
297
+ URI.parse(url)
298
+ end
299
+
300
+ # fetch from Gemcutter Dependency Endpoint API
301
+ def fetch_dependency_remote_specs(gem_names)
302
+ Bundler.ui.debug "Query Gemcutter Dependency Endpoint API: #{gem_names.join(',')}"
303
+ marshalled_deps = fetch dependency_api_uri(gem_names)
304
+ gem_list = Bundler.load_marshal(marshalled_deps)
305
+ deps_list = []
306
+
307
+ spec_list = gem_list.map do |s|
308
+ dependencies = s[:dependencies].map do |name, requirement|
309
+ dep = well_formed_dependency(name, requirement.split(", "))
310
+ deps_list << dep.name
311
+ dep
312
+ end
313
+
314
+ [s[:name], Gem::Version.new(s[:number]), s[:platform], dependencies]
315
+ end
316
+
317
+ [spec_list, deps_list.uniq]
318
+ end
319
+
320
+ # fetch from modern index: specs.4.8.gz
321
+ def fetch_all_remote_specs
322
+ Bundler.rubygems.sources = ["#{@remote_uri}"]
323
+ Bundler.rubygems.fetch_all_remote_specs
324
+ rescue Gem::RemoteFetcher::FetchError, OpenSSL::SSL::SSLError => e
325
+ case e.message
326
+ when /certificate verify failed/
327
+ raise CertificateFailureError.new(uri)
328
+ when /401|403/
329
+ # Gemfury uses a 403 for unauthenticated requests instead of a 401, so retry auth
330
+ # on both.
331
+ retry_with_auth { fetch_all_remote_specs }
332
+ else
333
+ Bundler.ui.trace e
334
+ raise HTTPError, "Could not fetch specs from #{uri}"
335
+ end
336
+ end
337
+
338
+ def well_formed_dependency(name, *requirements)
339
+ Gem::Dependency.new(name, *requirements)
340
+ rescue ArgumentError => e
341
+ illformed = 'Ill-formed requirement ["#<YAML::Syck::DefaultKey'
342
+ raise e unless e.message.include?(illformed)
343
+ puts # we shouldn't print the error message on the "fetching info" status line
344
+ raise GemspecError,
345
+ "Unfortunately, the gem #{s[:name]} (#{s[:number]}) has an invalid " \
346
+ "gemspec. \nPlease ask the gem author to yank the bad version to fix " \
347
+ "this issue. For more information, see http://bit.ly/syck-defaultkey."
348
+ end
349
+
350
+ def bundler_cert_store
351
+ store = OpenSSL::X509::Store.new
352
+ if Bundler.settings[:ssl_ca_cert]
353
+ if File.directory? Bundler.settings[:ssl_ca_cert]
354
+ store.add_path Bundler.settings[:ssl_ca_cert]
355
+ else
356
+ store.add_file Bundler.settings[:ssl_ca_cert]
357
+ end
358
+ else
359
+ store.set_default_paths
360
+ certs = File.expand_path("../ssl_certs/*.pem", __FILE__)
361
+ Dir.glob(certs).each { |c| store.add_file c }
362
+ end
363
+ store
364
+ end
365
+
366
+ # Attempt to retry with HTTP authentication, if it's appropriate to do so. Yields to a block;
367
+ # the caller should use this to re-attempt the failing request with the altered `@remote_uri`.
368
+ def retry_with_auth
369
+ # Authentication has already been attempted and failed.
370
+ raise BadAuthenticationError.new(uri) if @remote_uri.user
371
+
372
+ auth = Bundler.settings[@remote_uri.to_s]
373
+
374
+ # Authentication isn't provided at all, by "bundle config" or in the URI.
375
+ raise AuthenticationRequiredError.new(uri) if auth.nil?
376
+
377
+ @remote_uri.user, @remote_uri.password = *auth.split(":", 2)
378
+ yield
379
+ end
380
+
381
+ private
382
+ def redirected_uri
383
+ return bundler_uri if rubygems?
384
+ return @remote_uri
385
+ end
386
+
387
+ def rubygems?
388
+ @remote_uri.host == "rubygems.org"
389
+ end
390
+
391
+ def bundler_uri
392
+ URI.parse("#{@remote_uri.scheme}://bundler.#{@remote_uri.host}/")
393
+ end
394
+
395
+ end
396
+ end