require "spec_helper" require 'rake' require 'bundler/gem_helper' describe Bundler::GemHelper do let(:app_name) { "test" } let(:app_path) { bundled_app app_name } let(:app_gemspec_path) { app_path.join("#{app_name}.gemspec") } before(:each) do bundle "gem #{app_name}" end context "determining gemspec" do subject { Bundler::GemHelper.new(app_path) } context "fails" do it "when there is no gemspec" do FileUtils.rm app_gemspec_path expect { subject }.to raise_error(/Unable to determine name/) end it "when there are two gemspecs and the name isn't specified" do FileUtils.touch app_path.join("#{app_name}-2.gemspec") expect { subject }.to raise_error(/Unable to determine name/) end end context "interpolates the name" do it "when there is only one gemspec" do expect(subject.gemspec.name).to eq(app_name) end it "for a hidden gemspec" do FileUtils.mv app_gemspec_path, app_path.join(".gemspec") expect(subject.gemspec.name).to eq(app_name) end end it "handles namespaces and converts them to CamelCase" do bundle "gem #{app_name}-foo_bar" app_path = bundled_app "#{app_name}-foo_bar" lib = app_path.join("lib/#{app_name}/foo_bar.rb").read expect(lib).to include("module #{app_name.capitalize}") expect(lib).to include("module FooBar") end end context "gem management" do def mock_confirm_message(message) expect(Bundler.ui).to receive(:confirm).with(message) end def mock_build_message(name, version) message = "#{name} #{version} built to pkg/#{name}-#{version}.gem." mock_confirm_message message end subject! { Bundler::GemHelper.new(app_path) } let(:app_version) { "0.0.1" } let(:app_gem_dir) { app_path.join("pkg") } let(:app_gem_path) { app_gem_dir.join("#{app_name}-#{app_version}.gem") } let(:app_gemspec_content) { File.read(app_gemspec_path) } before(:each) do content = app_gemspec_content.gsub("TODO: ", "") File.open(app_gemspec_path, "w") { |file| file << content } end it "uses a shell UI for output" do expect(Bundler.ui).to be_a(Bundler::UI::Shell) end describe "#install" do let!(:rake_application) { Rake.application } before(:each) do Rake.application = Rake::Application.new end after(:each) do Rake.application = rake_application end context "defines Rake tasks" do let(:task_names) { %w[build install release] } context "before installation" do it "raises an error with appropriate message" do task_names.each do |name| expect { Rake.application[name] }. to raise_error("Don't know how to build task '#{name}'") end end end context "after installation" do before do subject.install end it "adds Rake tasks successfully" do task_names.each do |name| expect { Rake.application[name] }.not_to raise_error expect(Rake.application[name]).to be_instance_of Rake::Task end end it "provides a way to access the gemspec object" do expect(subject.gemspec.name).to eq(app_name) end end end end describe "#build_gem" do context "when build failed" do it "raises an error with appropriate message" do # break the gemspec by adding back the TODOs File.open(app_gemspec_path, "w"){ |file| file << app_gemspec_content } expect { subject.build_gem }.to raise_error(/TODO/) end end context "when build was successful" do it "creates .gem file" do mock_build_message app_name, app_version subject.build_gem expect(app_gem_path).to exist end end end describe "#install_gem" do context "when installation was successful" do it "gem is installed" do mock_build_message app_name, app_version mock_confirm_message "#{app_name} (#{app_version}) installed." subject.install_gem expect(app_gem_path).to exist expect(`gem list`).to include("#{app_name} (#{app_version})") end end context "when installation fails" do it "raises an error with appropriate message" do # create empty gem file in order to simulate install failure allow(subject).to receive(:build_gem) do FileUtils.mkdir_p(app_gem_dir) FileUtils.touch app_gem_path app_gem_path end expect { subject.install_gem }.to raise_error(/Couldn't install gem/) end end end describe "#release_gem" do before do Dir.chdir(app_path) do `git init` `git config user.email "you@example.com"` `git config user.name "name"` end end context "fails" do it "when there are unstaged files" do expect { subject.release_gem }. to raise_error("There are files that need to be committed first.") end it "when there are uncommitted files" do Dir.chdir(app_path) { `git add .` } expect { subject.release_gem }. to raise_error("There are files that need to be committed first.") end it "when there is no git remote" do # silence messages allow(Bundler.ui).to receive(:confirm) allow(Bundler.ui).to receive(:error) Dir.chdir(app_path) { `git commit -a -m "initial commit"` } expect { subject.release_gem }.to raise_error end end context "succeeds" do before do Dir.chdir(gem_repo1) { `git init --bare` } Dir.chdir(app_path) do `git remote add origin file://#{gem_repo1}` `git commit -a -m "initial commit"` end end it "on releasing" do mock_build_message app_name, app_version mock_confirm_message "Tagged v#{app_version}." mock_confirm_message "Pushed git commits and tags." expect(subject).to receive(:rubygem_push).with(app_gem_path.to_s) Dir.chdir(app_path) { sys_exec("git push origin master", true) } subject.release_gem end it "even if tag already exists" do mock_build_message app_name, app_version mock_confirm_message "Tag v#{app_version} has already been created." expect(subject).to receive(:rubygem_push).with(app_gem_path.to_s) Dir.chdir(app_path) do `git tag -a -m \"Version #{app_version}\" v#{app_version}` end subject.release_gem end end end end end