require File.expand_path(File.dirname(__FILE__) + '/spec_helper') describe Thor::Actions do def runner(options={}) @runner ||= MyCounter.new([1], options, { :destination_root => destination_root }) end def action(*args, &block) capture(:stdout){ runner.send(*args, &block) } end def file File.join(destination_root, "foo") end describe "on include" do it "adds runtime options to the base class" do MyCounter.class_options.keys.must include(:pretend) MyCounter.class_options.keys.must include(:force) MyCounter.class_options.keys.must include(:quiet) MyCounter.class_options.keys.must include(:skip) end end describe "#initialize" do it "has default behavior invoke" do runner.behavior.must == :invoke end it "can have behavior revoke" do MyCounter.new([1], {}, :behavior => :revoke).behavior.must == :revoke end it "when behavior is set to force, overwrite options" do runner = MyCounter.new([1], { :force => false, :skip => true }, :behavior => :force) runner.behavior.must == :invoke runner.options.force.must be_true runner.options.skip.must_not be_true end it "when behavior is set to skip, overwrite options" do runner = MyCounter.new([1], ["--force"], :behavior => :skip) runner.behavior.must == :invoke runner.options.force.must_not be_true runner.options.skip.must be_true end end describe "accessors" do describe "#destination_root=" do it "gets the current directory and expands the path to set the root" do base = MyCounter.new([1]) base.destination_root = "here" base.destination_root.must == File.expand_path(File.join(File.dirname(__FILE__), "..", "here")) end it "does not use the current directory if one is given" do root = File.expand_path("/") base = MyCounter.new([1]) base.destination_root = root base.destination_root.must == root end it "uses the current directory if none is given" do base = MyCounter.new([1]) base.destination_root.must == File.expand_path(File.join(File.dirname(__FILE__), "..")) end end describe "#relative_to_original_destination_root" do it "returns the path relative to the absolute root" do runner.relative_to_original_destination_root(file).must == "foo" end it "does not remove dot if required" do runner.relative_to_original_destination_root(file, false).must == "./foo" end it "always use the absolute root" do runner.inside("foo") do runner.relative_to_original_destination_root(file).must == "foo" end end describe "#source_paths_for_search" do it "add source_root to source_paths_for_search" do MyCounter.source_paths_for_search.must include(File.expand_path("fixtures", File.dirname(__FILE__))) end it "keeps only current source root in source paths" do ClearCounter.source_paths_for_search.must include(File.expand_path("fixtures/bundle", File.dirname(__FILE__))) ClearCounter.source_paths_for_search.must_not include(File.expand_path("fixtures", File.dirname(__FILE__))) end it "customized source paths should be before source roots" do ClearCounter.source_paths_for_search[0].must == File.expand_path("fixtures/doc", File.dirname(__FILE__)) ClearCounter.source_paths_for_search[1].must == File.expand_path("fixtures/bundle", File.dirname(__FILE__)) end it "keeps inherited source paths at the end" do ClearCounter.source_paths_for_search.last.must == File.expand_path("fixtures/broken", File.dirname(__FILE__)) end end end describe "#find_in_source_paths" do it "raises an error if source path is empty" do lambda { A.new.find_in_source_paths("foo") }.must raise_error(Thor::Error, /Currently you have no source paths/) end it "finds a template inside the source path" do runner.find_in_source_paths("doc").must == File.expand_path("doc", source_root) lambda { runner.find_in_source_paths("README") }.must raise_error new_path = File.join(source_root, "doc") runner.instance_variable_set(:@source_paths, nil) runner.source_paths.unshift(new_path) runner.find_in_source_paths("README").must == File.expand_path("README", new_path) end end end describe "#inside" do it "executes the block inside the given folder" do runner.inside("foo") do Dir.pwd.must == file end end it "changes the base root" do runner.inside("foo") do runner.destination_root.must == file end end it "creates the directory if it does not exist" do runner.inside("foo") do File.exists?(file).must be_true end end describe "when verbose" do it "logs status" do capture(:stdout) do runner.inside("foo", :verbose => true) {} end.must =~ /inside foo/ end it "uses padding in next status" do capture(:stdout) do runner.inside("foo", :verbose => true) do runner.say_status :cool, :padding end end.must =~ /cool padding/ end it "removes padding after block" do capture(:stdout) do runner.inside("foo", :verbose => true) {} runner.say_status :no, :padding end.must =~ /no padding/ end end end describe "#in_root" do it "executes the block in the root folder" do runner.inside("foo") do runner.in_root { Dir.pwd.must == destination_root } end end it "changes the base root" do runner.inside("foo") do runner.in_root { runner.destination_root.must == destination_root } end end it "returns to the previous state" do runner.inside("foo") do runner.in_root { } runner.destination_root.must == file end end end describe "#apply" do before(:each) do @template = <<-TEMPLATE @foo = "FOO" say_status :cool, :padding TEMPLATE @template.stub(:read).and_return(@template) @file = '/' runner.stub(:open).and_return(@template) end it "accepts a URL as the path" do @file = "http://gist.github.com/103208.txt" runner.should_receive(:open).with(@file, "Accept" => "application/x-thor-template").and_return(@template) action(:apply, @file) end it "accepts a secure URL as the path" do @file = "https://gist.github.com/103208.txt" runner.should_receive(:open).with(@file, "Accept" => "application/x-thor-template").and_return(@template) action(:apply, @file) end it "accepts a local file path with spaces" do @file = File.expand_path("fixtures/path with spaces", File.dirname(__FILE__)) runner.should_receive(:open).with(@file).and_return(@template) action(:apply, @file) end it "opens a file and executes its content in the instance binding" do action :apply, @file runner.instance_variable_get("@foo").must == "FOO" end it "applies padding to the content inside the file" do action(:apply, @file).must =~ /cool padding/ end it "logs its status" do action(:apply, @file).must =~ / apply #{@file}\n/ end it "does not log status" do content = action(:apply, @file, :verbose => false) content.must =~ /cool padding/ content.must_not =~ /apply http/ end end describe "#run" do before(:each) do runner.should_receive(:`).with("ls") end it "executes the command given" do action :run, "ls" end it "logs status" do action(:run, "ls").must == " run ls from \".\"\n" end it "does not log status if required" do action(:run, "ls", :verbose => false).must be_empty end it "accepts a color as status" do runner.shell.should_receive(:say_status).with(:run, 'ls from "."', :yellow) action :run, "ls", :verbose => :yellow end end describe "#run_ruby_script" do before(:each) do Thor::Util.stub!(:ruby_command).and_return("/opt/jruby") runner.should_receive(:`).with("/opt/jruby script.rb") end it "executes the ruby script" do action :run_ruby_script, "script.rb" end it "logs status" do action(:run_ruby_script, "script.rb").must == " run jruby script.rb from \".\"\n" end it "does not log status if required" do action(:run_ruby_script, "script.rb", :verbose => false).must be_empty end end describe "#thor" do it "executes the thor command" do runner.should_receive(:`).with("thor list") action :thor, :list, :verbose => true end it "converts extra arguments to command arguments" do runner.should_receive(:`).with("thor list foo bar") action :thor, :list, "foo", "bar" end it "converts options hash to switches" do runner.should_receive(:`).with("thor list foo bar --foo") action :thor, :list, "foo", "bar", :foo => true runner.should_receive(:`).with("thor list --foo 1 2 3") action :thor, :list, :foo => [1,2,3] end it "logs status" do runner.should_receive(:`).with("thor list") action(:thor, :list).must == " run thor list from \".\"\n" end it "does not log status if required" do runner.should_receive(:`).with("thor list --foo 1 2 3") action(:thor, :list, :foo => [1,2,3], :verbose => false).must be_empty end end end